• 【Android


    参考资料:

    View事件分发:http://blog.csdn.net/pi9nc/article/details/9281829

    ViewGroup事件分发:http://blog.csdn.net/guolin_blog/article/details/9153747

    1      概述

      Android中的布局是按树形结构层级排列的,根布局往往是一个ViewGroup,如LinearLayout,其中包裹一些子View或子ViewGroup,子ViewGroup中又包裹着它的子View或子ViewGroup,依此类推。

      每当触摸了屏幕上的一个“东西”,都会触发这个“东西”的dispatchTouchEvent()方法,负责事件的分发。这个方法是从它的父类View或ViewGroup中继承来的。

      因此,我们想要了解Android中的事件分发机制,就需要分别了解View的事件分发机制和ViewGroup的事件分发机制。

    2      View事件分发

      一般地,我们给一个控件添加事件,通常会调用setOnTouchListener()和setOnClickListener()这两个方法,前者是为了监听手指的按下、抬起、滑动等细节操作,而后者只是为了监听点击控件的操作。从上面的概述可以知道,无论是哪种操作,都是通过dispatchTouchEvent()方法来分配的。

      在View类的dispatchTouchEvent()方法中,先判断这个View是否注册了OnTouchListener监听器,如果注册了,则回调其中的onTouch()方法;如果没有注册,则调用View类中的onTouchEvent()方法。

      OnTouchListener中的onTouch()方法是回调给程序员编写的,这个方法有一个布尔类型的返回值,如果返回true则终止本次事件的分发,否则依然会调用View类中的onTouchEvent()方法。对应的部分源码如下:

    if (li != null && li.mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED 
            && li.mOnTouchListener.onTouch(this, event)) {
        result = true;
    }
    if (!result && onTouchEvent(event)) {
        result = true;
    }

      在View类的onTouchEvent()方法中判断这个View是否是可点击的(clickable),如果是则判断这个View是否注册了OnClickListener,如果注册了就调用其中的onClick()方法。对应的部分源码如下:

    public boolean onTouchEvent(MotionEvent event) {
        if (((viewFlags & CLICKABLE) == CLICKABLE || (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE) 
                 || (viewFlags & CONTEXT_CLICKABLE) == CONTEXT_CLICKABLE) {
            performClick();
            return true;
        }
        return false;
    }
    public void performClick() {
        if (li != null && li.mOnClickListener != null) {
            li.mOnClickListener.onClick(this);
        }
    }
     从上面的代码中可以看出,如果一个控件时clickable的,那么onTouchEvent()就一定会返回true,否则返回false。

      因此,当我们为一个控件同时设置了setOnTouchListener()和setOnClickListener()这两个方法时,会先回调OnTouchListener中的onTouch()事件,当onTouch()返回true时,才会再回调OnClickListener中的onClick()事件。

      这里还需要注意的一点是,给一个控件设置了touch事件后,每次点击它时,都会触发一系列的ACTION_DOWN、ACTION_MOVE和ACTION_UP等事件。在此过程中,如果某一个ACTION返回了false,那么后面的一系列ACTION就不会再执行了。简单的说,当dispatchTouchEvent()在进行事件分发的时候,只有前一个ACTION返回true,才会触发后一个ACTION。

    3      ViewGroup事件分发

      当一个事件到达一个ViewGroup的时候,会递归的在它的树级结构中查找这个事件的处理者:依次遍历这个ViewGroup中的所有子View和子ViewGroup,调用它们的dispatchTouchEvent()方法,如果判断可以处理这个事件则返回true,否则返回false;当一个ViewGroup中的所有子View和子ViewGroup的dispatchTouchEvent()方法都返回false,则这个ViewGroup的dispatchTouchEvent()也返回false。

      ViewGroup中有一个方法叫做onInterceptTouchEvent(),它标识这个ViewGroup是否要拦截其下的View的事件,默认返回的是false,表示不拦截。我们可以通过在自定义ViewGroup中重写这个方法设置其返回值为true,或者通过requestDisallowInterceptTouchEvent()方法获取拦截权限。

      当ViewGroup的onInterceptTouchEvent()方法返回true时,所有属于这个ViewGroup中某个View的事件都会被这个ViewGroup拦截,即这个ViewGroup就成为了这个事件的处理者来处理这个事件。

    4      综述

      每当触摸了一次屏幕,根布局(ViewGroup)就会发出一次递归搜索,在其下层的布局或控件树中寻找拦截这次事件的控件,如果有控件或布局对这次事件进行了拦截,则表示这个控件或布局就是这次事件的处理者。

      无论最终获取这个事件处理权的是View还是ViewGroup(以下统称为控件),都调用它的dispatchTouchEvent()方法对事件进行处理。

      总之,Android中的事件分发机制的流程图如下图所示:

    最后,用文字来描述一下完整的事件分发机制:

    1、当我们在手机屏幕上点击了一下,就是在当前界面的Activity上点击了一下,就会触发当前Activity的dispatchTouchEvent()方法;

    2、在Activity的dispatchTouchEvent()方法中,通过getWindow()方法找到当前Activity的Window对象(PhoneWindow对象),然后调用Window对象的superDispatchTouchEvent()方法;

    3、在Window的superDispatchTouchEvent()方法中,通过getDecorView()方法找到当前Activity中的顶层View(顶层View一般都是ViewGroup),开始View的事件传递;

    4、顶层View调用dispatchTouchEvent()方法,开始实际的事件分发,具体流程如下:

      (1)判断这个ViewGroup是否拦截事件,即它的onInterceptTouchEvent()方法是否返回true,如果返回true则事件会交给这个ViewGroup处理,即它的onTouchEvent()方法会被调用;

      (2)如果这个ViewGroup不拦截事件,则递归调用其下所有View的dispatchTouchEvent()方法,找到事件处理者的View;

      (3)先判断目标View是否注册了OnTouchListener,如果注册了,则调用OnTouchListener中的onTouch()方法,验证其返回值;

      (4)如果onTouch()方法返回false,表示View还没有将这个事件消费掉,会继续执行View的onTouchEvent()方法;如果onTouch()返回true,则不会继续执行onTouchEvent()方法,直接结束事件传递;

      (5)如果这个View重写了onTouchEvent()方法,则会先调用重写的内容;

      (6)如果给这个View设置了OnClickListener,则会在onTouchEvent()中的代码执行之后,调用OnClickListener中onClick()方法中的代码;

    5、在上面的事件传递过程中,如果有一个View接管了事件,那么事件就不会继续向下传递,将这个View的信息层层上报到顶层View之后,结束事件传递;

    6、如果顶层View发现底下的所有View都没有接管这个事件,那么这个事件最终会回传给Activity处理,即Activity的onTouchEvent()方法会被调用。

  • 相关阅读:
    在AE中通过SDE添加图层(转)
    上一视图下一视图功能实现(C#+SuperMap Objects)
    [Python入门及进阶笔记]Python基础内置函数小结
    Javascript小球
    64位centos下安装python的PIL模块
    [Python入门及进阶笔记]Python基础集合小结
    C语言的那些小秘密之【链表(二)】
    巴斯卡三角形
    [Python入门及进阶笔记]Python基础数字处理相关模块
    C语言的那些小秘密之【链表(一)】
  • 原文地址:https://www.cnblogs.com/itgungnir/p/6211363.html
Copyright © 2020-2023  润新知