View的事件分发流程

View中dispatchTouchEvent()的源码分析如下:

public boolean dispatchTouchEvent(MotionEvent event) {   

   if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED &&  mOnTouchListener.onTouch(this, event)) {        

  return true;      

}      

return onTouchEvent(event);  

}


从上面可以看出:只有以下三个条件都为真,dispatchTouchEvent()才返回true;否则执行onTouchEvent(event)方法。

�第一个条件:mOnTouchListener != null

//mOnTouchListener是在View类下setOnTouchListener方法里赋值的

public void setOnTouchListener(OnTouchListener l) {

//即只要我们给控件注册了Touch事件,mOnTouchListener就一定被赋值(不为空)   

 mOnTouchListener = l;

 }

mOnTouchListener 是在View类下setOnTouchListener方法中赋值的,只要在控件中注册了Touch事件,那么mOnTouchListener一定会不为null。 

‚第二个条件:mViewFlags & ENABLED_MASK) == ENABLED

该条件是判断当前控件是否是enable的(很多控件默认都是enable的),因此该条件恒定为true。

第三个条件:mOnTouchListener.onTouch(this, event)

回调注册Touch事件的的onTouch()方法。

button.setOnTouchListener(new View.OnTouchListener() {  

  @Override  

  public boolean onTouch(View view, MotionEvent motionEvent) {  

      Log.d("kate", "onTouch: ");       

     return false;   

  }

 });

若onTouch()方法返回true,那么上述三个条件全部成立,从而整个方法返回true。

若onTouch()方法返回false,那么就会走 onTouchEvent()方法。

接下来继续分析onTouchEvent()方法,其源码如下:

public boolean onTouchEvent(MotionEvent event) {       ……    

       //如果该控件是可以点击的就会进入到下两行的switch判断中去;  

       if (((viewFlags & CLICKABLE) == CLICKABLE ||   (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)) {   

     //如果当前的事件是抬起手指,则会进入到MotionEvent.ACTION_UP这个case当中。     

     switch (event.getAction()) {              

          case MotionEvent.ACTION_UP:           

             ……                 

                   if (!focusTaken) {                   

                         if (!post(mPerformClick)) {                             

                                             performClick();                    

                        }                  

                  }               

                       ……             

                 break;            

           case MotionEvent.ACTION_DOWN:             

             ……             

             break;            

               ……  

}  //如果该控件是可以点击的,就一定会返回true      

  return true;     

 }  

//如果该控件是不可以点击的,就一定会返回false 

   return false;

 }

当触发MotionEvent.ACTION_UP事件,就会调用perfromClick()函数。函数代码如下:

public boolean performClick() {    

    if (mOnClickListener != null) {     

         playSoundEffect(SoundEffectConstants.CLICK);      

         mOnClickListener.onClick(this);       

        return true;     

      }     

 return false;

 }

当mOnClickListener变量不为null时,就会调用 mOnClickListener.onClick(this)函数。那么什么时候该变量会赋值呢?

public void setOnClickListener(OnClickListener l) {     

 if (!isClickable()) {      

    setClickable(true);   

   }    

  mOnClickListener = l;

 }

当控件调用setOnClickListener()函数注册一个点击事件的时候,就会给mOnClickListener赋值,并调用onClick()函数。

总结:

1、事件传递到View层时,若该view注册了onTouchListener,那么onTouchListener的onTouch会被回调。这时view事件处理顺序取决于onTouch的返回值(默认false),若返回false,那么就会调用该view的onTouchEvent()方法。反之,返回true,那么就不会调用View的onTouchEvent()方法。由此可见,onTouchListener的调用优先级要比onTouchEvent高。

2、事件传递到View层时,若该view注册了onClickListener,只要当up事件触发的时候,onTouchEvent方法才会调用该onClickListener的onClick方法。由此可见,onClickListener其优先级最低,处于事件尾端。

3、给view设置onTouchListener的onTouch方法的返回值很重要,其返回值决定该view的onTouchEvent()的方法是否执行。而onTouchEvent()方法是否执行也很重要,若执行,那么当up时间出发时,就会调用onClickListener的onClick方法。

你可能感兴趣的:(View的事件分发流程)