Android View 焦點尋址的來龍去脈 !!!
一、前言
Android 系統的智能電視,最近兩年基本上越來越火了,而在現在手機 App 開發中,都是使用觸摸的模式進行操作,但是在開發智能電視 App 的時候,View 的焦點尋址,就是我們繞不過去的坎。
那么,本文就 View 如何控制 Focus,來分析它背后的實現邏輯。
二、View 的焦點控制
在 Android 中,是有一套自己的焦點查找的算法,簡單來說,就近原則,就是按方向就近查找下一個符合條件的 View。
如果我們有對一個 View 焦點控制的需求,需要對這個 View 強制指定它上下左右之后的下一個獲取焦點的 View。可以通過 View 的屬性來控制,只需要在對應方向上設定我們需要焦點轉移的下一個 View 的 ID 即可。
- android:nextFocusDown:按下的時候,焦點尋址的 ViewId。
- android:nextFocusUp:按上的時候,焦點尋址的 ViewId。
- android:nextFocusLeft:按左的時候,焦點尋址的 ViewId。
- android:nextFocusRight:按右的時候,焦點尋址的 ViewId。
- android:nextFocusForward:向前的時候,焦點尋址的 ViewId。
這些都是最基本的,不是本文的主題。那么如果沒有設定這些屬性,而 Android 對 View 的 就近原則 的焦點查找算法,到底是如何實現的呢?
三、View 的焦點尋址
Android 中,是如何做到對 View 焦點的控制呢?
我們就先從 View 的源碼看起,看看它是如何找到“下一個”位置的 View的,在 View 中,查找下一個應該獲取焦點的 View ,使用的方法是 focusSearch()。
它需要傳遞一個 direction 參數,這個 direction 就是指定獲取什么方向上的下一個位置的焦點。
而 View 并不會處理焦點尋址的具體邏輯,而是將焦點的查找委托給 mParent 來實現,mParent 是一個接口,它的實現類是 ViewRootImpl。
ViewRootImlp.focusSearch() 最終又將焦點尋址的任務,交托給 FocusFinder 來處理。
findNextFocus()中可以看到,findNextUserSpecifiedFocus() 方法正是用于查找我們對 View 設定不同方向的下一個焦點的 ViewId ,它的優先級是***的,如果沒有找到,才會進行 findNextFocus() 通過算法來查找對應的 View。
為了證實這個說法,我們先看看 findNextUserSpecifiedFocus()的源碼。
最終,又調回到 View.findUserSetNextFocus() 方法去尋找。
到這里也證實了我們的猜測,確實是通過 direction 來進行 View 的焦點尋址。
再回過頭來看看 findNextFocus() 方法,如果通過 findNextUserSpecifiedFocus() 方法沒有找到我們指定的 View,就會繼續向下執行。這里聲明的一個局部變量 focusables 就是用于存放符合算法的所有 View 。
接下來再看看如何向 focusables 這個 List 中,添加符合尋址要求的 View。在 findNextFocus() 中可以看到,它最終會調用 findNextFocusInAbsoluteDirection() 方法。
從 findNextFocusInAbsoluteDirection() 方法可以看出,就近原則就是在這里實現的,通過 View 的坐標點,計算出***要求的 View ,最終將找到的 View 返回過去。
到這里,基本就追蹤到 View 對焦點尋址的完整邏輯。***補一個方法調用的流程圖。
四、總結
如果有對 View 焦點的控制,可以考慮通過設置 View 的屬性,還可以通過重寫 View.focusSearch() 方法,來定制 View 焦點的尋址規則。不過一般而言,不推薦重寫 focusSearch() 方法,只使用屬性控制也能滿足我們的需求。
【本文為51CTO專欄作者“張旸”的原創稿件,轉載請通過微信公眾號聯系作者獲取授權】