二.ViewGroup事件分發源碼分析之事件處理

關鍵源碼(對dispatchTouchEvent()的重要部分)

片段一 是否攔截事件

 if (actionMasked == MotionEvent.ACTION_DOWN|| mFirstTouchTarget != null) {
    final boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0;
     if (!disallowIntercept) {
        intercepted = onInterceptTouchEvent(ev);
        ev.setAction(action); // restore action in case it was changed
     } else {
         intercepted = false;
     }
   } else {
        // There are no touch targets and this action is not an initial down
        // so this view group continues to intercept touches.
         intercepted = true;
    }

片段二 如果ViewGroup不攔截事件


if (!canceled && !intercepted) {
 for(遍歷child){
  if (dispatchTransformedTouchEvent(ev, false, child, idBitsToAssign)) {
        newTouchTarget = addTouchTarget(child, idBitsToAssign);
        break;
  }
 }
}

片段三 有無Child處理事件

if (mFirstTouchTarget == null) {
  // No touch targets so treat this as an ordinary view.
          handled = dispatchTransformedTouchEvent(ev, canceled, null,TouchTarget.ALL_POINTER_IDS);
}else{
// dispatched to it.  Cancel touch targets if necessary.
         TouchTarget predecessor = null;
         TouchTarget target = mFirstTouchTarget;
         while (target != null) {
                final TouchTarget next = target.next;
                if (alreadyDispatchedToNewTouchTarget && target == newTouchTarget) {
                        handled = true;
                } else {
                    final boolean cancelChild = resetCancelNextUpFlag(target.child)
                                || intercepted;//攔截時,cancelChild為true
                    if (dispatchTransformedTouchEvent(ev, cancelChild,
                                target.child, target.pointerIdBits)) {
                            handled = true;
                    }
                    if (cancelChild) {
                        if (predecessor == null) {
                               mFirstTouchTarget = next;//會將mFirsrtTouchTarget置空
                        } else {
                                predecessor.next = next;
                        }
                        target.recycle();
                        target = next;
                        continue;
                        }
                }
                    predecessor = target;
                    target = next;
           }
           
           if (canceled
                    || actionMasked == MotionEvent.ACTION_UP
                    || actionMasked == MotionEvent.ACTION_HOVER_MOVE) {
                resetTouchState();
            } 
            
        }
    }
    

disallowIntercept 默認值為false,通過viewGroup.requestDisallowInterceptTouchEvent(boolean)進行設置。本文只討論disallowIntercept 為false的情況。
mFirstTouchTarget==null 沒有child處理事件,有兩種可能

  1. ViewGroup攔截了事件
  2. ViewGroup沒有攔截了事件,但是沒有child處理事件

mFirstTouchTarget!=null 有child處理事件

  1. ViewGrop沒有攔截事件,且有child消費了事件

intercepted表示是否攔截事件,值為true表示攔截事件。

對ACTION_DOWN事件的處理

  1. 由onInterceptTouchEvent()判斷是否攔截(片段一)
  2. 如果ViewGroup攔截事件,則遍歷child,將事件交給child處理,如果有child消耗了事件,則調用addTouchTarget()給mFirstTouchTarget賦值。(片段二)
  3. 如果child不消費ACTION_DWON(mFirstTouchTarget==null),則ViewGroup處理該ACTION_DOWN事件。(片段三)

對ACTION_MOVE事件的處理

  1. 如果有child消費正在處理事件( mFirstTouchTarget != null),則由onInterceptTouchEvent()判斷是否攔截,否則攔截事件(片段一)
  2. 如果ViewGroup攔截ACTION_MOVE事件,則遍歷child,將事件交給child處理,如果有child消費了事件,則調用addTouchTarget()給mFirstTouchTarget賦值。(片段二)
  3. 如果沒有child消費正在處理事件(mFirstTouchTarget==null),則ViewGroup處理該ACTION_MOVE事件,否則 如果ViewGroup攔截了ACTION_MOVE,則mFirstTouchTarget設置為空。(片段三)

對ACTOIN_UP事件的處理

  1. 如果有child正在處理事件( mFirstTouchTarget != null),則由onInterceptTouchEvent()判斷是否攔截,否則攔截事件(片段一)
  2. 如果ViewGroup攔截ACTION_UP事件,則遍歷child,將事件交給child處理,如果有child消耗了事件,則調用addTouchTarget()給mFirstTouchTarget賦值。(片段二)
  3. 如果沒有child正在消費事件,則ViewGroup處理該ACTION_MOVE事件,
    否則
    1. 如果ViewGroup攔截了ACTION_UP,則mFirstTouchTarget設置為空。
    2. 重置TouchState狀態

總結

ACTION_DWON作為起始動作,所以要判斷是否攔截

ACTION_MOVE,ACTION_UP作為后續事件
如果child在處理事件,ViewGroup要判斷是否攔截事件;
如果沒有child在處理事件,ViewGroup自然攔截事件。

如果ViewGroup不攔截事件,事件交給child處理。

如果沒有Child正在處理事件,則事件交給ViewGroup處理;
如果有Child正在處理事件,此時如果ViewGroup攔截了當前的事件,則要將ViewGroup標志位正在處理事件即mFirstTouchTarget設置為空。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容