Android View和ViewGroup事件分发机制源码分析
来源:互联网 发布:linux有几个版本 编辑:程序博客网 时间:2024/05/16 19:18
请尊重个人劳动成果,转载请注明出处,谢谢!
http://blog.csdn.net/amazing7/article/details/51274481在Android开发中会经常遇到滑动冲突(比如ScrollView与ListView的嵌套)的问题,需要我们深入的了解android事件响应机制才能解决。
1.涉及到事件响应的常用方法构成
用户在手指与屏幕接触过程中通过MotionEvent对象产生一系列事件,它有四种状态:
MotionEvent.ACTION_DOWN :手指按下屏幕的瞬间(一切事件的开始)
MotionEvent.ACTION_MOVE :手指在屏幕上移动
MotionEvent.ACTION_UP :手指离开屏幕瞬间
MotionEvent.ACTION_CANCEL :取消手势,一般由程序产生,不会由用户产生
Android中的事件onClick, onLongClick,onScroll, onFling等等,都是由许多个Touch事件构成的(一个ACTION_DOWN, n个ACTION_MOVE,1个ACTION_UP)。
android 事件响应机制是先 分发(先由外部的View接收,然后依次传递给其内层的最小View)再 处理 (从最小View单元(事件源)开始依次向外层传递。)的形式实现的。
复杂性表现在:可以控制每层事件是否继续传递(分发和拦截协同实现),以及事件的具体消费(事件分发也具有事件消费能力)。
2.android事件处理涉及到的三个重要函数
事件分发:public boolean dispatchTouchEvent(MotionEvent ev)
事件拦截:public boolean onInterceptTouchEvent(MotionEvent ev)
该方法为ViewGroup独有。事件响应:public boolean onTouchEvent(MotionEvent ev)
ViewGroup未实现,而是调用父类view的该方法。
因为在View和ViewGroup中稍有异同,所有分别结合其源码在下面对这些函数做详细说明。
注:以下为Android 5.1.1的源码
3.View源码分析
Android中ImageView、textView、Button等继承于View但没有重写的dispatchTouchEvent方法,所以都用的View的该方法进行事件分发。
View中dispatchTouchEvent函数部分源码:
- 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
- 33
- 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
- 33
关于注册回调接口ListenerInfo 处的三个条件判断:
(1)查看是否给button设置了OnTouchListener()事件;
(2)控件是否Enable;(控件默认都是enable的)
(3)view里面实现的OnTouchListener监听里的onTouch()方法是否返回true;
如果条件都满足,则该事件被消耗掉,不再进入onTouchEvent中处理。否则将事件将交给onTouchEvent方法处理。
- 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
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 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
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
①在dispatchTouchEvent顺序调用mOnTouchListener和onTouchEvent,mOnTouchListener的onTouch返回值可中断onTouchEvent,在onTouchEvent中回调onClick接口。
②只有我们注册OnTouchListener时重写的
onTouch()方法中返回false —> 执行onTouchEvent方法 —> 导致onClick()回调方法执行
返回true —> onTouchEvent方法不执行 —> 导致onClick()回调方法不会执行
4.ViewGroup源码分析
Android中诸如LinearLayout等的五大布局控件,都是继承自ViewGroup,而ViewGroup本身是继承自View,所以ViewGroup的事件处理机制对这些控件都有效。
dispatchTouchEvent部分源码:
- 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
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 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
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
当有监听到事件时,首先由Activity进行捕获,进入事件分发处理流程。(因为activity没有事件拦截,ViewGroup有)会将事件传递给最外层ViewGroup的dispatchTouchEvent(MotionEvent ev)方法,该方法对事件进行分发。
return true :表示该ViewGroup内部消化掉了所有事件。
return false :事件在本层不再继续进行分发,并交由上层控件的onTouchEvent方法进行消费(如果本层控件已经是Activity,那么事件将被系统消费或处理)。
如果事件分发返回系统默认的 super.dispatchTouchEvent(ev),事件将分发给本层的事件拦截onInterceptTouchEvent 方法进行处理(父类默认)。
onInterceptTouchEvent源码:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 1
- 2
- 3
- 4
- 5
- 6
- 7
return true :表示将事件进行拦截,并将拦截到的事件交由本层控件 的 onTouchEvent 进行处理;
return false :则表示不对事件进行拦截,事件得以成功分发到子View。并由子View的dispatchTouchEvent进行处理。
如果返回super.onInterceptTouchEvent(ev),默认表示不拦截该事件(父类默认返回false),和return false一样。
ViewGroup是继承自View,在ViewGroup中并没有复写onTouchEvent函数,而是直接调用父类的onTouchEvent。
在dispatchTouchEvent(事件分发)返回super.dispatchTouchEvent(ev)并且onInterceptTouchEvent事件拦截返回true的情况下,那么事件会传递到本层的onTouchEvent方法中,该方法对事件进行响应。
在onTouchEvent方法中:
如果return true,表示onTouchEvent处理完事件后消费了此次事件。此时事件终结;
如果return fasle,则表示不响应事件,那么该事件将会不断向上层View的onTouchEvent方法传递,直到某个View的onTouchEvent方法返回true,如果到了最顶层View还是返回false,那么认为该事件不消耗,则在同一个事件系列中,当前View无法再次接收到事件,该事件会交由Activity的onTouchEvent进行处理;
如果return super.dispatchTouchEvent(ev),则表示不响应事件,结果与return false一样。
从以上过程中可以看出,dispatchTouchEvent无论返回true还是false,事件都不再进行分发,只有当其返回super.dispatchTouchEvent(ev),才表明其具有向下层分发的愿望,但是是否能够分发成功,则需要经过事件拦截onInterceptTouchEvent的审核。事件是否向上传递处理是由onTouchEvent的返回值决定的。
(图来自网络)
1、dispatchTouchEvent作用:决定事件是否由onInterceptTouchEvent来拦截处理。
返回super.dispatchTouchEvent时,由onInterceptTouchEvent来决定事件的流向
返回false时,会继续分发事件,自己内部只处理了ACTION_DOWN
返回true时,不会继续分发事件,自己内部处理了所有事件(ACTION_DOWN,ACTION_MOVE,ACTION_UP)2、onInterceptTouchEvent作用:拦截事件,用来决定事件是否传向子View
返回true时,拦截后交给自己的onTouchEvent处理
返回false时,交给子View来处理3、onTouchEvent作用:事件最终到达这个方法
返回true时,内部处理所有的事件,换句话说,后续事件将继续传递给该view的onTouchEvent()处理
返回false时,事件会向上传递,由onToucEvent来接受,如果最上面View中的onTouchEvent也返回false的话,那么事件就会消失
5.总结
如果ViewGroup找到了能够处理该事件的View,则直接交给子View处理,自己的onTouchEvent不会被触发;
可以通过复写onInterceptTouchEvent(ev)方法,拦截子View的事件(即return true),把事件交给自己处理,则会执行自己对应的onTouchEvent方法。
子View可以通过调用getParent().requestDisallowInterceptTouchEvent(true); 阻止ViewGroup对其MOVE或者UP事件进行拦截;
一个点击事件产生后,它的传递过程如下:
Activity->Window->View。顶级View接收到事件之后,就会按相应规则去分发事件。如果一个View的onTouchEvent方法返回false,那么将会交给父容器的onTouchEvent方法进行处理,逐级往上,如果所有的View都不处理该事件,则交由Activity的onTouchEvent进行处理。如果某一个View开始处理事件,如果他不消耗ACTION_DOWN事件(也就是onTouchEvent返回false),则同一事件序列比如接下来进行ACTION_MOVE,则不会再交给该View处理。
ViewGroup默认不拦截任何事件。
诸如TextView、ImageView这些不作为容器的View,一旦接受到事件,就调用onTouchEvent方法,它们本身没有onInterceptTouchEvent方法。正常情况下,它们都会消耗事件(返回true),除非它们是不可点击的(clickable和longClickable都为false),那么就会交由父容器的onTouchEvent处理。
点击事件分发过程如下 dispatchTouchEvent—->OnTouchListener的onTouch方法—->onTouchEvent–>OnClickListener的onClick方法。也就是说,我们平时调用的setOnClickListener,优先级是最低的,所以,onTouchEvent或OnTouchListener的onTouch方法如果返回true,则不响应onClick方法…
- Android View和ViewGroup事件分发机制源码分析
- Android View和ViewGroup事件分发机制源码分析
- Android View 和 ViewGroup 事件分发机制
- android之View和ViewGroup事件分发机制分析(一)(View的事件分发机制)
- Android View 事件分发机制源码详解(ViewGroup篇)
- Android View 事件分发机制 源码解析(ViewGroup篇)
- Android中ViewGroup、View事件分发机制源码分析总结(雷惊风)
- Android View 事件分发机制 && Android ViewGroup 事件分发机制 源码解析 --总结
- android中view 和 viewgroup事件分发机制
- Android事件分发机制源码分析之ViewGroup篇
- Android中View和ViewGroup事件分发拦截机制完美分析
- Android View、ViewGroup 事件分发机制(一)
- Android View、ViewGroup 事件分发机制(二)
- Android ViewGroup/View 事件分发机制详解
- ViewGroup和View的事件分发机制
- ViewGroup事件分发机制源码分析
- Android开发-事件分发机制实验分析ViewGroup、View事件分发,结合职责链模式
- Android事件分发机制源码分析下----ViewGroup事件分发分析
- matlab练习程序(PSNR)
- 笔记:css命名相关规范
- 第一行代码学习2(4)
- 枚举和结构体
- java HashMap相关
- Android View和ViewGroup事件分发机制源码分析
- Linux 下Mongodb的安装
- 《Map中HashMap与TreeMap的排序以及四种遍历方式》
- Android_ListView_有Header或Footer时onItemClick里的position的问题
- 浅谈MyBatis 之 集成SpringMVC(六)
- webpack 配置项选项详解
- html5 and css
- LDA主题模型和Gibbs Sampling 学习整理
- Flume与Kafka整合