• Android-事件分发(ViewGroup)


    http://blog.csdn.net/guolin_blog/article/details/9153747

    http://blog.csdn.net/lmj623565791/article/details/39102591

    上一篇讲了view的事件分发,这一篇主要是viewGroup

    首先我们来探讨一下,什么是ViewGroup?它和普通的View有什么区别?

    顾名思义,ViewGroup就是一组View的集合,它包含很多的子View和子VewGroup,是Android中所有布局的父类或间接父类,像LinearLayout、RelativeLayout等都是继承自ViewGroup的。但ViewGroup实际上也是一个View,只不过比起View,它多了可以包含子View和定义布局参数的功能。ViewGroup继承结构示意图如下所示:

     

    可以看到,我们平时项目里经常用到的各种布局,全都属于ViewGroup的子类。

    Android-事件分发机制框架概述可以知道,viewgroup返回true是消费,super是判断拦截,false是返回dispatchTouchEvent

    源码细节可以在这里看:http://blog.csdn.net/lmj623565791/article/details/39102591

    整个逻辑是这样的:

    1、ACTION_DOWN中,ViewGroup捕获到事件,然后判断是否拦截,如果没有拦截,则找到包含当前x,y坐标的子View,赋值给mMotionTarget,然后调用 mMotionTarget.dispatchTouchEvent

    2、ACTION_MOVE中,ViewGroup捕获到事件,然后判断是否拦截,如果没有拦截,则直接调用mMotionTarget.dispatchTouchEvent(ev)

    3、ACTION_UP中,ViewGroup捕获到事件,然后判断是否拦截,如果没有拦截,则直接调用mMotionTarget.dispatchTouchEvent(ev)

    当然了在分发之前都会修改下坐标系统,把当前的x,y分别减去child.left 和 child.top ,然后传给child;

    1、如何拦截

    上面的总结都是基于:如果没有拦截;那么如何拦截呢?

    复写ViewGroup的onInterceptTouchEvent方法:

    public boolean onInterceptTouchEvent(MotionEvent ev)  
        {  
            int action = ev.getAction();  
            switch (action)  
            {  
            case MotionEvent.ACTION_DOWN:  
                //如果你觉得需要拦截  
                return true ;   
            case MotionEvent.ACTION_MOVE:  
                //如果你觉得需要拦截  
                return true ;   
            case MotionEvent.ACTION_UP:  
                //如果你觉得需要拦截  
                return true ;   
            }  
              
            return false;  
        }

    默认是不拦截的,即返回false;如果你需要拦截,只要return true就行了,这要该事件就不会往子View传递了,并且如果你在DOWN retrun true ,则DOWN,MOVE,UP子View都不会捕获事件;如果你在MOVE return true , 则子View在MOVE和UP都不会捕获事件。

    原因很简单,当onInterceptTouchEvent(ev) return true的时候,会把mMotionTarget 置为null ; 

    2、如何不被拦截

    如果ViewGroup的onInterceptTouchEvent(ev) 当ACTION_MOVE时return true ,即拦截了子View的MOVE以及UP事件;

    此时子View希望依然能够响应MOVE和UP时该咋办呢?

    Android给我们提供了一个方法:requestDisallowInterceptTouchEvent(boolean) 用于设置是否允许拦截,我们在子View的dispatchTouchEvent中直接这么写:

    public boolean dispatchTouchEvent(MotionEvent event)  
        {  
            getParent().requestDisallowInterceptTouchEvent(true);    
            int action = event.getAction();  
      
            switch (action)  
            {  
            case MotionEvent.ACTION_DOWN:  
                Log.e(TAG, "dispatchTouchEvent ACTION_DOWN");  
                break;  
            case MotionEvent.ACTION_MOVE:  
                Log.e(TAG, "dispatchTouchEvent ACTION_MOVE");  
                break;  
            case MotionEvent.ACTION_UP:  
                Log.e(TAG, "dispatchTouchEvent ACTION_UP");  
                break;  
      
            default:  
                break;  
            }  
            return super.dispatchTouchEvent(event);  
        } 

    getParent().requestDisallowInterceptTouchEvent(true);  这样即使ViewGroup在MOVE的时候return true,子View依然可以捕获到MOVE以及UP事件。

    从源码也可以解释:

    ViewGroup MOVE和UP拦截的源码是这样的:

    if (!disallowIntercept && onInterceptTouchEvent(ev)) {  
                final float xc = scrolledXFloat - (float) target.mLeft;  
                final float yc = scrolledYFloat - (float) target.mTop;  
                mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT;  
                ev.setAction(MotionEvent.ACTION_CANCEL);  
                ev.setLocation(xc, yc);  
                if (!target.dispatchTouchEvent(ev)) {  
                    // target didn't handle ACTION_CANCEL. not much we can do  
                    // but they should have.  
                }  
                // clear the target  
                mMotionTarget = null;  
                // Don't dispatch this event to our own view, because we already  
                // saw it when intercepting; we just want to give the following  
                // event to the normal onTouchEvent().  
                return true;  
            }

    当我们把disallowIntercept设置为true时,!disallowIntercept直接为false,于是拦截的方法体就被跳过了~

    注:如果ViewGroup在onInterceptTouchEvent(ev)  ACTION_DOWN里面直接return true了,那么子View是木有办法的捕获事件的~~~

    4、如果没有找到合适的子View

    我们的实例,直接点击ViewGroup内的按钮,当然直接很顺利的走完整个流程;

    但是有两种特殊情况

    1、ACTION_DOWN的时候,子View.dispatchTouchEvent(ev)返回的为false ; 

    如果你仔细看了,你会注意到ViewGroup的dispatchTouchEvent(ev)的ACTION_DOWN代码是这样的

    if (child.dispatchTouchEvent(ev)) 
    {  
              // Event handled, we have a target now.  
              mMotionTarget = child;  
              return true;  
    }

    只有在child.dispatchTouchEvent(ev)返回true了,才会认为找到了能够处理当前事件的View,即mMotionTarget = child;

    但是如果返回false,那么mMotionTarget 依然是null

    mMotionTarget 为null会咋样呢?

    其实ViewGroup也是View的子类,如果没有找到能够处理该事件的子View,或者干脆就没有子View;

    那么,它作为一个View,就相当于View的事件转发了~~直接super.dispatchTouchEvent(ev);

    源码是这样的:

    final View target = mMotionTarget;  
           if (target == null) {  
               // We don't have a target, this means we're handling the  
               // event as a regular view.  
               ev.setLocation(xf, yf);  
               if ((mPrivateFlags & CANCEL_NEXT_UP_EVENT) != 0) {  
                   ev.setAction(MotionEvent.ACTION_CANCEL);  
                   mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT;  
               }  
               return super.dispatchTouchEvent(ev);  
           }

    我们没有一个能够处理该事件的目标元素,意味着我们需要自己处理~~~就相当于传统的View~

    2、那么什么时候子View.dispatchTouchEvent(ev)返回的为true

    如果你仔细看了上篇博客,你会发现只要子View支持点击或者长按事件一定返回true~~

    源码是这样的

    if (((viewFlags & CLICKABLE) == CLICKABLE ||  
                    (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)) {     
                 return true ;

    5、总结

    关于代码流程上面已经总结过了~

    1、如果ViewGroup找到了能够处理该事件的View,则直接交给子View处理,自己的onTouchEvent不会被触发;

    2、可以通过复写onInterceptTouchEvent(ev)方法,拦截子View的事件(即return true),把事件交给自己处理,则会执行自己对应的onTouchEvent方法

    3、子View可以通过调用getParent().requestDisallowInterceptTouchEvent(true);  阻止ViewGroup对其MOVE或者UP事件进行拦截;

    好了,那么实际应用中能解决哪些问题呢?

    比如你需要写一个类似slidingmenu的左侧隐藏menu,主Activity上有个Button、ListView或者任何可以响应点击的View,你在当前View上死命的滑动,菜单栏也出不来;因为MOVE事件被子View处理了~ 你需要这么做:在ViewGroup的dispatchTouchEvent中判断用户是不是想显示菜单,如果是,则在onInterceptTouchEvent(ev)拦截子View的事件;自己进行处理,这样自己的onTouchEvent就可以顺利展现出菜单栏了~~

  • 相关阅读:
    Java-技术专区-JVM调优常用参数
    Alibaba-技术专区-Dubbo的SPI应用与原理
    Java-技术专区-深入理解SPI机制
    Alibaba-技术专区-Dubbo SPI扩展点加载机制
    Java-技术专区-线程基础
    Java-技术专区-精巧好用的DelayQueue
    Alibaba-技术专区-Sentinel流量降级监控
    Java-技术专区-Java线程池ThreadPoolExecutor的理解
    SpringCloud-技术专区-SpringCloud组件原理
    机器码和伪指令
  • 原文地址:https://www.cnblogs.com/qlky/p/6681903.html
Copyright © 2020-2023  润新知