先写个简单的Demo,在布局中创建简单的ImageView和Button,布局代码如下:
<ImageView
android:id="@+id/imageview"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/ic_launcher" />
<Button
android:id="@+id/button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="按钮"/>
在代码中设置点击事件:
imageview.setOnTouchListener(new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
Log.i(tag, "imageview event = "+event.getAction());
return false;
}
});
button.setOnTouchListener(new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
Log.i(tag, "button event = "+event.getAction());
return false;
}
});
点击image和button对应的打印结果:注意是点击不是滑动。。。。
imageview event = 0
button event = 0
button event = 1
我们都知道:
int ACTION_DOWN = 0;
int ACTION_UP = 1;
int ACTION_MOVE = 2;
为什么imageView只会处理ACTION_DOWN事件而button会响应ACTION_DOWN和ACTION_UP呢?
因为ImageView和Button都是继承至View,所以这就涉及到的View的事件分发机制
我们都知道所有的事件的分发都是通过dispatchTouchEvent来处理的,所以这就需要我们先去查看继承至View的子类是否有重写该方法
通过查看源码发现ImageView和Button并没有重写dispatchTouchEvent方法,所以View的dispatchTouchEvent方法对两者均适用
分析View的dispatchTouchEvent方法源码:
public boolean dispatchTouchEvent(MotionEvent event) {
//mOnTouchListener != null:设置Touch事件监听时传递的对象 != null,在setOnTouchListener里赋值
//(mViewFlags & ENABLED_MASK) == ENABLED:当前控件是否可用,Android下所有控件默认都是可用的,可手动设置不可用
//mOnTouchListener.onTouch(this, event)):即调用设置OnTouchListener时重写的onTouch方法获取其中的返回值
if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED &&
mOnTouchListener.onTouch(this, event)) {
return true;
}
return onTouchEvent(event);
}
所以当子View在设置OnTouchListener时重写onTouch方法return false,则View的dispatchTouchEvent方法继续往下走调用return onTouchEvent(event);
查看View的onTouchEvent(event)方法:
代码很多,但是事件处理在Switch中执行,所以我们只需要查看核心代码即可
//(viewFlags & CLICKABLE) == CLICKABLE 当前控件是否可点击
//(viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE),当前控件是否长按
if (((viewFlags & CLICKABLE) == CLICKABLE ||
(viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)) {
switch (event.getAction()) {
case MotionEvent.ACTION_UP:
break;
case MotionEvent.ACTION_DOWN:
break;
case MotionEvent.ACTION_CANCEL:
break;
case MotionEvent.ACTION_MOVE:
break;
}
//如果可点击返回true
return true;
}
//控件不可点击返回false
return false;
通过分析可以得出结论:
imageview.setOnTouchListener(new OnTouchListener() {
//false 打印1次 dispatchTouchEvent返回false,表示事件没有响应完毕,所以只有ACTION_DOWN事件
//true 打印2次 dispatchTouchEvent返回true,表示事件响应完毕,所以在点击事件中有ACTION_DOWN和ACTION_UP事件
@Override
public boolean onTouch(View v, MotionEvent event) {
Log.v(tag, "imageview event = "+event.getAction());
return false;
}
});
button.setOnTouchListener(new OnTouchListener() {
//false 打印2次 dispatchTouchEvent返回true,表示事件响应完毕,所以在点击事件中有ACTION_DOWN和ACTION_UP事件
//true 打印2次 dispatchTouchEvent返回true,表示事件响应完毕,所以在点击事件中有ACTION_DOWN和ACTION_UP事件
@Override
public boolean onTouch(View v, MotionEvent event) {
Log.v(tag, "button event = "+event.getAction());
return false;
}
});
结论:
1、在事件处理中系统是先执行我们重写的onTouch方法,如果满足3个条件即:有给当前控件设置监听、当前控件可用、重写的onTouch方法返回true,则返回true表示当前控件响应事件,否则执行onTouchEvent方法继续进行判断
2、在onTouchEvent方法方法中如果当前控件可点击则在switch中处理down、move、up、cancle事件并返回true表示事件已处理,否则返回false表示当前控件未处理事件
也就是说:打印一次即未响应事件的最终原因是:在重写的onTouch方法返回false并且当前控件不可点击!!
整理思路:当我们在onTuch事件中reutrn true表示当前控件要响应事件,如果是return false则调用onTuchEvent方法判断当前控件是否可以点击,如果可以点击则即使在onTuch中返回false也会响应事件
所以当给ImageView设置点击事件时又会是什么情况呢?
imageview.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Log.i(tag, "imageview OnClickListener");
}
});
查看ImageView并没有对setOnClickListener重写,所以还是看View的源码:
public void setOnClickListener(OnClickListener l) {
if (!isClickable()) {
setClickable(true);
}
mOnClickListener = l;
}
即:只要设置控件的点击事件则View类中会将此控件设置为可点击,onClick方法会在onTouch后再去执行
只有当onTouch返回false后才会去onTouchEvent里触发onClick,即点击事件是从触摸事件里解析出来的
代码体现:在View的onTouchEvent的ACTION_UP事件中,当满足点击条件时:
if (!post(mPerformClick)) {
performClick();
}
查看performClick方法
public boolean performClick() {
sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);
if (mOnClickListener != null) {
playSoundEffect(SoundEffectConstants.CLICK);
mOnClickListener.onClick(this);
return true;
}
return false;
}
即回调了我们重写的onClick方法
总结:dispatchTouchEvent 返回false 事件响应不完毕
dispatchTouchEvent 返回true 事件响应完毕
onTouch-->决定是否执行onTouchEvent-->决定是否执行onClick
注意:此处暂时未研究事件的onInterceptTouchEvent方法,所以默认父控件未对事件传递进行拦截