Android Touch事件分发机制学习

Android中的事件分为按键事件和触摸事件。
Touch事件是由一个ACTION_DOWN,n个ACTION_MOVE,一个ACTION_UP组成onClick,onLongClick,onScroll等事件。
Android 中与 Touch 事件相关的方法及其对应的作用:
dispatchTouchEvent(MotionEvent ev) 事件分发
onInterceptTouchEvent(MotionEvent ev) 事件拦截
onTouchEvent(MotionEvent ev) 事件响应

Activity:dispatchTouchEvent(MotionEvent ev)、onTouchEvent(MotionEvent ev)
View: dispatchTouchEvent(MotionEvent ev)、onTouchEvent(MotionEvent ev)
ViewGroup继承View,上面方法都是public的,所以他拥有上面所有方法,ViewGroup类里面新添加了方法onInterceptTouchEvent(MotionEvent ev)。

为了比较清楚的了解执行流程,我做了如图所示的小demo:
技术分享
这里我们要清楚一个东西,控件分为两种:一种是继承View不能包含其他控件的控件;另一种是继承ViewGroup可以包含其他控件的控件,暂且称为容器控件,比如ListView,GridView,LinearLayout等。
上面demo的Button就是继承自View的,浅绿色区域是一个横向的LinearLayout,而他是继承自ViewGroup的。

先概述一下demo的制作:
里面有三个类:
MyActivity extends Activity
MyButton extends Button
MyLinearLayout extends LinearLayout
在MyActivity中:

public class MyActivity extends Activity
{
    private static final String TAG = MyActivity.class.getSimpleName();

    @Override
    protected void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    public boolean dispatchTouchEvent(MotionEvent ev) {
        Log.i(TAG, MotionEvent.actionToString(ev.getAction()) + " dispatchTouchEvent()");
        return super.dispatchTouchEvent(ev);
    }

    public boolean onTouchEvent(MotionEvent event) {
        Log.i(TAG, MotionEvent.actionToString(event.getAction()) + " onTouchEvent()");
        return super.onTouchEvent(event);
    }
}

MyButton和MyLinearLayout十分相似,MyLinearLayout只多了一个onInterceptTouchEvent方法,这里贴出MyButton的代码,如果需要MyLinearLayout的代码可以在后面直接下载,看完整的代码。
MyButton

public class MyButton extends Button
{
    private static final String TAG = MyButton.class.getSimpleName();

    public MyButton(Context context)
    {
        super(context);
    }

    public MyButton(Context context, AttributeSet attrs)
    {
        super(context, attrs);
    }   

    public MyButton(Context context, AttributeSet attrs, int defStyleAttr)
    {
        super(context, attrs, defStyleAttr);
        // TODO Auto-generated constructor stub
    }

    public boolean dispatchTouchEvent(MotionEvent ev) {
        Log.i(TAG, MotionEvent.actionToString(ev.getAction()) + " dispatchTouchEvent()");
        return super.dispatchTouchEvent(ev);
    }

    public boolean onTouchEvent(MotionEvent event) {
        Log.i(TAG, MotionEvent.actionToString(event.getAction()) + " onTouchEvent()");
        return super.onTouchEvent(event);
    }
}

Java代码都是非常简单的
因为我们要使用自己定义的控件,所以在xml布局代码中是这样的:

<per.lee.eventdispatchtest.MyLinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:background="#00FF00"  >

    <per.lee.eventdispatchtest.MyButton
        android:id="@+id/button1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Button" />

</per.lee.eventdispatchtest.MyLinearLayout>

log的显示为 Tag显示:具体类, Text显示:具体动作 + 调用方法名

点击MyButton(深绿色区域),出现的结果是:
技术分享

点击MyLinearLayout 非MyButton区域(浅绿色区域),结果是:
技术分享

点击MyActivity 非MyLinearLayout区域(空白区域),结果是:
技术分享

从以上Log可以看出:
一个Touch事件会从外层逐层分发到内层,直到事件处理为止(return true),表面现象分析完毕了,后面我找机会研究一下源码,内部具体到底是怎么处理的。

这个小demo的代码下载地址:EventDispatchTest.zip

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。