Android触摸事件传递机制,这一篇就够了
整个触摸事件牵涉到的是,Activity,View,ViewGroup三者的传递机制。
这个触摸事件就是从外层往内层一层层的传递。
整个传递机制,分为3个步骤:分发,拦截,和消费。
1. 触摸事件的类型
事件类型是MotionEvent类:看下最新的sdk29的源码,一堆的Action,我们常用的其实就3个,ACTION_DOWN,ACTION_MOVE,ACTION_UP。
Down和Up事件,是触摸过程中一定会发生的事件。Move事件看具体的情况。
2.事件传递的顺序
从外层往内层一层层的传递,外层是那个,内层是那个?
我们从android系统的角度看。事件首先传给的肯定是Activity,因为Activity的UI布局,事物逻辑,作为底层来说,不知道,也并不关心。
所以只要把事件送到Activity层,就可以。
同样的道理,事件先送到ViewGroup,然后在送到View。这样从外层到内层的过程。
其实每一层都会对触摸事件有响应,这个事件是有具体的业务逻辑决定的,所以我们需要一些操作来控制这个事件的过程。
比如点击事件,假设点击了某个按钮,这样它的父ViewGroup要不要处理,页面要不要处理,从表面来看他们都被点击了,从实际人们的经验来看,
就是View被点击了,不需要ViewGroup和Activtiy后面2个事件触发。
Activity->ViewGroup->ViewGroup...->View 整个事件流程传递就是这样。
View->ViewGroup...->ViewGroup->Activity 这个就是事件的消费过程。
3.事件传递的3个阶段
所以android把触摸事件传递分为3个步骤:分发,拦截,和消费。
3.1 分发
分发:事件分发对应着dispatchTouchEvent方法,在Android中,所有的触摸事件都是这个方法来分发的。
public boolean dispatchTouchEvent(MotionEvent ev)
3.1.1 Activity分发
分发是触摸事件首先到达。
- 返回true,表示事件被消耗掉。不需要分发下去。消费了事件,后续的事件会继续传递到这里。
- 返回false,表示事件被废弃掉。因为dispatchTouchEvent是总入口,所以就算是false,后续事件还会出现。
- 返回值是super.dispatchTouchEvent,事件继续传递下去,这个是整个事件传递的中枢,从源码看,一切从这来开始,也从这里结束。
如果view hierarchy都不处理这个touch事件,就由Activity的onTouch事件来处理。可见上面第二节的结论是正确的。最先接收,最后处理。
/**
* Called to process touch screen events. You can override this to
* intercept all touch screen events before they are dispatched to the
* window. Be sure to call this implementation for touch screen events
* that should be handled normally.
*
* @param ev The touch screen event.
*
* @return boolean Return true if this event was consumed.
*/
public boolean dispatchTouchEvent(MotionEvent ev) {
if (ev.getAction() == MotionEvent.ACTION_DOWN) {
onUserInteraction();
}
if (getWindow().superDispatchTouchEvent(ev)) {
return true;
}
return onTouchEvent(ev);
}
3.1.2VIewGroup分发
- 返回true,表示事件被消耗掉。不需要分发下去。消费了事件,后续的事件会继续传递到这里。
- 返回false,表示事件被废弃掉。后续的其他事件不会被传递到这里。
- 返回值是super.dispatchTouchEvent,事件继续传递下去,这个是整个事件传递的中枢,从源码看,ViewGroup的分发流程分以下几步
有拦截,拦截了touch事件,就走ViewGroup自己的onTouchEvent。
没有拦截,就找到触摸点的位置,上面是否有View,有就调view.dispatchTouchEvent,如果没有,就直接交给ViewGroup处理。
这里还有一点,如果action_down已经处理了,后面action_up 和 action_move都使用相同的处理方式。
3.1.3 View的分发
View的分发,也许会觉得奇怪,分发给谁?
从源码看,2个途径,onTouch listener,或者onTouchEvent。前者优先级必后者高。
- 返回true,表示事件被消耗掉。不需要分发下去。消费了事件,后续的事件会继续传递到这里。并且这里会认为事件被View消费掉了,所以上层的VIewGroup不会收到要处理onTouchEvent的事件。
- 返回false,表示事件被废弃掉。后续的其他事件不会被传递到这里。而且可以理解为,我这个View不想处理这个事件,所以ViewGroup会触发onTouchEvent事件。
- 返回值是super.dispatchTouchEvent
ListenerInfo li = mListenerInfo;
if (li != null && li.mOnTouchListener != null
&& (mViewFlags & ENABLED_MASK) == ENABLED
&& li.mOnTouchListener.onTouch(this, event)) {
result = true;
}
if (!result && onTouchEvent(event)) {
result = true;
}
onTouch是否处理,return true,就表面被第三方处理了,不需要onTouchEvent来处理。不然就是onTouchEvent处理。
3.2 拦截
事件拦截对应onInterceptTouchEvent。事件是从外向内传递的,当事件在分发的过程中,要不要拦截,
public boolean onInterceptTouchEvent(MotionEvent ev)
有些情况下需要子View来响应,有些情况需要当前ViewGroup来处理。这个方法Activity没有,只有ViewGroup才有。
这个简单来说就是,是否需要分发事件给到子View,true就是不分发。
3.3 消费
其实dispatchTouchEvent也有消费的概念在里面。
如果说分发的流程,是从外层到内层,
消费的流程正好相反,从内层到外层。
public boolean onTouchEvent(MotionEvent event)
- 返回true,消费事件,不在向外层传递
-返回false,不消费事件,向外层传递
-返回spuer.onTouchEvent,这里会判断是否是click事件。只有这个写法才会触发click事件。
4. 流程图
5.总结
流程很复杂,每个部分返回不同的true/fase/super都会产生不同的流程结果。
事件分发流程:
从外层到内层分发,
从内层到外层消费。
dispatchTouchEvent 和onTouchEvent,返回true,代表由自己处理这个事情。阻断后面的流程。
dispatchTouchEvent 和onTouchEvent,返回false,表示不由自己来处理这个事件,流程交由上一层级来处理。
dispatchTouchEvent 和onTouchEvent,返回spuer,就是上面的流程。
更多内容:demanmath
公共号:
Android触摸事件传递机制,这一篇就够了的更多相关文章
- Android触摸事件传递机制
简单梳理一下Android触摸事件传递机制的知识点. 一.View与ViewGroup的关系 View和ViewGroup二者的继承关系如下图所示: View是Android中最基本的一种UI组件,它 ...
- 初识Android触摸事件传递机制
前言 今天总结的一个知识点是Andorid中View事件传递机制,也是核心知识点,相信很多开发者在面对这个问题时候会觉得困惑,另外,View的另外一个难题滑动冲突,比如在ScrollView中嵌套Li ...
- Android Touch事件传递机制引发的血案
尊重原创:http://blog.csdn.net/yuanzeyao/article/details/38942135 关于Android Touch事件传递机制我之前也写过两篇文章,自觉得对Tou ...
- Android touch 事件传递机制
前言: (1)在自定义view的时候经常会遇到事件拦截处理,比如在侧滑菜单的时候,我们希望在侧滑菜单里面有listview控件,但是我们希望既能左右滑动又能上下滑动,这个时候就需要对触摸的touch事 ...
- iOS 和 Android 触摸事件传递
先看文章,写得很好 ios 触摸事件传递 http://www.cnblogs.com/Quains/p/3369132.html 另外一篇 http://blog.csdn.net/yongyinm ...
- Android 的事件传递机制,详解
Android 的事件传递机制,详解 前两天和一个朋友聊天的时候.然后说到事件传递机制.然后让我说的时候,忽然发现说的不是非常清楚,事实上Android 的事件传递机制也是知道一些,可是感觉自己知道的 ...
- Android Touch事件传递机制 二:单纯的(伪生命周期)
转载于:http://blog.csdn.net/yuanzeyao/article/details/38025165 在前一篇文章中,我主要讲解了Android源码中的Touch事件的传递过程,现在 ...
- Android Touch事件传递机制 一: OnTouch,OnItemClick(监听器),dispatchTouchEvent(伪生命周期)
ViewGroup View Activity dispatchTouchEvent 有 有 有 onInterceptTouchEvent 有 无 无 onTouchEvent 有 有 有 例 ...
- 【转】Android TouchEvent事件传递机制
Android TouchEvent事件传递机制 事件机制参考地址: http://www.cnblogs.com/sunzn/archive/2013/05/10/3064129.html ht ...
随机推荐
- 【学习笔鸡】整体二分(P2617 Dynamic Rankings)
[学习笔鸡]整体二分(P2617 Dynamic Rankings) 可以解决一些需要树套树才能解决的问题,但要求询问可以离线. 首先要找到一个具有可二分性的东西,比如区间\(k\)大,就很具有二分性 ...
- 洛谷P1147 连续自然数和 题解 枚举
题目链接:https://www.luogu.com.cn/problem/P1147 题目大意: 给你一个数 \(M\) ,求有多少对连续自然数对之和为 \(M\),输出这列连续自然数对的首项和末项 ...
- DispatcherServlet的url-pattern尽量不要配置为"/*"
DispatcherServlet的url-pattern尽量不要配置为"/*" 原因 当Dispatcher的url配置为"/*"时,会把tomcat的web ...
- gcc 命令详解
1. gcc -E source_file.c-E,只执行到预编译.直接输出预编译结果. 2. gcc -S source_file.c -S,只执行到源代码到汇编代码的转换,输出汇编代码. 3. g ...
- 2018铁人三项测评题 IOS99
下面这一部分是我从网上复制过来的, 2.IOS 解题链接:http://ctf4.shiyanbar.com/web/IOS/index.php 这题页面中提示系统升级到了IOS99,我们可以想到修改 ...
- java小项目之:植物大战僵尸,这个僵尸有点冷!内附素材源码
Java小项目之:植物大战僵尸! <植物大战僵尸>是由PopCap Games开发的一款益智策略类单机游戏,于2009年5月5日发售,这款游戏可谓是无人不知无人不晓. 在我身边,上到40岁 ...
- hdfs/hbase 程序利用Kerberos认证超过ticket_lifetime期限后异常
问题描述 业务需要一个长期运行的程序,将上传的文件存放至HDFS,程序启动后,刚开始一切正常,执行一段时间(一般是一天,有的现场是三天),就会出现认证错误,用的JDK是1.8,hadoop-clien ...
- VS 超级好用的 Ctrl E E
C# Interactive 推荐!!! 先看我怎么用的:随便创建一个类 有些编译期的的值不知道查文档又太麻烦怎么办?自己写个控制台测试咩?试试 C# 交互 罢, 选中这个类ctrl EE 然后输入 ...
- APICloud联合腾讯云推出“云主机解决方案“,各种福利等你拿
为了帮助开发者一站式打通云.开发.运维全流程服务,更全面提供基于自身业务情况的云服务器.数据库.存储等基础设施服务,APICloud联合腾讯云重磅推出“云主机解决方案“.开发者可通过控制台简单清晰的购 ...
- CSP-S rp++
心无旁骛 认真思考 努力骗分(哈哈) I Love CSP! 反正像我这种大菜鸟也考不了多少 尽力打 本次考试期望 day1 100 70-100 30-? day2 100 ? ? 总:300-? ...