Android View焦點總結(jié)

本來之前說view下篇是寫onMeasure,onLayou,onDraw相關(guān)的,筆者做盒子開發(fā)粟按,遙控器按鍵诬滩,碰到的都是焦點控制相關(guān)霹粥。所以先把焦點放到了onMeasure,onLayou,onDraw之前。

    • ViewRoot

    • View的焦點

    • ViewGroup的焦點

    • 父容器焦點的處理

    • 失去焦點或清除焦點

    • 焦點移動

    • FocusFinder查找焦點

    • 總結(jié)

Android View焦點

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

ViewRoot

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

View的焦點

基本流程如下


View(包括ViewGroup)獲取焦點都通過如下三個方法

View.java


從上面可以看到前兩個最終會執(zhí)行到第三個方法.

最后的requestFocusNoSearch先判斷是否可以獲取焦點, 然后進(jìn)入下面的最后流程:

View.java


上面的流程比較簡單: 如果當(dāng)前沒有焦點, 先置焦點標(biāo)志, 再通知parent, 然后刷新圖片.

  1. 主要的流程在mParent的requestChildFocus里面, 后面會分析. 那里會逐層向上修改焦點View并清除原來有焦點的View的焦點

  2. onFocusChange會觸發(fā)invalidate刷新, 然后調(diào)用onFocusChangeListener. 默認(rèn)情況每個View只能設(shè)置一個onFocusChangeListener, 而開發(fā)中經(jīng)常遇到需要設(shè)置多個Listener的情況, 我們就可以重寫onFocusChange方法, 實現(xiàn)回調(diào)多個onFocusChangeListener的需求.

ViewGroup的焦點

ViewGroup獲取焦點是在View獲取焦點流程中多了內(nèi)部焦點處理

ViewGroup.java


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

  • FOCUS_BLOCK_DESCENDANTS 不允許子View獲取焦點, 那么按照View的流程進(jìn)行

  • FOCUS_BEFORE_DESCENDANTS 先按照View的流程處理, 如果自己不能獲取焦點則給孩子處理

  • FOCUS_AFTER_DESCENDANTS 先嘗試給孩子焦點, 如果沒有可獲取焦點再按照View流程自己獲取焦點

默認(rèn)值FOCUS_BEFORE_DESCENDANTS, 我們可以通過setDescendantFocusability(int d)設(shè)置

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

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

父容器焦點的處理

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

ViewGroup.java


先清除自己的焦點, 如果原來內(nèi)部有焦點, 先清除其焦點, 保存獲取焦點的孩子, 然后調(diào)用上一層的requestChildFocus. 最后的調(diào)用可知, 這個方法會一直調(diào)用到View的樹的root節(jié)點.

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

失去焦點或清除焦點

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

ViewGroup.java


View.java


注意上面的方法是默認(rèn)package訪問級別的, 我們無法重寫也不能調(diào)用

也可以主動清除焦點, 與獲取焦點流程相似

ViewGroup.java


View.java


ViewGroup.java


以上是安卓View系統(tǒng)焦點處理的全部流程和涉及到的方法, ViewRootImpl的requestChildFocus和clearChildFocus實現(xiàn)我們不需要關(guān)注

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

  • boolean isFocusable() View是否可以獲取焦點

  • boolean isFocused() View是否獲取焦點

  • boolean hasFocus() View/ViewGroup內(nèi)部是否有焦點

  • View findFocus() 取到View/ViewGroup內(nèi)部的焦點View

  • View getFocusedChild() 取到ViewGroup內(nèi)部有焦點的子View

  • View getRootView() 取到根節(jié)點View(一般是DecorView或頂層ViewGroup)

焦點移動

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

ViewRootImpl.java



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

  1. 如果View沒有處理按鍵, 把上下左右tab等按鍵轉(zhuǎn)換成對應(yīng)方向

  2. 在當(dāng)前焦點View上通過focusSearch方法查找對應(yīng)方向的下一個View

  3. 查找到的View調(diào)用requestFocus
    因此主要的流程在focusSearch中

View.java


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

ViewGroup.java


ViewRootImpl


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

FocusFinder查找焦點

FocusFinder為public的工具類, 主要就兩個方法, 可以在給定的View內(nèi)在指定方向查找指定View或坐標(biāo)的下一個焦點
如下:


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

總結(jié)

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

  1. requestFocus和clearFocus直接對View清除或轉(zhuǎn)移焦點

  2. 除了onFocusChangeListener, 還可以在onFocusChange方法中實現(xiàn)一些View失去/獲得焦點時通知

  3. 對ViewGroup, 如果只需要在子View獲取焦點時得到通知, 有requestChildFocus方法.

  4. 重寫onRequestFocusInDescendants方法可以控制某些情景下ViewGroup焦點

  5. 控制焦點移動可以重寫focusSearch方法

  6. 另外還有FocusFinder工具和上面的輔助方法.


最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末疼鸟,一起剝皮案震驚了整個濱河市后控,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌空镜,老刑警劉巖浩淘,帶你破解...
    沈念sama閱讀 211,265評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異吴攒,居然都是意外死亡张抄,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,078評論 2 385
  • 文/潘曉璐 我一進(jìn)店門洼怔,熙熙樓的掌柜王于貴愁眉苦臉地迎上來署惯,“玉大人,你說我怎么就攤上這事镣隶〖辏” “怎么了?”我有些...
    開封第一講書人閱讀 156,852評論 0 347
  • 文/不壞的土叔 我叫張陵安岂,是天一觀的道長轻猖。 經(jīng)常有香客問我,道長域那,這世上最難降的妖魔是什么咙边? 我笑而不...
    開封第一講書人閱讀 56,408評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮琉雳,結(jié)果婚禮上样眠,老公的妹妹穿的比我還像新娘友瘤。我一直安慰自己翠肘,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,445評論 5 384
  • 文/花漫 我一把揭開白布辫秧。 她就那樣靜靜地躺著束倍,像睡著了一般。 火紅的嫁衣襯著肌膚如雪盟戏。 梳的紋絲不亂的頭發(fā)上绪妹,一...
    開封第一講書人閱讀 49,772評論 1 290
  • 那天,我揣著相機(jī)與錄音柿究,去河邊找鬼邮旷。 笑死,一個胖子當(dāng)著我的面吹牛蝇摸,可吹牛的內(nèi)容都是我干的婶肩。 我是一名探鬼主播办陷,決...
    沈念sama閱讀 38,921評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼律歼!你這毒婦竟也來了民镜?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,688評論 0 266
  • 序言:老撾萬榮一對情侶失蹤险毁,失蹤者是張志新(化名)和其女友劉穎制圈,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體畔况,經(jīng)...
    沈念sama閱讀 44,130評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡鲸鹦,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,467評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了问窃。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片亥鬓。...
    茶點故事閱讀 38,617評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖域庇,靈堂內(nèi)的尸體忽然破棺而出嵌戈,到底是詐尸還是另有隱情,我是刑警寧澤听皿,帶...
    沈念sama閱讀 34,276評論 4 329
  • 正文 年R本政府宣布熟呛,位于F島的核電站,受9級特大地震影響尉姨,放射性物質(zhì)發(fā)生泄漏庵朝。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,882評論 3 312
  • 文/蒙蒙 一又厉、第九天 我趴在偏房一處隱蔽的房頂上張望九府。 院中可真熱鬧,春花似錦覆致、人聲如沸侄旬。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,740評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽儡羔。三九已至,卻和暖如春璧诵,著一層夾襖步出監(jiān)牢的瞬間汰蜘,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,967評論 1 265
  • 我被黑心中介騙來泰國打工之宿, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留族操,地道東北人。 一個月前我還...
    沈念sama閱讀 46,315評論 2 360
  • 正文 我出身青樓比被,卻偏偏與公主長得像色难,于是被迫代替她去往敵國和親炕婶。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,486評論 2 348

推薦閱讀更多精彩內(nèi)容