(转)Android ViewGroup事件分发机制

三、关于拦截

一、如何拦截

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

复写ViewGroup的onInterceptTouchEvent方法:ide

 

[java]  view plain  copy
 
  1. @Override  
  2.     public boolean onInterceptTouchEvent(MotionEvent ev)  
  3.     {  
  4.         int action = ev.getAction();  
  5.         switch (action)  
  6.         {  
  7.         case MotionEvent.ACTION_DOWN:  
  8.             //若是你以为须要拦截  
  9.             return true ;   
  10.         case MotionEvent.ACTION_MOVE:  
  11.             //若是你以为须要拦截  
  12.             return true ;   
  13.         case MotionEvent.ACTION_UP:  
  14.             //若是你以为须要拦截  
  15.             return true ;   
  16.         }  
  17.           
  18.         return false;  
  19.     }  


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

 

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

二、如何不被拦截

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

此时子View但愿依然可以响应MOVE和UP时该咋办呢?orm

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

 

[java]  view plain  copy
 
  1. @Override  
  2.     public boolean dispatchTouchEvent(MotionEvent event)  
  3.     {  
  4.         getParent().requestDisallowInterceptTouchEvent(true);    
  5.         int action = event.getAction();  
  6.   
  7.         switch (action)  
  8.         {  
  9.         case MotionEvent.ACTION_DOWN:  
  10.             Log.e(TAG, "dispatchTouchEvent ACTION_DOWN");  
  11.             break;  
  12.         case MotionEvent.ACTION_MOVE:  
  13.             Log.e(TAG, "dispatchTouchEvent ACTION_MOVE");  
  14.             break;  
  15.         case MotionEvent.ACTION_UP:  
  16.             Log.e(TAG, "dispatchTouchEvent ACTION_UP");  
  17.             break;  
  18.   
  19.         default:  
  20.             break;  
  21.         }  
  22.         return super.dispatchTouchEvent(event);  
  23.     }  


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

 

 

从源码也能够解释:ip

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

 

[java]  view plain  copy
 
  1. if (!disallowIntercept && onInterceptTouchEvent(ev)) {  
  2.             final float xc = scrolledXFloat - (float) target.mLeft;  
  3.             final float yc = scrolledYFloat - (float) target.mTop;  
  4.             mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT;  
  5.             ev.setAction(MotionEvent.ACTION_CANCEL);  
  6.             ev.setLocation(xc, yc);  
  7.             if (!target.dispatchTouchEvent(ev)) {  
  8.                 // target didn't handle ACTION_CANCEL. not much we can do  
  9.                 // but they should have.  
  10.             }  
  11.             // clear the target  
  12.             mMotionTarget = null;  
  13.             // Don't dispatch this event to our own view, because we already  
  14.             // saw it when intercepting; we just want to give the following  
  15.             // event to the normal onTouchEvent().  
  16.             return true;  
  17.         }  


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

 

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

 

四、若是没有找到合适的子View

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

可是有两种特殊状况

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

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

 

[java]  view plain  copy
 
  1. if (child.dispatchTouchEvent(ev))  {  
  2.                               // Event handled, we have a target now.  
  3.                               mMotionTarget = child;  
  4.                               return true;  
  5.                           }  


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

 

可是若是返回false,那么mMotionTarget 依然是null

mMotionTarget 为null会咋样呢?

其实ViewGroup也是View的子类,若是没有找到可以处理该事件的子View,或者干脆就没有子View;

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

源码是这样的:

 

[java]  view plain  copy
 
  1. final View target = mMotionTarget;  
  2.        if (target == null) {  
  3.            // We don't have a target, this means we're handling the  
  4.            // event as a regular view.  
  5.            ev.setLocation(xf, yf);  
  6.            if ((mPrivateFlags & CANCEL_NEXT_UP_EVENT) != 0) {  
  7.                ev.setAction(MotionEvent.ACTION_CANCEL);  
  8.                mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT;  
  9.            }  
  10.            return super.dispatchTouchEvent(ev);  
  11.        }  


咱们没有一个可以处理该事件的目标元素,意味着咱们须要本身处理~~~就至关于传统的View~

 

二、那么何时子View.dispatchTouchEvent(ev)返回的为true

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

源码是这样的:

 

[java]  view plain  copy
 
  1.    
  2. if (((viewFlags & CLICKABLE) == CLICKABLE ||  
  3.                 (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)) {     
  4.              return true ;                                                                                                                                                                                                                                                                                                   }  

 

五、总结

 

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

一、若是ViewGroup找到了可以处理该事件的View,则直接交给子View处理,本身的onTouchEvent不会被触发;

二、能够经过复写onInterceptTouchEvent(ev)方法,拦截子View的事件(即return true),把事件交给本身处理,则会执行本身对应的onTouchEvent方法

三、子View能够经过调用getParent().requestDisallowInterceptTouchEvent(true);  阻止ViewGroup对其MOVE或者UP事件进行拦截;

 

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

好比你须要写一个相似slidingmenu的左侧隐藏menu,主Activity上有个Button、ListView或者任何能够响应点击的View,你在当前View上死命的滑动,菜单栏也出不来;由于MOVE事件被子View处理了~ 你须要这么作:在ViewGroup的dispatchTouchEvent中判断用户是否是想显示菜单,若是是,则在onInterceptTouchEvent(ev)拦截子View的事件;本身进行处理,这样本身的onTouchEvent就能够顺利展示出菜单

相关文章
相关标签/搜索