事件分发机制
Android 中与 Touch 事件相关的方法包括:dispatchTouchEvent(MotionEvent ev)、onInterceptTouchEvent(MotionEvent ev)、onTouchEvent(MotionEvent ev);能够响应这些方法的控件包括:ViewGroup 及其子类、Activity。方法与控件的对应关系如下表所示:
Touch 事件相关方法 | 方法功能 | ViewGroup | Activity |
public boolean dispatchTouchEvent(MotionEvent ev) | 事件分发 | Yes | Yes |
public boolean onInterceptTouchEvent(MotionEvent ev) | 事件拦截 | Yes | No |
public boolean onTouchEvent(MotionEvent ev) | 事件响应 | Yes | Yes |
从这张表中我们可以看到 ViewGroup 及其子类对与 Touch 事件相关的三个方法均能响应,而 Activity 对 onInterceptTouchEvent(MotionEvent ev) 也就是事件拦截不进行响应。
另外需要注意的是 View 对 dispatchTouchEvent(MotionEvent ev) 和 onInterceptTouchEvent(MotionEvent ev) 的响应的前提是可以向该 View 中添加子 View,如果当前的 View 已经是一个最小的单元 View(比如 TextView),那么就无法向这个最小 View 中添加子 View,也就无法向子 View 进行事件的分发和拦截,所以它没有 dispatchTouchEvent(MotionEvent ev) 和 onInterceptTouchEvent(MotionEvent ev),只有 onTouchEvent(MotionEvent ev)。
一、Touch 事件分析
//自己写的,有什么意见可以留言,希望和大家一起讨论
当dispatchTouchEvent onInterceptTouchEvent onTouchEvent 三个方法都返回默认的super. 的时候,事件会一直传递的最终的子view控件上 触发onClient事件
▐
事件分发:public boolean dispatchTouchEvent(MotionEvent ev)
Touch 事件发生时 Activity 的 dispatchTouchEvent(MotionEvent ev) 方法会以
隧道方式
(从根元素依次往下传递直到最内层子元素或在中间某一元素中由于某一条件停止传递)
将事件传递给最外层 View 的 dispatchTouchEvent(MotionEvent ev) 方法,并由该 View 的 dispatchTouchEvent(MotionEvent ev) 方法对事件进行分发。dispatchTouchEvent 的事件分发逻辑如下:
- 如果 return true,事件会分发给当前 View 并由 dispatchTouchEvent 方法进行消费,同时事件会停止向下传递;
- 摁下
- 03-26 19:49:42.249 15512-15512/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 19:49:42.249 15512-15512/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
抬起
03-26 19:49:43.189 15512-15512/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 19:49:43.189 15512-15512/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
- 如果 return false,事件分发分为两种情况:
- 如果当前 View 获取的事件直接来自 Activity,则会将事件返回给 Activity 的 onTouchEvent 进行消费;
摁下
03-26 19:51:34.319 17031-17031/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 19:51:34.319 17031-17031/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
抬起
03-26 19:51:34.319 17031-17031/com.example.my_event_fenfa D/Event: Activity ------- > onTouchEvent
03-26 19:51:37.199 17031-17031/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 19:51:37.199 17031-17031/com.example.my_event_fenfa D/Event: Activity ------- > onTouchEvent
- 如果当前 View 获取的事件来自外层父控件,则会将事件返回给父 View 的 onTouchEvent 进行消费。
这里我没有试,如果方便的话 大家可以自己敲个Demo试一下
- 如果返回系统默认的 super.dispatchTouchEvent(ev),事件会自动的分发给当前 View 的 onInterceptTouchEvent 方法。
- 摁下
- 这里先不要注意打那么多Log,只要注意Demo走了当前View的onInterceptTouchEvent就可以了
- 03-26 19:54:41.659 18953-18953/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 19:54:41.659 18953-18953/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 19:54:41.659 18953-18953/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 19:54:41.659 18953-18953/com.example.my_event_fenfa D/Event: MyView ------- > onTouchEvent
抬起
03-26 19:54:43.379 18953-18953/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 19:54:43.379 18953-18953/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 19:54:43.379 18953-18953/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 19:54:43.379 18953-18953/com.example.my_event_fenfa D/Event: MyView ------- > onTouchEvent
03-26 19:54:43.379 18953-18953/com.example.my_event_fenfa D/Event: View ------- > onClick
▐
事件拦截:public boolean onInterceptTouchEvent(MotionEvent ev)
在外层 View 的 dispatchTouchEvent(MotionEvent ev) 方法返回系统默认的 super.dispatchTouchEvent(ev) 情况下,事件会自动的分发给当前 View 的 onInterceptTouchEvent 方法。onInterceptTouchEvent 的事件拦截逻辑如下:
- 如果 onInterceptTouchEvent 返回 true,则表示将事件进行拦截,并将拦截到的事件交由当前 View 的 onTouchEvent 进行处理;
- 摁下
- 这里也不要在意那么多Log 只要看到事件走了当前View的onTouchEvent方法
- 03-26 20:00:09.319 24201-24201/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:00:09.319 24201-24201/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:00:09.319 24201-24201/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 20:00:09.319 24201-24201/com.example.my_event_fenfa D/Event: LinearLayout ------- > onTouchEvent
03-26 20:00:09.319 24201-24201/com.example.my_event_fenfa D/Event: Activity ------- > onTouchEvent
抬起
03-26 20:00:10.449 24201-24201/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEve nt
03-26 20:00:10.449 24201-24201/com.example.my_event_fenfa D/Event: Activity ------- > onTouchEvent
- 如果 onInterceptTouchEvent 返回 false,则表示将事件放行,当前 View 上的事件会被传递到子 View 上,再由子 View 的 dispatchTouchEvent 来开始这个事件的分发;
- 摁下
- 注意我这里在布局中只写了一个TextView控件 所以事件会传递给TextView的onTouchEvent事件 而TextView是没有拦截事件和分发事件 大家如果不满意的话可以自己再加有个ListView试一下,这里我就不做过多步骤了
- 03-26 20:04:08.879 27684-27684/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:04:08.879 27684-27684/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:04:08.879 27684-27684/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 20:04:08.879 27684-27684/com.example.my_event_fenfa D/Event: MyView ------- > onTouchEvent
抬起
03-26 20:04:10.029 27684-27684/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:04:10.029 27684-27684/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:04:10.029 27684-27684/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 20:04:10.029 27684-27684/com.example.my_event_fenfa D/Event: MyView ------- > onTouchEvent
03-26 20:04:10.039 27684-27684/com.example.my_event_fenfa D/Event: View ------- > onClick
- 如果 onInterceptTouchEvent 返回 super.onInterceptTouchEvent(ev),和false返回的结果一样
▐
事件响应:public boolean onTouchEvent(MotionEvent ev)
在 dispatchTouchEvent 返回 super.dispatchTouchEvent(ev) 并且 onInterceptTouchEvent 返回 true 的情况下 onTouchEvent 会被调用。onTouchEvent 的事件响应逻辑如下:
- 如果事件传递到当前 View 的 onTouchEvent 方法,而该方法返回了 false,那么这个事件会从当前 View 向上传递,并且都是由上层 View 的 onTouchEvent 来接收,如果传递到上面的 onTouchEvent 也返回 false,这个事件就会“消失”,而且接收不到下一次事件。
- 摁下
03-26 20:26:29.259 12650-12650/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:26:29.259 12650-12650/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:26:29.259 12650-12650/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 20:26:29.259 12650-12650/com.example.my_event_fenfa D/Event: LinearLayout ------- > onTouchEvent
03-26 20:26:29.259 12650-12650/com.example.my_event_fenfa D/Event: Activity ------- > onTouchEvent
抬起
03-26 20:26:30.149 12650-12650/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:26:30.149 12650-12650/com.example.my_event_fenfa D/Event: Activity ------- > onTouchEvent
- 如果返回了 true 则会接收并消费该事件。
- 摁下
- 03-26 20:23:03.889 10765-10765/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:23:03.889 10765-10765/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:23:03.889 10765-10765/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 20:23:03.889 10765-10765/com.example.my_event_fenfa D/Event: LinearLayout ------- > onTouchEvent
抬起
03-26 20:23:04.939 10765-10765/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:23:04.939 10765-10765/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:23:04.939 10765-10765/com.example.my_event_fenfa D/Event: LinearLayout ------- > onTouchEvent
- 如果返回 super.onTouchEvent(ev) 默认处理事件的逻辑和返回 false 时相同。
到这里,与 Touch 事件相关的三个方法就分析完毕了。下面的内容会通过各种不同的的测试案例来验证上文中三个方法对事件的处理逻辑。
最后呢 补充一个小知识点
onTouchEvent和onClick
一般情况下 onTouchEvent 会比Onclick 执行的早 而当你在onTouchEvent 返回值为true时 是不会执行onClick方法的 但是false也没有执行 只有默认执行了
以下 是我打的Log 大家可以看下 同时也可以发表一下建议 一起讨论
true是没有执行onclick方法
摁下
03-26 20:40:37.309 26045-26045/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:40:37.309 26045-26045/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:40:37.309 26045-26045/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 20:40:37.309 26045-26045/com.example.my_event_fenfa D/Event: MyView ------- > onTouchEvent
抬起
03-26 20:40:37.969 26045-26045/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:40:37.969 26045-26045/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:40:37.969 26045-26045/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 20:40:37.969 26045-26045/com.example.my_event_fenfa D/Event: MyView ------- > onTouchEvent
false 也没有执行onClick方法
摁下
03-26 20:41:27.319 26643-26643/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:41:27.319 26643-26643/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:41:27.319 26643-26643/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 20:41:27.319 26643-26643/com.example.my_event_fenfa D/Event: MyView ------- > onTouchEvent
03-26 20:41:27.319 26643-26643/com.example.my_event_fenfa D/Event: LinearLayout ------- > onTouchEvent
03-26 20:41:27.319 26643-26643/com.example.my_event_fenfa D/Event: Activity ------- > onTouchEvent
抬起
03-26 20:41:27.889 26643-26643/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:41:27.889 26643-26643/com.example.my_event_fenfa D/Event: Activity ------- > onTouchEvent
默认
return super.onTouchEvent(event);
这里执行了onClient方法
摁下
03-26 20:43:04.919 28076-28076/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:43:04.919 28076-28076/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:43:04.919 28076-28076/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 20:43:04.919 28076-28076/com.example.my_event_fenfa D/Event: MyView ------- > onTouchEvent
抬起
03-26 20:43:05.689 28076-28076/com.example.my_event_fenfa D/Event: Activity ------- > dispatchTouchEvent
03-26 20:43:05.689 28076-28076/com.example.my_event_fenfa D/Event: LinearLayout ------- > dispatchTouchEvent
03-26 20:43:05.689 28076-28076/com.example.my_event_fenfa D/Event: LinearLayout ------- > onInterceptTouchEvent
03-26 20:43:05.689 28076-28076/com.example.my_event_fenfa D/Event: MyView ------- > onTouchEvent
03-26 20:43:05.699 28076-28076/com.example.my_event_fenfa D/Event: View ------- > onClick