最近在回顾android touch事件的分发机制,发现我对他的理解还是有点偏差,故写了一个demo测试一下,加深理解。
分发事件中主要有三个方法:
- dispatchTouchEvent:(activity、ViewGroup、View共有)是处理触摸事件分发,事件(多数情况)是从Activity的dispatchTouchEvent开始的。执行dispatchTouchEvent(ev),事件向下分发。
- onInterceptTouchEvent:(ViewGroup独有)默认返回false,返回true表示拦截。
onTouchEvent:(activity、ViewGroup、View共有)是View中提供的方法,ViewGroup也有这个方法。view中默认返回true,表示消费了这个事件。
几张原理图,摘自网上
图1.ACTION_DOWN都没被消费
图2-1.ACTION_DOWN被View消费了
图2-2.后续ACTION_MOVE和UP在不被拦截的情况下都会去找VIEW
图3.后续的被拦截了
注:这里有一个需要地方需要提醒一下,就是在ViewGroup中onInterceptTouchEvent返回true了,然后onTouchEvent没有消费Touch事件,那么这个Touch事件会回给Activity,那么接下来的一系列事件会直接交给Activity去消费,而不会再dispatch给ViewGroup。
举个栗子:
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="com.example.dispatcheventdemo.MainActivity" >
<com.example.dispatcheventdemo.MyLiearLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<com.example.dispatcheventdemo.MyButton
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/hello_world" />
</com.example.dispatcheventdemo.MyLiearLayout>
</RelativeLayout>
- RelativeLayout里面嵌套了一个MyLiearLayout,MyLiearLayout里面嵌套了一个MyButton
- LinearLayout是继承与ViewGroup的,我在MyLinearLayout中的onInterceptTouchEvent中返回true
- 当MyLinearLayout接收到ACTION_DOWN Event,如果我不消费它,即onTouchEvent方法中返回false,这时Event会被送给Activity去消费
- 在下一次收到ACTION_UP Event时,这个事件不再会分发给MyLinearLayout,而是直接被activity消费了
图4ACTION_DOWN一开始就被拦截
android中的Touch事件都是从ACTION_DOWN开始的:
单手指操作:ACTION_DOWN—ACTION_MOVE—-ACTION_UP
多手指操作:ACTION_DOWN—ACTION_POINTER_DOWN—ACTION_MOVE–ACTION_POINTER_UP—ACTION_UP.