Android事件分发机制详解,网易资深安卓架构师

事件序列:其实就是从手指触摸屏幕到离开屏幕所发生的一系列事件

2. 什么是事件分发

我们要讲的事件分发其实就是将点击事件传递到某个具体的View,这个传递的过程就叫做事件分发

3. 事件在哪些对象间进行传递、顺序是什么

ActivityUI界面由ActivityViewGroupView及其派生类组成

事件分发在这三个对象之间进行传递。

当点击事件发生后,事件先传到Activity,再传到ViewGroup,最终传到View

4. 事件分发有啥用?

默认情况下事件分发会按照由ActivityViewGroup再到View的顺序进行分发,当我们不想View进行处理,让ViewGroup处理,那就可以进行拦截,这些知识可以用于解决滑动冲突。

例如:外部滑动和内部滑动方向不一致,当ScrollView嵌套Fragment,且Fragemnt内部有个竖向的ListView,当用户左右滑动时,要让外部的View拦截单击事件,当用户上下滑动时,要让内部的View拦截点击事件。怎么拦截,在哪里拦截,就用到了我们这篇文章所讲的内容了。

5. 事件分发涉及到的函数及相应的作用

方法作用
dispatchTouchEvent进行事件分发
onInterceptTouchEvent事件拦截
onTouchEvent事件消耗(就是交给当前View处理)
  • dispatchTouchEvent: 用来进行事件分发,若事件能够传递到当前View,则此方法一定会被调用。
  • onInterceptTouchEvent:dispatchTouchEvent方法内被调用,用来判断是否拦截某个事件。若当前View拦截了某个事件,则该方法不会再被调用,返回结果表示是否拦截当前事件,该方法只在ViewGroup中存在。
  • onTouchEvent: 用来处理点击事件,返回结果表示是否消耗当前事件,若不消耗,则在同一事件序列中,当前View无法再次接收到事件。

这三个方法可用以下伪代码表示

public boolean dispatchTouchEvent(MotionEvent ev){
boolean consume = false;
if(onInterceptTouchEvent(ev)){
consume = onTouchEvent(ev);
}else{
consume = child.dispatchTouchEvent(ev);
}
return consume;
}

对应的根ViewGroup,当一个点击事件产生时,Activity会传递给它,这时它的dispatchTouchEvent就会被调用,若该ViewGrouponInterceptTouchEvent返回true,代表拦截该事件,但是否消耗该事件,还要看它的onTouchEvent的返回值,如果不拦截,则代表将事件分发下去给子View,接着子ViewdispatchTouchEvent方法会被调用,如此反复直到事件被最终处理。

6. Activity的事件分发

当一个点击事件发生时,事件最先传到ActivitydispatchTouchEvent()进行事件分发。这里主要要弄明白Activity是怎么将事件分发到ViewGroup中的

6.1 Demo演示

我们先看一个案例

(1) 自定义一个MyViewGroup,继承自ViewGroup,重写dispatchTouchEvent()方法

public class MyViewGroup extends ViewGroup{

public boolean dispatchTouchEvent(MotionEvent ev) {
Log.i(TAG, "dispatchTouchEvent: ");
//这里我们暂时先返回false
return false;
}
}

(2)Activity的布局中,使用该布局作为最外层布局

<com.ld.eventdispatchdemo.activitydispatch.MyViewGroup xmlns:android=“http://schemas.android.com/apk/res/android”
xmlns:app=“http://schemas.android.com/apk/res-auto”
xmlns:tools=“http://schemas.android.com/tools”
android:layout_width=“match_parent”
android:layout_height=“match_parent”
android:id=“@+id/myViewGroup”
android:orientation=“vertical”
tools:context=“.activitydispatch.ActivityDispatchActivity”>

</com.ld.eventdispatchdemo.activitydispatch.MyViewGroup>

(3) 重写该ActivitydispatchTouchEvent()onTouchEvent()方法,打印log日志

public class Activity extends AppCompatActivity{

private static final String TAG = “Activit_activitydispatch”;

@Override
public boolean dispatchTouchEvent(MotionEvent ev) {

if(ev.getAction()==MotionEvent.ACTION_DOWN){
Log.i(TAG, "dispatchTouchEvent: ");
}
//这里是仿照源码的格式写的
if(getWindow().superDispatchTouchEvent(ev)){
Log.i(TAG, “dispatchTouchEvent: 这里被调用”);
return true;
}
return onTouchEvent(ev);
}

@Override
public boolean onTouchEvent(MotionEvent event) {
Log.i(TAG, "onTouchEvent: ");
return super.onTouchEvent(event);
}
}

MyViewGroupdispatchTouchEvent返回false时,打印的log日志为:

Activit_activitydispatch: dispatchTouchEvent:
MyViewGroup_activitydispatch: dispatchTouchEvent:
Activit_activitydispatch: onTouchEvent:
Activit_activitydispatch: onTouchEvent:
Activit_activitydispatch: onTouchEvent:
Activit_activitydispatch: onTouchEvent:
Activit_activitydispatch: onTouchEvent:

MyViewGroupdispatchTouchEvent返回true时,打印的log日志为:

Activit_activitydispatch: dispatchTouchEvent:
MyViewGroup_activitydispatch: dispatchTouchEvent:
Activit_activitydispatch: dispatchTouchEvent: 这里被调用
MyViewGroup_activitydispatch: dispatchTouchEvent:
Activit_activitydispatch: dispatchTouchEvent: 这里被调用
MyViewGroup_activitydispatch: dispatchTouchEvent:
Activit_activitydispatch: dispatchTouchEvent: 这里被调用
MyViewGroup_activitydispatch: dispatchTouchEvent:
Activit_activitydispatch: dispatchTouchEvent: 这里被调用

仔细观察可以看到,当MyViewGroup的dispatchTouchEvent返回false时,Activity的onTouchEvent会被调用,返回true时,不会被调用,这是什么原因呢?

你可能会有疑问,我的ActivitydispatchTouchEvent()方法内为何要这样写呢?别急,看完下面的源码你就知道了。

6.2 源码解析

目的:

1、研究MyViewGroupdispatchTouchEvent返回false时,ActivityonTouchEvent会被调用,返回true时,不会被调用的原因

Activity中的dispatchTouchEvent()源码如下:

public boolean dispatchTouchEvent(MotionEvent ev) {
if (ev.getAction() == MotionEvent.ACTION_DOWN) {
//该方法为空方法,不用管它
onUserInteraction();
}
if (getWindow().superDispatchTouchEvent(ev)) {
return true;
}
return onTouchEvent(ev);
}

有没有很熟悉,为了方便观察打日志所以上面我们重写了ActivitydispatchTouchEvent方法,但内容和源码基本一致。

可以从源码看到,当getWindow().superDispatchTouchEvent(ev)==true时,那么此时return ture,自然就不会调用底下的onTouchEvent()方法,即ActivityonTouchEvent()

getWindow()返回Window对象,Window是抽象类,而PhoneWindowWindow的唯一实现类,所以getWindow().superDispatchTouchEvent(ev)其实就是调用的PhoneWindow内的superDispatchTouchEvent(ev)方法。

看看PhoneWindow类源码:

@Override
public boolean superDispatchTouchEvent(MotionEvent event) {
return mDecor.superDispatchTouchEvent(event);
}

PhoneWindow将事件直接传递给了DecorView,接下来看看DecorView是什么

public class DecorView extends FrameLayout implements RootViewSurfaceTaker, WindowCallbacks {

}

mDecorgetWindow().getDecorView()返回的View,通过setContentView设置的View是该View的子View

DecorView继承自FrameLayout(ViewGroup),所以mDecor.superDispatchTouchEvent(event)其实调用的就是ViewGroupdispatchTouchEvent()方法,所以到这里你就懂了吧,

if (getWindow().superDispatchTouchEvent(ev)) {
return true;
}

其实就相当于下面这个

if(viewgroup.DispatchTouchEvent(ev)){
return true;
}

所以说当我们的MyLayoutDispatchTouchEvent()返回true时,ActivityonTouchEvent就不会被调用。

6.3 事件时怎么从Activity分发到ViewGroup中的

从上面ActivitydispatchTouchEvent源码可知道,默认状态下,它内部一定会调用该方法,而if()条件中的内容其实就是调用ViewGroupdispatchTouchEvent()方法,也就是在这里完成了ActivityViewGroup的事件分发。

if (getWindow().superDispatchTouchEvent(ev)) {
return true;
}

6.4 小结:Activity分发的流程图

7. ViewGroup的事件分发

上面讲了ActivitydispatchTouchEvent内将事件传递到了ViewGroupdispatchTouchEvent()方法中,那么ViewGroup又是如何将事件进一步向下分发的呢?

7.1 Demo演示

(1) 自定义MyLayout,继承自LinearLayout,重写onInterceptTouchEvent()方法,并返回true,重写dispatchTouchEvent()onTouchEvent()方法,打印log日志

public class MyLayout extends LinearLayout {

private static final String TAG = “MyLayout_ViewGroupDispatch”;

@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
Log.i(TAG, "dispatchTouchEvent: ");
return super.dispatchTouchEvent(ev);
}

@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
Log.i(TAG, "onInterceptTouchEvent: ");
//此处暂时返回true观察现象
return true;
}

@Override
public boolean onTouchEvent(MotionEvent event) {
Log.i(TAG, "onTouchEvent: ");
return super.onTouchEvent(event);
}
}

(2)Activity的布局中,使用该布局作为最外层布局,并在该布局内添加一个按钮

<com.ld.eventdispatchdemo.viewgroupdispatch.MyLayout xmlns:android=“http://schemas.android.com/apk/res/android”
xmlns:app=“http://schemas.android.com/apk/res-auto”
xmlns:tools=“http://schemas.android.com/tools”
android:layout_width=“match_parent”
android:layout_height=“match_parent”
android:orientation=“vertical”
android:id=“@+id/myLayout”
tools:context=“.viewgroupdispatch.ViewGroupActivity”>

</com.ld.eventdispatchdemo.viewgroupdispatch.MyLayout>

(3)Activity内为按钮添加点击事件

public class ViewGroupActivity extends AppCompatActivity {

private Button btn1;
private static final String TAG = “Activity_ViewGroupDispatch”;
@Override
protected void onCreate(Bundle savedInstanceState) {

btn1 = findViewById(R.id.btn1);
btn1.setOnClickListener(new View.OnClickListener() {
@SuppressLint(“LongLogTag”)
@Override
public void onClick(View v) {
Log.i(TAG, “onClick: 点击了按钮1”);
}
});
}
}

MyLayoutonInterceptTouchEvent()返回true时,分别点击空白处、点击按钮,log日志如下:

//点击空白处
MyLayout_ViewGroupDispatch: dispatchTouchEvent:
MyLayout_ViewGroupDispatch: onInterceptTouchEvent:
MyLayout_ViewGroupDispatch: onTouchEvent:
//点击按钮
MyLayout_ViewGroupDispatch: dispatchTouchEvent:
MyLayout_ViewGroupDispatch: onInterceptTouchEvent:
MyLayout_ViewGroupDispatch: onTouchEvent:

MyLayoutonInterceptTouchEvent()返回false时,分别点击空白处、点击按钮,log日志如下:

//点击空白处
MyLayout_ViewGroupDispatch: dispatchTouchEvent:
MyLayout_ViewGroupDispatch: onInterceptTouchEvent:
MyLayout_ViewGroupDispatch: onTouchEvent:
//点击按钮
MyLayout_ViewGroupDispatch: onInterceptTouchEvent:
MyLayout_ViewGroupDispatch: dispatchTouchEvent:
MyLayout_ViewGroupDispatch: onInterceptTouchEvent:
MyLayout_ViewGroupDispatch: dispatchTouchEvent:
MyLayout_ViewGroupDispatch: onInterceptTouchEvent:
MyLayout_ViewGroupDispatch: dispatchTouchEvent:
MyLayout_ViewGroupDispatch: onInterceptTouchEvent:
MyLayout_ViewGroupDispatch: dispatchTouchEvent:
MyLayout_ViewGroupDispatch: onInterceptTouchEvent:
Activity_ViewGroupDispatch: onClick: 点击了按钮1

可以看到当ViewGroup(MyLayout)onInterceptTouchEvent()返回true时,并没有触发按钮的点击事件,并且自身的onTouchEvent()方法被调用,当返回false时,按钮的点击事件触发,但自身的onTouchEvent()方法未被调用。

而且在默认状态下,onInterceptTouchEvent()一定会被调用。

以上现象是什么原因呢?接下来我们看看ViewGroup的dispatchTouchEvent()方法的源码

7.2 源码解析

目的:

1、研究当ViewGroup(MyLayout)onInterceptTouchEvent()返回true时,并没有触发按钮的点击事件,并且自身的onTouchEvent()方法被调用,当返回false时,按钮的点击事件触发,但自身的onTouchEvent()方法未被调用的原因

2、分发事件是怎么从ViewGroup分发到View中的

ViewGroupdispatchTouchEvent()源码:

@Override
public boolean dispatchTouchEvent(MotionEvent ev) {

boolean handled = false;
if (onFilterTouchEventForSecurity(ev)) {

//一大堆代码
}

if (!handled && mInputEventConsistencyVerifier != null) {
mInputEventConsistencyVerifier.onUnhandledEvent(ev, 1);
}
return handled;
}

我们想要了解事件是如何分发,其实是主要看ViewGroupdispatchTouchEvent()方法什么时候返回true,什么时候返回false。看源码可以知道,ViewGroupdispatchTouchEvent()方法返回的是handle的值,所以我们只需要观察该方法内改变handle值的语句。

首先初始化了handle的值,默认为false

然后你可以看到dispatchTouchEvent()的大部分内容都在if (onFilterTouchEventForSecurity(ev)) {}这个条件判断内,也就是说如果onFilterTouchEventForSecurity(ev)方法返回true的话,那么就进入该if判断内。若返回false,则dispatchTouchEvent()返回初始值为falsehandled,表示不分发事件。

查看下onFilterTouchEventForSecurity(ev)方法

public boolean onFilterTouchEventForSecurity(MotionEvent event) {
//noinspection RedundantIfStatement
if ((mViewFlags & FILTER_TOUCHES_WHEN_OBSCURED) != 0
&& (event.getFlags() & MotionEvent.FLAG_WINDOW_IS_OBSCURED) != 0) {
// Window is obscured, drop this touch.
return false;
}
return true;
}

  • FILTER_TOUCHES_WHEN_OBSCUREDandroid:filterTouchesWhenObscured属性所对应的。android:filterTouchesWhenObscuredtrue的话,则表示其他视图在该视图之上,导致该视图被隐藏时,该视图就不再响应触摸事件。
  • MotionEvent.FLAG_WINDOW_IS_OBSCUREDtrue的话,则表示该视图的窗口是被隐藏的

而我们并没有在XML中为控件设置android:filterTouchesWhenObscured属性,所以它==0,没有进入if()方法,所以onFilterTouchEventForSecurity()方法返回true,那么 if (onFilterTouchEventForSecurity(ev))判断必定会进入如下的判断中。

接下来我们看看if(onFilterTouchEventForSecurity(ev))判断下的内容

if (onFilterTouchEventForSecurity(ev)) {
final int action = ev.getAction();
final int actionMasked = action & MotionEvent.ACTION_MASK;

if (actionMasked == MotionEvent.ACTION_DOWN) {
cancelAndClearTouchTargets(ev);

resetTouchState();
}


}

private void resetTouchState() {

clearTouchTargets();

resetCancelNextUpFlag(this);
mGroupFlags &= ~FLAG_DISALLOW_INTERCEPT;

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级安卓工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Android移动开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Android)
img

最后

愿你有一天,真爱自己,善待自己。

本文在开源项目:Android开发不会这些?如何面试拿高薪 中已收录,里面包含不同方向的自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中…

-1710508090609)]
[外链图片转存中…(img-GeTWlWYW-1710508090610)]
[外链图片转存中…(img-oJF6r4A5-1710508090610)]
[外链图片转存中…(img-AyrL5WGF-1710508090611)]

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Android)
[外链图片转存中…(img-aBAEVDbo-1710508090611)]

最后

愿你有一天,真爱自己,善待自己。

本文在开源项目:Android开发不会这些?如何面试拿高薪 中已收录,里面包含不同方向的自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中…

  • 21
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值