Android事件传递机制分析

准备阶段

为了方便了解整个事件传递的机制,我们设计一个场景:

角色:
一个经理:MotionEventViewGroupA,最外层的ViewGroupA;
一个组长:MotionEventViewGroupB,中间的ViewGroupB;
一个你:MotionEventViewC,最底层的码农。
模拟:
经理分派任务,下属处理这个任务的过程。

MotionEventViewGroupA

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
public class MotionEventViewGroupA extends LinearLayout {
public MotionEventViewGroupA(Context context) {
super(context);
}
public MotionEventViewGroupA(Context context, AttributeSet attrs) {
super(context, attrs);
}
public MotionEventViewGroupA(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
Log.e("wxl", "MotionEventViewGroupA dispatchTouchEventA");
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
Log.e("wxl", "MotionEventViewGroupA onInterceptTouchEventA");
return super.onInterceptTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
Log.e("wxl", "MotionEventViewGroupA onTouchEventA");
return super.onTouchEvent(event);
}
}

MotionEventViewGroupB

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
public class MotionEventViewGroupB extends LinearLayout {
public MotionEventViewGroupB(Context context) {
super(context);
}
public MotionEventViewGroupB(Context context, AttributeSet attrs) {
super(context, attrs);
}
public MotionEventViewGroupB(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
Log.e("wxl", "MotionEventViewGroupB dispatchTouchEventB");
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
Log.e("wxl", "MotionEventViewGroupB onInterceptTouchEventB");
return super.onInterceptTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
Log.e("wxl", "MotionEventViewGroupB onTouchEventB");
return super.onTouchEvent(event);
}
}

MotionEventViewC

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
public class MotionEventViewC extends View {
public MotionEventViewC(Context context) {
super(context);
}
public MotionEventViewC(Context context, AttributeSet attrs) {
super(context, attrs);
}
public MotionEventViewC(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
Log.e("wxl", "MotionEventViewC dispatchTouchEventC");
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
Log.e("wxl", "MotionEventViewC onTouchEventC");
return super.onTouchEvent(event);
}
}

xml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="com.wuxiaolong.apksample.MotionEventActivity">
<com.wuxiaolong.apksample.MotionEventViewGroupA
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_gravity="center"
android:background="@android:color/holo_orange_dark"
android:gravity="center">
<com.wuxiaolong.apksample.MotionEventViewGroupB
android:layout_width="300dp"
android:layout_height="300dp"
android:background="@android:color/holo_purple"
android:gravity="center">
<com.wuxiaolong.apksample.MotionEventViewC
android:layout_width="200dp"
android:layout_height="200dp"
android:background="@android:color/holo_green_dark"
android:gravity="center" />
</com.wuxiaolong.apksample.MotionEventViewGroupB>
</com.wuxiaolong.apksample.MotionEventViewGroupA>
</FrameLayout>

ViewGroup级别比较高,比View多一个onInterceptTouchEvent(拦截)。

情景分析

不做任何修改,点击MotionEventViewC

1
2
3
4
5
6
7
8
MotionEventViewGroupA dispatchTouchEventA
MotionEventViewGroupA onInterceptTouchEventA
MotionEventViewGroupB dispatchTouchEventB
MotionEventViewGroupB onInterceptTouchEventB
MotionEventViewC dispatchTouchEventC
MotionEventViewC onTouchEventC
MotionEventViewGroupB onTouchEventB
MotionEventViewGroupA onTouchEventA

log信息看出,正常情况,事件传递顺序:
经理 –> 组长 –> 你,先执行dispatchTouchEvent(分发),再执行onInterceptTouchEvent(拦截)

事件处理顺序:
你 –> 组长 –> 经理,事件处理都是执行onTouchEvent(处理)。

事件传递返回值:true,拦截,交给自己的onTouchEvent处理;false,不拦截,传给下属。

事件处理返回值:true,自己搞定,不用上报上司;false,上报上司处理。

初始返回都是false。

事件传递,dispatchTouchEvent一般不太会改写,只关心onInterceptTouchEvent。

经理觉得这个任务太简单,自己处理

即MotionEventViewGroupA里onInterceptTouchEvent返回true,我们看下log信息:

1
2
3
MotionEventViewGroupA dispatchTouchEventA
MotionEventViewGroupA onInterceptTouchEventA
MotionEventViewGroupA onTouchEventA

经理一个人干完了,没下面的人什么事了。

组长觉得这个任务太简单,自己处理

即MotionEventViewGroupB里onInterceptTouchEvent返回true,我们看下log信息:

1
2
3
4
5
6
MotionEventViewGroupA dispatchTouchEventA
MotionEventViewGroupA onInterceptTouchEventA
MotionEventViewGroupB dispatchTouchEventB
MotionEventViewGroupB onInterceptTouchEventB
MotionEventViewGroupB onTouchEventB
MotionEventViewGroupA onTouchEventA

组长干完了活,没你啥事了。以上对事件的分发、拦截的分析。

你迫于压力,辞职不干了,任务闲置

即MotionEventViewC里onTouchEvent返回true,我们看下log信息:

1
2
3
4
5
6
MotionEventViewGroupA dispatchTouchEventA
MotionEventViewGroupA onInterceptTouchEventA
MotionEventViewGroupB dispatchTouchEventB
MotionEventViewGroupB onInterceptTouchEventB
MotionEventViewC dispatchTouchEventC
MotionEventViewC onTouchEventC

事件处理到你这里就结束了。

组长觉得你任务完成太烂,不敢上报

即MotionEventViewB里onTouchEvent返回true,我们看下log信息:

1
2
3
4
5
6
7
MotionEventViewGroupA dispatchTouchEventA
MotionEventViewGroupA onInterceptTouchEventA
MotionEventViewGroupB dispatchTouchEventB
MotionEventViewGroupB onInterceptTouchEventB
MotionEventViewC dispatchTouchEventC
MotionEventViewC onTouchEventC
MotionEventViewGroupB onTouchEventB

事件处理经过你和组长,到组长那里就结束了。

是不是这样的分析让你对事件的传递不再是一头雾水了,大赞医生《Android群英传》。



联系作者

我的微信公众号:吴小龙同学,欢迎关注交流,公号回复关键字「1024」有惊喜哦。