編輯:關於Android編程
前言:View框架寫到第六篇,發現前面第二篇竟然沒有,然後事情是在微信公眾號發了,忘記在博客上更新,所以關注微信公眾號的應該都看過了,趁今天有時間遂補上。(PS:本篇文章中源碼均是android 6.0,請知曉)
本來之前說view下篇是寫onMeasure,onLayou,onDraw相關的,筆者做盒子開發,遙控器按鍵,碰到的都是焦點控制相關。所以先把焦點放到了onMeasure,onLayou,onDraw之前。以前踩過不少坑,最近公司同事做分享,遂下決心總結下,Agenda如下:
ViewRoot View的焦點 ViewGroup的焦點 父容器焦點的處理 失去焦點或清除焦點 焦點移動 FocusFinder查找焦點 總結Android焦點相關邏輯大部分都在都在View, ViewGroup和FocusFinder三個類中.
View對象都有一個mParent變量(添加到ViewGroup後), 代指其父容器. 絕大部分View的mParent都是ViewGroup類型, 除了根節點. 一個Window中View根節點DecorView的mParent稱為ViewRoot, 在安卓4.0後ViewRoot對應ViewRootImpl, 它不是View的子類, 而是個ViewParent. ViewRootImpl是連接Window和DecorView的紐帶, View的焦點, 按鍵, 布局, 渲染等流程都是從ViewRoot中開始的.
基本流程如下
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獲取焦點是在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為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方法就能重新時屬性生效.前言前段時間,公司由個同事分享的時候,提到了MVP模式,自己之前也了解過,但是真正在自己的編碼過程中使用的非常少。最近在幫助一個朋友做畢業設計,心想這是一個很好的機會練習
先看下面圖片:這是我在做登錄頁面的時候,調用系統的ProgressDialog 進行等待,可是看起來很不協調,左邊的等待圖片過大,右邊文字過小,看起來老別扭,雖然功能上不
SQLITE 構化查詢語言 (Structured Query Language)是什麼SQLite是一個輕量型的數據庫。怎麼樣輕量級 : SQLite數據庫是一個輕量級
上篇分析了View的事件分發流程,留了一個問題:如果上面的EventButton繼承TextView的話,按下抬起,會有一個現象,我可以告訴大家現象:就是只有dispat