Framework事件机制-深扒onInterceptTouchEvent拦截流程解析

Framework事件机制-深扒onInterceptTouchEvent拦截流程解析

一、基本知识

1.事件分发的三个函数

事件的分发 dispatchTouchEvent()

事件的拦截 onInterceptTouchEvent()

事件的处理(消费) onTouchEvent()

2.事件分发的对象

被分发的对象是那些?被分发的对象是用户触摸屏幕而产生的点击事件,事件主要包括:按下、滑动、抬起与取消。这些事件被封装成MotionEvent对象。

MotionEvent.ACTION_DOWN 在屏幕按下时 MotionEvent.ACTION_MOVE 在屏幕上滑动时 MotionEvent.ACTION_UP 在屏幕抬起时 MotionEvent.ACTION_CANCLE 滑动超出控件边界时

3.分发事件的组件

分发事件的组件,也称为分发事件者,包括Activity、View和ViewGroup。它们三者的一般结构为:

二、事件处理流程

首先,我们需要了解事件处理中的几个方法

1、在ViewGroup中,事件分为dispatchTouchEvent(事件的分发),onInterceptTouchEvent(事件的拦截),onTouchEvent(事件的处理)。

2、在View中,事件分为dispatchTouchEvent(事件的分发),onTouchEvent(事件的处理)。

下面是demo的界面结构,它是由两个自定义的ViewGroup和一个自定义的View组成,并分别重写了它们的以上几个方法。

其中MyViewGroupA代码如下:

public class MyViewGroupA extends LinearLayout { public MyViewGroupA(Context context) { super(context); } public MyViewGroupA(Context context, AttributeSet attrs) { super(context, attrs); } @Override public boolean dispatchTouchEvent(MotionEvent ev) { switch (ev.getAction()){ case MotionEvent.ACTION_DOWN: Log.i(“MyViewGroupA”,”dispatchTouchEvent_ACTION_DOWN”); break; case MotionEvent.ACTION_MOVE: Log.i(“MyViewGroupA”,”dispatchTouchEvent_ACTION_MOVE”); break; case MotionEvent.ACTION_UP: Log.i(“MyViewGroupA”,”dispatchTouchEvent_ACTION_UP”); break; } return super.dispatchTouchEvent(ev); } @Override public boolean onInterceptTouchEvent(MotionEvent ev) { switch (ev.getAction()){ case MotionEvent.ACTION_DOWN: Log.i(“MyViewGroupA”,”onInterceptTouchEvent_ACTION_DOWN”); break; case MotionEvent.ACTION_MOVE: Log.i(“MyViewGroupA”,”onInterceptTouchEvent_ACTION_MOVE”); break; case MotionEvent.ACTION_UP: Log.i(“MyViewGroupA”,”onInterceptTouchEvent_ACTION_UP”); break; } return super.onInterceptTouchEvent(ev); } @Override public boolean onTouchEvent(MotionEvent event) { switch (event.getAction()){ case MotionEvent.ACTION_DOWN: Log.i(“MyViewGroupA”,”onTouchEvent_ACTION_DOWN”); break; case MotionEvent.ACTION_MOVE: Log.i(“MyViewGroupA”,”onTouchEvent_ACTION_MOVE”); break; case MotionEvent.ACTION_UP: Log.i(“MyViewGroupA”,”onTouchEvent_ACTION_UP”); break; } return super.onTouchEvent(event); }}

MyViewGroupB代码如下:

public class MyViewGroupB extends LinearLayout { public MyViewGroupB(Context context) { super(context); } public MyViewGroupB(Context context, AttributeSet attrs) { super(context, attrs); } @Override public boolean dispatchTouchEvent(MotionEvent ev) { switch (ev.getAction()){ case MotionEvent.ACTION_DOWN: Log.i(“MyViewGroupB”,”dispatchTouchEvent_ACTION_DOWN”); break; case MotionEvent.ACTION_MOVE: Log.i(“MyViewGroupB”,”dispatchTouchEvent_ACTION_MOVE”); break; case MotionEvent.ACTION_UP: Log.i(“MyViewGroupB”,”dispatchTouchEvent_ACTION_UP”); break; } return super.dispatchTouchEvent(ev); } @Override public boolean onInterceptTouchEvent(MotionEvent ev) { switch (ev.getAction()){ case MotionEvent.ACTION_DOWN: Log.i(“MyViewGroupB”,”onInterceptTouchEvent_ACTION_DOWN”); break; case MotionEvent.ACTION_MOVE: Log.i(“MyViewGroupB”,”onInterceptTouchEvent_ACTION_MOVE”); break; case MotionEvent.ACTION_UP: Log.i(“MyViewGroupB”,”onInterceptTouchEvent_ACTION_UP”); break; } return super.onInterceptTouchEvent(ev); } @Override public boolean onTouchEvent(MotionEvent event) { switch (event.getAction()){ case MotionEvent.ACTION_DOWN: Log.i(“MyViewGroupB”,”onTouchEvent_ACTION_DOWN”); break; case MotionEvent.ACTION_MOVE: Log.i(“MyViewGroupB”,”onTouchEvent_ACTION_MOVE”); break; case MotionEvent.ACTION_UP: Log.i(“MyViewGroupB”,”onTouchEvent_ACTION_UP”); break; } return super.onTouchEvent(event); }}

MyView代码如下:

public class MyView extends View { public MyView(Context context) { super(context); } public MyView(Context context, AttributeSet attrs) { super(context, attrs); } @Override public boolean dispatchTouchEvent(MotionEvent event) { switch (event.getAction()){ case MotionEvent.ACTION_DOWN: Log.i(“MyView”,”dispatchTouchEvent_ACTION_DOWN”); break; case MotionEvent.ACTION_MOVE: Log.i(“MyView”,”dispatchTouchEvent_ACTION_MOVE”); break; case MotionEvent.ACTION_UP: Log.i(“MyView”,”dispatchTouchEvent_ACTION_UP”); break; } return super.dispatchTouchEvent(event); } @Override public boolean onTouchEvent(MotionEvent event) { switch (event.getAction()){ case MotionEvent.ACTION_DOWN: Log.i(“MyView”,”onTouchEvent_ACTION_DOWN”); break; case MotionEvent.ACTION_MOVE: Log.i(“MyView”,”onTouchEvent_ACTION_MOVE”); break; case MotionEvent.ACTION_UP: Log.i(“MyView”,”onTouchEvent_ACTION_UP”); break; } return super.onTouchEvent(event); }}

我们说过,事件传递是由上到下的,所以最外层的View首先对事件进行操作。而我们最外层是Activity,所以事件也是从这里开始。 Activity代码如下:

public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); } @Override public boolean dispatchTouchEvent(MotionEvent event) { switch (event.getAction()){ case MotionEvent.ACTION_DOWN: Log.i(“Activity”,”dispatchTouchEvent_ACTION_DOWN”); break; case MotionEvent.ACTION_MOVE: Log.i(“Activity”,”dispatchTouchEvent_ACTION_MOVE”); break; case MotionEvent.ACTION_UP: Log.i(“Activity”,”dispatchTouchEvent_ACTION_UP”); break; } return super.dispatchTouchEvent(event); } @Override public boolean onTouchEvent(MotionEvent event) { switch (event.getAction()){ case MotionEvent.ACTION_DOWN: Log.i(“Activity”,”onTouchEvent_ACTION_DOWN”); break; case MotionEvent.ACTION_MOVE: Log.i(“Activity”,”onTouchEvent_ACTION_MOVE”); break; case MotionEvent.ACTION_UP: Log.i(“Activity”,”onTouchEvent_ACTION_UP”); break; } return super.onTouchEvent(event); }}

现在我们通过触摸MyView开始进行分析。虽然dispatchTouchEvent是事件开始的第一步,但是在开发中,我们通常很少改写它,所以我们下面只讨论其他两个方法。 1、对以上方法均不作处理,都返回super。这意味着我们既不拦截,也不消费。

大家看输出结果:

I/Activity: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupA: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupA: onInterceptTouchEvent_ACTION_DOWNI/MyViewGroupB: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupB: onInterceptTouchEvent_ACTION_DOWNI/MyView: dispatchTouchEvent_ACTION_DOWNI/MyView: onTouchEvent_ACTION_DOWNI/MyViewGroupB: onTouchEvent_ACTION_DOWNI/MyViewGroupA: onTouchEvent_ACTION_DOWNI/Activity: onTouchEvent_ACTION_DOWNI/Activity: dispatchTouchEvent_ACTION_MOVEI/Activity: onTouchEvent_ACTION_MOVEI/Activity: dispatchTouchEvent_ACTION_UPI/Activity: onTouchEvent_ACTION_UP

结合输出结果,我们可以总结出以下的结论:

结合流程图,不难发现,如果我对事件既不拦截,也不消费,当触发ACTION_DOWN的时候,事件会经过Activity——MyViewGroupA——MyViewGroupB——MyView一层层的向下进行dispatchTouchEvent(分发)—onInterceptTouchEvent(拦截)调用。当到达最底层MyView后,开始触发消费操作,因为我均不消费,ACTION_DOWN将由底层一层层向上冒,移交上层处理。当抵达最上层Activity后,说明下层均不消费,之后触发的ACTION_MOVE和ACTION_UP将不再向下层分发传递,直接交由Activity分发给自己进行处理。

2、我们将MyVIewGroupB的onInterceptTouchEvent返回值改为true,其他均是super。这意味着仅仅MyViewGroupB进行事件拦截,但均无消费

输出结果如下:

I/Activity: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupA: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupA: onInterceptTouchEvent_ACTION_DOWNI/MyViewGroupB: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupB: onInterceptTouchEvent_ACTION_DOWNI/MyViewGroupB: onTouchEvent_ACTION_DOWNI/MyViewGroupA: onTouchEvent_ACTION_DOWNI/Activity: onTouchEvent_ACTION_DOWNI/Activity: dispatchTouchEvent_ACTION_MOVEI/Activity: onTouchEvent_ACTION_MOVEI/Activity: dispatchTouchEvent_ACTION_UPI/Activity: onTouchEvent_ACTION_UP

结合输出结果,总结如下:

当触发ACTION_DOWN的时候,事件依然是从Activity开始一层层向下传递,当传递到MyViewGroupB时,因为进行了事件拦截,所以执行完onInterceptTouchEvent后不再向下传递,而是直接交由MyViewGroupB的onTouchEvent进行消费处理。由于我们是只拦截,不消费,所以事件向上传递,交由上层处理,最终回到Activity。之后触发的ACTION_MOVE和ACTION_UP也不再向下传递,直接交由Activity分发给自己处理。

3、我们还是将MyViewGroupB的onInterceptTouchEvent返回super,但是将他的onTouchEvent返回true。这意味着我们不拦截,但是由MyViewGroupB进行事件处理。

输出结果如下:

I/Activity: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupA: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupA: onInterceptTouchEvent_ACTION_DOWNI/MyViewGroupB: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupB: onInterceptTouchEvent_ACTION_DOWNI/MyView: dispatchTouchEvent_ACTION_DOWNI/MyView: onTouchEvent_ACTION_DOWNI/MyViewGroupB: onTouchEvent_ACTION_DOWNI/Activity: dispatchTouchEvent_ACTION_MOVEI/MyViewGroupA: dispatchTouchEvent_ACTION_MOVEI/MyViewGroupA: onInterceptTouchEvent_ACTION_MOVEI/MyViewGroupB: dispatchTouchEvent_ACTION_MOVEI/MyViewGroupB: onTouchEvent_ACTION_MOVEI/Activity: dispatchTouchEvent_ACTION_UPI/MyViewGroupA: dispatchTouchEvent_ACTION_UPI/MyViewGroupA: onInterceptTouchEvent_ACTION_UPI/MyViewGroupB: dispatchTouchEvent_ACTION_UPI/MyViewGroupB: onTouchEvent_ACTION_UP

结合输出结果,总结如下:

可以看出,当触发ACTION_DOWN的时候,事件的分发传递过程和1的时候一样,从Activity开始一层层向下传递,最终传递到最底层MyView,触发消费操作,然后MyView将消费操作移交上层处理,然后到达MyViewGroupB的onTouchEvent,并且进行了消费处理,事件处理到此不在向上移交。当触发ACTION_MOVE和ACTION_UP操作时,事件依然需要由Activity开始向下分发传递,但是当传递到MyViewGroupB后,由于其消费了ACTION_DOWN,事件将不再继续向下分发,而是直接由MyViewGroupB分发给自己的onTouchEvent进行继续处理。事件处理也不再向上移交。

4、将MyViewGroupB的onInterceptTouchEvent和onTouchEvent的返回值均改为true。这意味着既拦截,又消费。

输出结果如下:

I/Activity: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupA: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupA: onInterceptTouchEvent_ACTION_DOWNI/MyViewGroupB: dispatchTouchEvent_ACTION_DOWNI/MyViewGroupB: onInterceptTouchEvent_ACTION_DOWNI/MyViewGroupB: onTouchEvent_ACTION_DOWNI/Activity: dispatchTouchEvent_ACTION_MOVEI/MyViewGroupA: dispatchTouchEvent_ACTION_MOVEI/MyViewGroupA: onInterceptTouchEvent_ACTION_MOVEI/MyViewGroupB: dispatchTouchEvent_ACTION_MOVEI/MyViewGroupB: onTouchEvent_ACTION_MOVEI/Activity: dispatchTouchEvent_ACTION_UPI/MyViewGroupA: dispatchTouchEvent_ACTION_UPI/MyViewGroupA: onInterceptTouchEvent_ACTION_UPI/MyViewGroupB: dispatchTouchEvent_ACTION_UPI/MyViewGroupB: onTouchEvent_ACTION_UP

结合输出结果,总结如下:

当触发ACTION_DOWN的时候,依然从Activity开始向下传递,当到达MyViewGroupB的是,因为在onInterceptTouchEvent进行了拦截操作,因此不再继续向下分发传递,而是交由MyViewGroupB的onTouchEvent进行处理消费。MyViewGroupB的onTouchEvent返回的是true,说明它决定对ACTION_DOWN进行处理,因此事件也就不再移交上层处理。当触发ACTION_MOVE和ACTION_UP的时候,事件还是从Activity开始向下传递,当到达MyViewGroupB的时候,由于之前进行了拦截操作,因此,MyViewGroupB直接将事件分发给自己的onTouchEvent进行处理,不在向下分发传递。事件处理也不再向上层移交。

三、文末

事件分发机制就犹如数学的定理是一样道理的,只有记住定理,才能在具体应用中具体分析,有人可能不知道在什么地方会用到,如果你做的项目中,比如有一个控件点击不能反应,那么就有可能是事件分发的结果。

至于对源码的分析可能内容比较复杂,内容也多;需要我这套framework进阶学习全套笔记的可以前往私信我;发送“核心笔记”或“手册”即可领取!整理不易!记得点赞分享哦!

郑重声明:本文内容及图片均整理自互联网,不代表本站立场,版权归原作者所有,如有侵权请联系管理员(admin#wlmqw.com)删除。
(0)
用户投稿
上一篇 2022年7月9日
下一篇 2022年7月9日

相关推荐

  • 最低仅1599元,这5款搭载“旗舰芯”的高性能千元机,闭眼随便买

    您在阅读前请点击上面的“关注”二字,后续会第一时间为您提供更多有价值的相关内容,感谢您的支持。 虽然说每个人都喜欢性能强悍的高端机,但是销量最高的手机,还是性价比最高的千元机,现在…

    2022年6月28日
  • 抖音不要盲目开通橱窗(抖音怎么卖东西)

    随着互联网的普及及5G时代的发展,使用抖音的用户越来越多。有人的地方就有商机,因此无数人都将自己的目光放到抖音营销上,那么今天小编就来给大家介绍几个抖音上常见的营销方法,大家可以参…

    2022年11月13日
  • 华为mate40pro再度回归!麒麟9000+4400毫安,售价依然“6”开头

    其实我不是很明白这波操作的目的是什么?前段时间华为mate40pro下架的消息,想必比较关心华为的粉丝都知道,大家一度以为这次华为mate50或许真要来了的时候,华为在618的最后…

    2022年6月20日
  • HarmonyOS是否已挑起国产手机系统的大梁?

    “今天我们鸿蒙HarmonyOS 3.0来了。”7月27日晚间,华为常务董事、终端BG CEO、智能汽车解决方案BU CEO余承东正式宣布HarmonyOS 3.0正式亮相。 此次…

    2022年8月20日
  • 《宝可梦朱紫》怎么交换精灵?交换精灵方法一览

    宝可梦朱紫中玩家可以和好友或者npc进行交换宝可梦获得自己没有的宝可梦,很多玩家想知道宝可梦朱紫怎么交换精灵,下面就带来宝可梦朱紫交换精灵方法一览,感兴趣的小伙伴不要错过,希望能帮…

    2022年11月23日
  • 魅族19再创辉煌,黑科技一体屏+天玑9000,渲染图很明确

    在国内手机市场,魅族是一家很特别的手机厂商。与小米所走的性价比路线不同,魅族更注重用户体验。外观设计非常细致,内置Flyme OS与MIUI、Color OS等系统不同,没有任何广…

    2022年7月3日
  • 央行开展5000亿元MLF操作 利率不变

    为维护银行体系流动性合理充裕,10月17日中国人民银行开展5000亿元中期借贷便利(MLF)操作和20亿元公开市场逆回购操作,充分满足了金融机构需求,中标利率均与前次持平。 央行官…

    2022年10月18日
  • 每月2100的工资如何理财?

    钱多有钱多的理财,钱少有钱少的方法。每月2100元,就不能理财了吗?10年前,我的工资才900块一个月。照样理财。 每月2100元,不多,如果能够累计一年,多的不算,至少可存1万,…

    2022年4月19日
  • Pallapay 在迪拜以现金或银行转账方式出售USDT Tether的革命性方法

    迪拜是一个由摩天大楼、港口和海滩组成的城市,在这里,大企业和寻求阳光的旅游业同时存在。由于其庞大的外籍人口,它给人的感觉就像一个中东大熔炉,而且气氛普遍很宽容。 迪拜和阿布扎比正在…

    2022年6月14日
  • JSON 对象的这些操作和使用场景你知道多少?

    JSON 对象对应前端的同学一定不陌生,使用地非常频繁和常见,在这里顺便总结一下对 JSON 对象的操作和使用场景。 1、添加 JSON 的属性 可通过 . 或 [] 的方式对 J…

    2022年6月28日

联系我们

联系邮箱:admin#wlmqw.com
工作时间:周一至周五,10:30-18:30,节假日休息