这是个困惑我好久的问题,搞了好久才明白一些,看了很多大牛的技术文章,也写了一些小demo测试了一番,菜鸟在这里和大家分享一下了!
首先,这里涉及到View和ViewGroup,但是View只有事件的响应与否,我的理解是应为View不是容器型的控件,但是ViewGroup是容器型的控件,所以其涉及到事件的传递与回传,所谓回传就是响应!
好了,我们切入主题,
button.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Log.d("TAG", "onClick execute");
}
});
button.setOnTouchListener(new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
Log.d("TAG", "onTouch execute, action " + event.getAction());
return false;
}
});
当我们触碰到任何一个控件的时候,都会调用该控件的dispatchTouchEvent方法,这里我们查看程序的运行结果如下所示
onTouch execute,action0
onTouch execute,aciton1
onClick execute
可以看到,onTouch是优先于onClick执行的,并且onTouch执行了两次,一次是ACTION_DOWN,一次是ACTION_UP(你还可能会有多次ACTION_MOVE的执行,如果你手抖了一下)。因此事件传递的顺序是先经过onTouch,再传递到onClick。
但是如果我们把onTouch方法的返回值返回true的话,我们会发现onClick竟然没有执行
追究其原因,我们就要仔细的去看一下View的dispatchTouchEvent源码:
public boolean dispatchTouchEvent(MotionEvent event) {
if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED &&
mOnTouchListener.onTouch(this, event)) {
return true;
}
return onTouchEvent(event);
}
进入if语句首先我们要知道mOnTouchListener 到底是谁,才能进而判断出其是不是为null,按住ctrl+k我们发现了一个惊天大秘密。
public void setOnTouchListener(OnTouchListener l) {
mOnTouchListener = l;
}
与那里这里的mOnTouchListener,就是我们写的内部类
new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
Log.d("TAG", "onTouch execute, action " + event.getAction());
return false;
}
我们已经赋值了,它是绝对不会为空的。再看第二个参数(mViewFlags & ENABLED_MASK) == ENABLED,它的意思就是该控件是否可用的enabled,所有控件默认都是可用的,所以这里我们只需要看第三个条件即可,mOnTouchListener.onTouch(this, event),其实这里就是回调啦,这里正是我们自己写的onTouch函数呀,当该函数返回true的时候,disPatcheTouchEvent的返回值就是true,这时代表着onTouch事件是被消耗掉的,所以onclick函数是不会被执行的。
由此可知,onclick事件的执行一定是在onTouchEvent(event)中调用的!所以当onTouch函数返回false的时候,就会调用onTouchEvent函数,好了接下来我们继续看源码
public boolean onTouchEvent(MotionEvent event) {
final int viewFlags =