android中ListView点击和ImageView点击不能同时生效问题解决

最近碰到了个问题,在正在下载页面,点击取消一个下载任务的时候,总是出现点击事件的混乱,要么是点击取消按钮(一个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点击的灵异事件告一段落。心得:遇到不会不懂的地方除了网上查询资料之外,也可以多多去尝试每种属性的作用,多阅读官方文档(我始终觉得还是读原文的比翻译的理解的会更好)。

推荐阅读更多精彩内容