Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> 關於Android編程 >> Android View框架總結(二)View焦點

Android View框架總結(二)View焦點

編輯:關於Android編程

前言:View框架寫到第六篇,發現前面第二篇竟然沒有,然後事情是在微信公眾號發了,忘記在博客上更新,所以關注微信公眾號的應該都看過了,趁今天有時間遂補上。(PS:本篇文章中源碼均是android 6.0,請知曉)

本來之前說view下篇是寫onMeasure,onLayou,onDraw相關的,筆者做盒子開發,遙控器按鍵,碰到的都是焦點控制相關。所以先把焦點放到了onMeasure,onLayou,onDraw之前。以前踩過不少坑,最近公司同事做分享,遂下決心總結下,Agenda如下:

ViewRoot View的焦點 ViewGroup的焦點 父容器焦點的處理 失去焦點或清除焦點 焦點移動 FocusFinder查找焦點 總結

Android View焦點

Android焦點相關邏輯大部分都在都在View, ViewGroup和FocusFinder三個類中.

ViewRoot

View對象都有一個mParent變量(添加到ViewGroup後), 代指其父容器. 絕大部分View的mParent都是ViewGroup類型, 除了根節點. 一個Window中View根節點DecorView的mParent稱為ViewRoot, 在安卓4.0後ViewRoot對應ViewRootImpl, 它不是View的子類, 而是個ViewParent. ViewRootImpl是連接Window和DecorView的紐帶, View的焦點, 按鍵, 布局, 渲染等流程都是從ViewRoot中開始的.

View的焦點

基本流程如下

這裡寫圖片描述

View(包括ViewGroup)獲取焦點都通過如下三個方法:
View.java -> requestFocus()

這裡寫圖片描述vcewwb249tfu1tW74da00NC1vbXayP249re9t6guPGJyIC8+DQrX7rrztcRyZXF1ZXN0Rm9jdXNOb1NlYXJjaM/IxdC2z8rHt/G/ydLUu/HIob25teMsIMi7uvO9+Mjrz8LD5rXEwfezzDo8YnIgLz4NClZpZXcuamF2YSAtJmd0OyByZXF1ZXN0Rm9jdXNOb1NlYXJjaCgpPC9wPg0KPHA+PGltZyBhbHQ9"這裡寫圖片描述" src="/uploadfile/Collfiles/20160822/201608220944101336.png" title="\" />

View.java -> handleFocusGainInternal()

這裡寫圖片描述

上面的流程比較簡單: 如果當前沒有焦點, 先置焦點標志, 再通知parent, 然後刷新圖片.

主要的流程在mParent的requestChildFocus裡面, 後面會分析. 那裡會逐層向上修改焦點View並清除原來有焦點的View的焦點
onFocusChange會觸發invalidate刷新, 然後調用onFocusChangeListener. 默認情況每個View只能設置一個onFocusChangeListener, 而開發中經常遇到需要設置多個Listener的情況, 我們就可以重寫onFocusChange方法, 實現回調多個onFocusChangeListener的需求.

ViewGroup的焦點

ViewGroup獲取焦點是在View獲取焦點流程中多了內部焦點處理
ViewGroup.java -> requestFocus()

這裡寫圖片描述

上面代碼中descendantFocusability決定了是先按View焦點流程處理(自己處理焦點)還是先把給子View處理

FOCUS_BLOCK_DESCENDANTS 不允許子View獲取焦點,那麼按照View的流程進行 FOCUS_BEFORE_DESCENDANTS 先按照View的流程處理, 如果自己不能獲取焦點則給孩子處理 FOCUS_AFTER_DESCENDANTS 先嘗試給子view焦點, 如果沒有可獲取焦點再按照View流程自己獲取焦點 FOCUS_BEFORE_DESCENDANTS,默認值 我們可以通過setDescendantFocusability(int d)設置

onRequestFocusInDescendants方法是給子類重寫使用, 可以控制子View處理焦點. 默認按照子View順序處理, direction向下或向右則從第一個開始, 向上或向左則從最後一個開始, 直到某個子View獲取焦點

(注意此方法只在此ViewGroup及其上層View上調用requestFocus時會執行到)

父容器焦點的處理

在View獲取焦點流程中會調用mParent.requestChildFocus, 維護View樹上焦點唯一, 在各層ViewGroup中保存有焦點的子View

ViewGroup.java -> requestChildFocus()

這裡寫圖片描述

先清除自己的焦點, 如果原來內部有焦點, 先清除其焦點, 保存獲取焦點的孩子, 然後調用上一層的requestChildFocus. 最後的調用可知, 這個方法會一直調用到View的樹的root節點.

在當前ViewGroup內部, 任何一個孩子取得焦點都會執行到這個方法, 因此此方法也是ViewGroup得知孩子焦點變化的方法之一.(可惜不能得知孩子失去焦點)

失去焦點或清除焦點

獲取焦點可以是主動的, 但失去焦點一般都是被動的(見上面的代碼), 因此邏輯相對簡單, 只要清除焦點狀態即可.

ViewGroup.java -> unFocus()

這裡寫圖片描述

View.java -> unFocus() -> clearFocusInternal()

這裡寫圖片描述

注意上面的方法是默認package訪問級別的, 我們無法重寫也不能調用

也可以主動清除焦點, 與獲取焦點流程相似
View.java -> clearFocus()

這裡寫圖片描述

ViewGroup.java -> clearFocus()

這裡寫圖片描述

ViewGroup.java -> clearChildFocus()

這裡寫圖片描述

以上是安卓View系統焦點處理的全部流程和涉及到的方法, ViewRootImpl的requestChildFocus和clearChildFocus實現我們不需要關注

另外還有以下一些輔助方法

boolean isFocusable() View是否可以獲取焦點 boolean isFocused() View是否獲取焦點 boolean hasFocus() View/ViewGroup內部是否有焦點 View findFocus() 取到View/ViewGroup內部的焦點View View getFocusedChild() 取到ViewGroup內部有焦點的子View View getRootView() 取到根節點View(一般是DecorView或頂層ViewGroup)

焦點移動

除了在代碼裡面控制焦點, 系統對沒有處理的方向鍵等一些按鍵自動按照焦點移動來處理, 見下面代碼

ViewRootImpl.java -> processKeyEvent()

這裡寫圖片描述
這裡寫圖片描述
這裡寫圖片描述

代碼比較上, 但是主要做了三個步驟

如果View沒有處理按鍵, 把上下左右tab等按鍵轉換成對應方向 在當前焦點View上通過focusSearch方法查找對應方向的下一個View 查找到的View調用requestFocus,因此主要的流程在focusSearch中

View.java -> focusSearch()

這裡寫圖片描述

普通View查找什麼都沒做, 交給parent來完成.

ViewGroup.java -> focusSearch()

這裡寫圖片描述

ViewRootImpl.java -> focusSearch()

這裡寫圖片描述

我們可以重寫focusSearch控制焦點移動順序, 而默認的焦點移動順序由FocusFinder決定

FocusFinder查找焦點

FocusFinder為public的工具類, 主要就兩個方法, 可以在給定的View內在指定方向查找指定View或坐標的下一個焦點
如下:
FocusFinder.java -> findNextFocus()

這裡寫圖片描述

核心邏輯就兩步, 先查找setNextFocusXXId設置的View, 如果沒有按照就近算法查找.
具體算法不再分析, SDK裡面有源碼.

總結

綜合上面的流程分析, 我們在實現自定義View時, 對焦點的特殊需求有如下思路

requestFocus和clearFocus直接對View清除或轉移焦點 除了onFocusChangeListener,還可以在onFocusChange方法中實現一些View失去/獲得焦點時通知 對ViewGroup, 如果只需要在子View獲取焦點時得到通知, 有requestChildFocus方法. 重寫onRequestFocusInDescendants方法可以控制某些情景下ViewGroup焦點 控制焦點移動可以重寫focusSearch方法 查找焦點的過程,主要是從View的focusSearch(…)方法開始,從當前焦點開始逐層往外,最終在最外層布局執行FocusFinder中的核心方法來獲得下個焦點所在的視圖view. 另外還有FocusFinder工具和上面的輔助方法. 在view獲得焦點之前,必須先判斷該view是否具有獲得焦點的權限,可通過isFocusable和isFocusableInTouchMode來判斷;//這個前文代碼中有 同時可以通過setFocusable和setFocusableInTouchMode來設置指定view具有獲取焦點的權限. 在XML阻止子view獲得焦點的屬性是:android:descendantFocusability =”blocksDescendants”,但是如果當前activity經過了pause或者stop後再重新resume後該屬性會失效,這時可以在onresume裡面加上requestFocusFromTouch方法就能重新時屬性生效.
  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved