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
從上面可以看到前兩個最終會執行到第三個方法.
最後的requestfocusnosearch先判斷是否可以擷取焦點, 然後進入下面的最後流程:
上面的流程比較簡單: 如果目前沒有焦點, 先置焦點标志, 再通知parent, 然後重新整理圖檔.
主要的流程在mparent的requestchildfocus裡面, 後面會分析. 那裡會逐層向上修改焦點view并清除原來有焦點的view的焦點
onfocuschange會觸發invalidate重新整理, 然後調用onfocuschangelistener. 預設情況每個view隻能設定一個onfocuschangelistener, 而開發中經常遇到需要設定多個listener的情況, 我們就可以重寫onfocuschange方法, 實作回調多個onfocuschangelistener的需求.
viewgroup的焦點
viewgroup擷取焦點是在view擷取焦點流程中多了内部焦點處理
viewgroup.java
上面代碼中descendantfocusability決定了是先按view焦點流程處理(自己處理焦點)還是先把給子view處理
focus_block_descendants 不允許子view擷取焦點, 那麼按照view的流程進行
focus_before_descendants 先按照view的流程處理, 如果自己不能擷取焦點則給孩子處理
focus_after_descendants 先嘗試給孩子焦點, 如果沒有可擷取焦點再按照view流程自己擷取焦點
預設值focus_before_descendants, 我們可以通過setdescendantfocusability(int d)
設定
onrequestfocusindescendants方法是給子類重寫使用, 可以控制子view處理焦點. 預設按照子view順序處理, direction向下或向右則從第一個開始, 向上或向左則從最後一個開始, 直到某個子view擷取焦點
注意此方法隻在此viewgroup及其上層view上調用requestfocus時會執行到
父容器焦點的處理
在view擷取焦點流程中會調用mparent.requestchildfocus, 維護view樹上焦點唯一, 在各層viewgroup中儲存有焦點的子view
先清除自己的焦點, 如果原來内部有焦點, 先清除其焦點, 儲存擷取焦點的孩子, 然後調用上一層的requestchildfocus. 最後的調用可知, 這個方法會一直調用到view的樹的root節點.
在目前viewgroup内部, 任何一個孩子取得焦點都會執行到這個方法, 是以此方法也是viewgroup得知孩子焦點變化的方法之一.(可惜不能得知孩子失去焦點)
失去焦點或清除焦點
擷取焦點可以是主動的, 但失去焦點一般都是被動的(見上面的代碼), 是以邏輯相對簡單, 隻要清除焦點狀态即可.
注意上面的方法是預設package通路級别的, 我們無法重寫也不能調用
也可以主動清除焦點, 與擷取焦點流程相似
以上是安卓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
代碼比較上, 但是主要做了三個步驟
如果view沒有處理按鍵, 把上下左右tab等按鍵轉換成對應方向
在目前焦點view上通過focussearch方法查找對應方向的下一個view
查找到的view調用requestfocus是以主要的流程在focussearch中
普通view查找什麼都沒做, 交給parent來完成.
viewrootimpl
我們可以重寫focussearch控制焦點移動順序, 而預設的焦點移動順序由focusfinder決定
focusfinder查找焦點
focusfinder為public的工具類, 主要就兩個方法, 可以在給定的view内在指定方向查找指定view或坐标的下一個焦點如下:
核心邏輯就兩步, 先查找setnextfocusxxid設定的view, 如果沒有按照就近算法查找.具體算法不再分析, sdk裡面有源碼.
總結
綜合上面的流程分析, 我們在實作自定義view時, 對焦點的特殊需求有如下思路
requestfocus和clearfocus直接對view清除或轉移焦點
除了onfocuschangelistener, 還可以在onfocuschange方法中實作一些view失去/獲得焦點時通知
對viewgroup, 如果隻需要在子view擷取焦點時得到通知, 有requestchildfocus方法.
重寫onrequestfocusindescendants方法可以控制某些情景下viewgroup焦點
控制焦點移動可以重寫focussearch方法
另外還有focusfinder工具和上面的輔助方法.
本文作者:佚名
來源:51cto