参考《android中的事件传递和处理机制》 侵删
理解一些事情
在ViewGroup(如LinearLayout、RelativeLayout等继承ViewGroup的)中,有下面三个方法:
dispatchTouchEvent 该方法用来分发事件
onInterceptTouchEvent 用来拦截事件
onTouchEvent 用来处理事件
而View(如Button、TextView等继承View的)中,只有两个方法,即:
dispatchTouchEvent 该方法用来分发事件
onTouchEvent 用来处理事件
为什么View没有onInterceptTouchEvent方法?因为View里面已经不能再包含View了,不需要再处理拦截时间的分发。
例子
写了个布局
<?xml version="1.0" encoding="utf-8"?>
<com.zyou.myapplication.ViewGroupA
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center"
android:background="@android:color/holo_red_light">
<com.zyou.myapplication.MyView
android:id="@+id/btn_button"
android:layout_width="100dp"
android:layout_height="100dp"
android:background="@android:color/holo_blue_dark"/>
</com.zyou.myapplication.ViewGroupA>
红色的就是ViewGroupA,蓝色就是MyView。
并在View、ViewGroupA、ViewGroupB复写方法,并加入LOG信息。
public class ViewGroupA extends LinearLayout {
private final String TAG="fzy";
public ViewGroupA(Context context) {
super(context);
}
public ViewGroupA(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public ViewGroupA(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
Log.d(TAG,"ViewGroupA onInterceptTouchEvent ");
return super.onInterceptTouchEvent(ev);
}
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
Log.d(TAG,"ViewGroupA dispatchTouchEvent ");
return super.dispatchTouchEvent(event);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
Log.d(TAG,"ViewGroupA onTouchEvent ");
return super.onTouchEvent(event);
}
}
现在来点击中间黄色的MyView,观察下打印结果,如下:
MainActivity2 dispatchTouchEvent: 0
ViewGroupA dispatchTouchEvent: 0
ViewGroupA onInterceptTouchEvent: 0
MyView dispatchTouchEvent: 0
MyView onTouchEvent :0
ViewGroupA onTouchEvent: 0
MainActivity2 onTouchEvent: 0
MainActivity2 dispatchTouchEvent: 2
MainActivity2 onTouchEvent: 2
MainActivity2 dispatchTouchEvent: 2
MainActivity2 onTouchEvent: 2
MainActivity2 dispatchTouchEvent: 2
MainActivity2 onTouchEvent: 2
MainActivity2 dispatchTouchEvent: 1
MainActivity2 onTouchEvent: 1
可以看到我们当touch的是MyView:
1. 首先处理事件分发的是MainActivity2的dispatchTouchEvent,然后把事件往下传递给ViewGroupA。
2. 如果ViewGroupA的onInterceptTouchEvent没有进行拦截,则把事件传给MyView。
3. 由MyView的onTouchEvent进行时间处理。如果方法都返回super.onTouchEvent,最后的ACITON_MOVE/ACTION_UP事件都不再经过ViewGroupA和MyView,而是直接在MainActivity2处理。
下面我们来测试一下,把ViewGroupA的onInterceptTouchEvent返回true,表示事件不再往下传递,即传递不到MyView了,表示ViewGroupA已经处理,不需要往下传了,以下是LOG:
MainActivity2 dispatchTouchEvent: 0
ViewGroupA dispatchTouchEvent: 0
ViewGroupA onInterceptTouchEvent: 0
ViewGroupA onTouchEvent: 0
MainActivity2 onTouchEvent: 0
MainActivity2 dispatchTouchEvent: 2
MainActivity2 onTouchEvent: 2
MainActivity2 dispatchTouchEvent: 2
MainActivity2 onTouchEvent: 2
MainActivity2 dispatchTouchEvent: 2
MainActivity2 onTouchEvent: 2
MainActivity2 dispatchTouchEvent: 1
MainActivity2 onTouchEvent: 1
总结
用户点击屏幕,分发事件是由Activity–>ViewGroupA–>MyView走的,而onTouchEvent是从MyView–>ViewGroupA–>Activity,中途如果有onInterceptTouchEvent拦截事件,则dispatchTouchEvent不会往下传递,就回去了。整个过程有点像跳高,onInterceptTouchEvent相当于是块板。
最后补一张事件分发图: