Android触摸事件传递机制

在安卓开发中会时常碰到滑动冲突的情况,比如有一个水平滑动的 ViewPager 里面有 Fragment,
Fragment 里面有水平滑动的 ViewGroup,当你的手指在屏幕水平滑动时,就会发生冲突,那怎么
解决这种冲突呢,首先开发者需要对 Activity、ViewGroup、View 的事件传递机制有个很好的认识。

触摸事件的类型

触摸事件对应的是MotionEvent类,事件类型主要有三种:
- ACTION_DOWN: 用户手指按下屏幕,标志着一次触摸事件开始。
- ACTION_MOVE: 用户手指按下屏幕之后,在松开之前,如果滑动的距离超过了一定值,则会
被判定为 ACITON_MOVE 操作。
- ACTION_UP: 用户手指离开屏幕,标志着一次完整触摸事件结束。

一次屏幕触摸操作,一定会产生 ACTION_DOWNACTION_UP 两种事件,如果用户按下
之后手指右移动一定的距离,则会发生 ACTION_MOVE 事件,如果只是点了一下,那么则不
会发生。

触摸事件传递

触摸事件的传递有三个阶段:

  • 分发(dispatch): 所有的事件都是通过这个方法分发,对应的方法原型为
    public boolean dispatchTouchEvent(MotionEvent event)
    如果方法返回true,则当前视图消费事件,不在分发,super.dispatchTouchEvent 表示继续
    分发事件,如果当前视图是 ViewGroup,则会调用 interceptTouchEvent

  • 拦截(intercept): 这个方法中 ViewGroup 中,顾名思义,就是对触摸事件的拦截,方法原型为
    public boolean onInterceptHoverEvent(MotionEvent event)

  • 消费(consume: 事件最终需要被消费,对应的方法原型为
    public boolean onTouchEvent(MotionEvent event)
    如果方法返回 true,则说明当前视图可以消费事件,不会再向上传递事件;返回值为 false,则
    表示消费事件,继而向上层父视图传递事件,让父视图来处理。

触摸事件传递的三个角色:

  • Activity: 拥有 dispatchTouchEvent 和 onTouchEvent 两个方法。
  • ViewGroup: 拥有 dispatchTouchEvent 、onInterceptTouchEvent 和 onTouchEvent 三个方法。
  • View: 拥有 dispatchTouchEvnet 和 onTouchEvent 两个方法。

触摸事件传递三种类型

  • View 不消费事件
    DOWN 事件不消费,onTouchEvnet 返回 false,父视图 onTouchEvent 方法执行,接下来的
    MOVE、UP 事件也不会再传递给该 View,而是由消费了 DOWN 事件的父视图接收

Android触摸事件传递机制

  • View 消费事件
    View 消费了 DOWN 事件,onTouchEvent 返回 true,父视图的 onTouchEvent 方法不会被执行,
    接下来的 MOVE、UP 事件继续传递给该 View,除非传递过程被拦截

![View 消费事件](https://charmingw.github.io/images/MotionEvent_interested _view.png)

  • 事件被拦截
    DOWN 事件被 View 消费了,但接下来的 MOVE、up 事件被拦截了,所以 View 无法消费整套触摸事件
    Android触摸事件传递机制