从Listview与Button点击事件冲突视安卓点击事件分发机制

从Listview与Button点击事件冲突看安卓点击事件分发机制

题目有点长。其实实现Listview的时候大家都可能会碰到这样的一个问题,那就是Listview的OnItemClickListener点击事件与Button(或者checkbox)的touch(或者click)事件冲突的问题。

声明一下,非常感谢郭大师的这篇blog:

http://blog.csdn.net/guolin_blog/article/details/9097463

原理参考了这篇blog,事实上也是本人功力不够不能阅读源码的缺陷啊。

下面说下自己的解决步骤:

1)首先先set一下button的焦点,button是会抢占Listview的焦点的。

replyBt.setFocusable(false)

我这样写就可以解决抢占焦点的问题了。

2)重写Button类,复写dispatchTouchEvent方法。

首先我们得要知道,点击事件首先是被Listview接收然后才传到Button的,如果中间某一个地方把点击事件消费掉了后面就再也不能接收到了。

我们需要接收到Button的click事件,那么什么时候click会被调用呢?

    public boolean dispatchTouchEvent(MotionEvent event) {  
        if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED &&  
                mOnTouchListener.onTouch(this, event)) {  
            return true;  
        }  
        return onTouchEvent(event);  
    }  

这是系统的dispatchTouchEvent源码,这个函数负责该view点击事件的分发,click事件是在onTouchEvent里面调用的,我们首先要先知道什么时候会调用dispatchTouchEvent这个函数呢?

其实只要在这个view上有一个点击操作都会触发这个函数的,这个点击操作包括:手指按下,移动手指,抬起手指

其中移动手指会多次触发这个函数。

@Override
	public boolean dispatchTouchEvent(MotionEvent event) {
		// TODO Auto-generated method stub
		//super.dispatchTouchEvent(event);
		System.out.println("disdis2222222222");
		//performClick();
		switch (event.getAction()) {
		case MotionEvent.ACTION_DOWN:
			System.out.println("disdis");
			break;

		default:
			break;
		}
		return true;
	}

这是我随便写的一个测试代码,正常的一次点击会触发3次这个函数(按下,移动,抬起)

既然知道这件事情,我们就需要这样想:我们总不能在手指只要在这个view有操作都多次调用这个函数吧。

所以最后我只在按下的时候才调用click的函数。

那接着探讨下一个问题:怎么调用click函数呢?

通过上面我推荐的那篇blog,我们知道最后调用这个函数的是在performClick这个函数里面,由于我不关心其他点击事件,所以我就这么写了:

@Override
	public boolean dispatchTouchEvent(MotionEvent event) {
		// TODO Auto-generated method stub
		switch (event.getAction()) {
		case MotionEvent.ACTION_DOWN:
			performClick();
			break;

		default:
			break;
		}
		return true;
	}

注意一下返回值,我返回true意思是这个函数消费了点击事件,因为这次点击事件是在这里被消费掉的,所以最好返回true,不然的话不知道会出现什么错误呢。

下面是我的整个重写的Button类:

package com.example.luntan;

import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.widget.Button;

public class MyButton extends Button {
	public MyButton(Context context) {
		// TODO Auto-generated constructor stub
		super(context);
	}

	public MyButton(Context context, AttributeSet attrs) {
		super(context, attrs);
		// TODO Auto-generated constructor stub
	}

	@Override
	public boolean dispatchTouchEvent(MotionEvent event) {
		// TODO Auto-generated method stub
		switch (event.getAction()) {
		case MotionEvent.ACTION_DOWN:
			performClick();
			break;

		default:
			break;
		}
		return false;
	}
}

在其他Activity中设置onClick:

holder.replyBt.setOnClickListener(new OnClickListener() {

				@Override
				public void onClick(View v) {
					// TODO Auto-generated method stub
					Intent intent = new Intent(ForumMain.this,
							ForumReply.class);
					String topicId = info.TopicID;
					intent.putExtra("topicId", topicId);
					startActivity(intent);
				}
			});