最近碰到了个问题,在正在下载页面,点击取消一个下载任务的时候,总是出现点击事件的混乱,要么是点击取消按钮(一个imageview)没反应,要么是选中了整个item。
通过分析发现,暂停下载的话,是不存在这个问题的,因此排除了imageview点击区域不够大的原因,也排除了父控件和子控件的focus争抢的问题。
最终解决问题是刷到了这个:
http://blog.csdn.net/guolin_blog/article/details/45586553
这篇文章让我对listview的原理有了更加充分的认识,“ListView中的子View其实来来回回就那么几个,移出屏幕的子View会很快被移入屏幕的数据重新利用起来”,也就是说,listview的子view是被不断复用的,相同的view,设上不同的数据,出来的就是不同的行。
基于此,我猜想,当我点击的时候,被点击的view可能已经被刷新了(不是当时的那个view了)。仔细排查代码发现,每次来一条下载进度更新的消息,都使用NotifyDataSetChanged()刷新了整个列表。这样也就解释通了,为什么暂停下载时点击没有问题,而下载的时候列表点击就有问题了。
解决问题的方式:
(1)列表局部更新:每当收到一条进度更新的消息的时候,从消息中取出对应的数据,来只更新其中的一行。参考其中的解决方案一 ,使用findViewWithTag,来找到要更新的view,这更新这一行的相关view就行,避免刷新整个列表
(2)给view设tag,tag能帮忙标示一个view。在onclick事件中,通过getTag的方式,把view中的数据取出来。
在自定义Adapter的getview()函数里面,给需要和进度关联的几个控件设上tag:
public View getView(int position, View convertView, ViewGroup parent) {
if (mListView == null) {
mListView = (ListView) parent;
}
if (convertView == null) {
convertView = LayoutInflater.from(mContext).inflate(R.layout.downloading_item, null);
itemView = new ViewHolder();
itemView.mCtrl = (ImageView) convertView.findViewById(R.id.ctrl);
itemView.mProgressText = (TextView) convertView.findViewById(R.id.progress_text);
itemView.mProgressBar = (ProgressBar) convertView.findViewById(R.id.progress);
itemView.mDeleteView = convertView.findViewById(R.id.delete);
convertView.setTag(itemView);
} else {
itemView = (ViewHolder) convertView.getTag();
}
long cloudId = mSonglist.get(position).getCloudId();
String progress_text_tag = String.valueOf(cloudId).concat("_text");
String ctrl_tag = String.valueOf(cloudId).concat("_ctrl");
itemView.mProgressBar.setTag(cloudId);
itemView.mProgressText.setTag(progress_text_tag);
itemView.mCtrl.setTag(ctrl_tag);
// imageview的点击事件
itemView.mDeleteView.setOnClickListener(mOnDeleteClickListener);
itemView.mDeleteView.setTag(cloudId);
//整个item的点击事件
convertView.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View view) {
//...
}
});
return convertView;
}
在收到某条进度更新的消息后,通过mListView的findViewWithTag,找到要更新的view,设置上相应的状态信息
(注意:不要直接调用notifyDataSetChanged(),这样会刷新整个列表)
public void onEventMainThread(DownloadEvents.OnDownloadProgressChanged event) {
DownloadSong downloadSong = event.downloadSong;
long progress = downloadSong.getFileLengthDownloaded();
long total = downloadSong.getFileLength();
long cloudId = downloadSong.getCloudId();
String progress_text_tag = String.valueOf(cloudId).concat("_text");
String progress_ctrl_tag = String.valueOf(cloudId).concat("_ctrl");
ProgressBar progressBar = (ProgressBar)mListView.findViewWithTag(cloudId);
TextView progressText = (TextView)mListView.findViewWithTag(progress_text_tag);
ImageView ctrl = (ImageView)mListView.findViewWithTag(progress_ctrl_tag);
if (progressBar != null && progressText != null && ctrl != null) {
updateView(status, progress, total, progressBar, progressText, ctrl);
}
}
解决问题的过程中还参考学习了以下:
http://txlong-onz.iteye.com/blog/907186
listview 更新进度条
http://www.myexception.cn/mobile/1880588.html
http://f303153041.iteye.com/blog/1838092
android:descendantFocusability用法简析
开发中很常见的一个问题,项目中的listview不仅仅是简单的文字,常常需要自己定义listview,自己的Adapter去继承BaseAdapter,在adapter中按照需求进行编写,问题就出现了,可能会发生点击每一个item的时候没有反应,无法获取的焦点。原因多半是由于在你自己定义的Item中存在诸如ImageButton,Button,CheckBox等子控件(也可以说是Button或者Checkable的子类控件),此时这些子控件会将焦点获取到,所以常常当点击item时变化的是子控件,item本身的点击没有响应。
这时候就可以使用descendantFocusability来解决啦,API描述如下:
android:descendantFocusability
Defines the relationship between the ViewGroup and its descendants when looking for a View to take focus.
Must be one of the following constant values.
该属性是当一个为view获取焦点时,定义viewGroup和其子控件两者之间的关系。
属性的值有三种:
beforeDescendants:viewgroup会优先其子类控件而获取到焦点
afterDescendants:viewgroup只有当其子类控件不需要获取焦点时才获取焦点
blocksDescendants:viewgroup会覆盖子类控件而直接获得焦点
通常我们用到的是第三种,即在Item布局的根布局加上android:descendantFocusability=”blocksDescendants”的属性就好了,至此listview点击的灵异事件告一段落。心得:遇到不会不懂的地方除了网上查询资料之外,也可以多多去尝试每种属性的作用,多阅读官方文档(我始终觉得还是读原文的比翻译的理解的会更好)。