在Android系统中,当你点击App某个按钮从你按下到抬起到底发生了什么?
要想解释这个问题首先需要了解Android的几个事件类型。
Android 的事件类型分为以下四种:
1、MotionEvent.ACTION_DOWN 手指按下时触发。
2、MotionEvent.ACTION_MOVE 手指移动时触发。
3、MotionEvent.ACTION_UP 手指抬起时触发。
4、MotionEvent.ACTION_CANCEL 事件意外关闭时触发。
MotionEvent类中包含了点击事件的类型是按下还是移动还是抬起、点击位置的坐标、以及其他等等信息。
前面三种很好理解。当你点击某个view时会先触发MotionEvent.ACTION_DOWN,当你手指开始移动的时候就会触发MotionEvent.ACTION_MOVE,当你手指抬起就会触发MotionEvent.ACTION_UP。
一组完整的事件应该是以Down开始以Up结束。
当你点击某个View的时候,事件的传递顺序是
Activity-->ViewGroup-->View
在Android里事件是由最外围依次向上分发的。
那么Android是如何分发事件的呢?
这就不得不提到Android的事件分发机制,事件分发机制其实就是Android为了解决触摸事件冲突而设立的机制,他主要是以下三个方法:
dispatchTouchEvent 事件分发
onInterceptTouchEvent 事件拦截
onTouchEvent 事件消费
要想理解这三个方法我们带入实际案例来讲解。
这种情况下如果你点击可点击的View,这个事件会被View拦截消费掉,如果点击的是不可点击的View那么事件将会被他的ViewGroup消费掉。这时Android本身的事件处理,那么Android是如何做到这一切的?
就是靠的事件分发机制,当你点击子View的时候,系统会调用Activity中的根View的onInterceptTouchEvent询问他是否拦截事件,如果不进行拦截那么系统就会依次往上的调用上一层的onInterceptTouchEvent直到找到需要拦截事件的View然后调用它的onTouchEvent去消费掉此事件,如果所有的View都不对事件进行拦截,那么就会进入第二个流程,从上到下依次从调用每层View的onTouchEvent看看是否消费该事件如过都不进行消费事件最终会回到Activity。
那么我们如果要给自己的View做点击事件处理怎么做呢,很简单重写View的onTouchEvent事件拦截MotionEvent.ACTION_DOWN事件返回true就可以了。
onTouchEvent当拦截到Down事件后那么这组事件的后续的MOVE、UP都会被这个View所消费。
那么还有一种情况,比如在一个ListView中,我的每个条目上都有一个可点击的View那么我手指按在这个可以点击的View上时然后划动这个ListView,此时应该是列表去相应这个move事件,那么这种跨View的事件处理要怎么做呢。
因为系统在调用onTouchEvent前会依次往上的调用上一层的onInterceptTouchEvent询问他是否拦截这个事件,因为ListView上层的View是需要消费Down事件的,所以我们不可以在ListView中直接拦截下Down事件,否则上层View将无法接收到事件。
onInterceptTouchEvent默认是返回false的也就是不拦截事件,那么我们就可以重写ListView的onInterceptTouchEvent方法中返回true来拦截下ACTION_MOVE事件(此时接受到Down事件的View就会触发ACTION_CANCEL意外关闭事件),那么这个事件的后续就会交给ListView的onTouchEvent来进行消费从而实现划动。
那么dispatchTouchEvent呢
这三个方法的关系是onInterceptTouchEvent和onTouchEvent是在dispatchTouchEvent中执行的。
一张图来表示:
(图片来源于网络)
说了这么多现在我们回到开头提的问题,当你点击App某个可点击按钮从你按下到抬起到底发生了什么?
当你点击这个View之后系统会最先交给activity会来处理,然后传递给PhoneWindow,再传递给DecorView,然后传给顶层ViewGroup,最后才是最上层的View。在你按下时会从根View依次向上的调用onInterceptTouchEvent,如果都不消费就从上到下依次调用onTouchEvent看是否消费,按下时就会触发这个可点击按钮的onTouchEvent然后触发ACTION_DOWN 到ACTION_UP 一系列事件如果你设置了点击事件那么在ACTION_UP 后还会调用你的onClick方法。