MeasureSpec 是一個 32 位的 int 類型听盖,并且取了最前面的兩位代表 Mode胀溺,后 30 位代表大小 Size裂七。
3種模式
EXACTLY(精確模式,在這種模式下仓坞,尺寸的值是多少背零,那么這個組件的長或?qū)捑褪嵌嗌伲瑢?yīng) MATCH_PARENT 和確定的值)
AT_MOST(WRAP_CONTENT无埃,最大就是父控件的大嗅闫俊)
UNSPECIFIED(當前組件,可以隨便用空間嫉称,不受限制侦镇,系統(tǒng)用,具體就體現(xiàn)在 NestedScrollView 和 ScrollView 中)
我們開發(fā)中织阅,常用AT_MOST壳繁,EXACTLY
我們可能會給定默認大小,或者根據(jù)內(nèi)容變化荔棉,經(jīng)常都是固定大小
根視圖
ViewRoot - performTraversals(執(zhí)行遍歷)
ViewGroup 還需要負責通知自己的子 View 進行繪制操作
private void performTraversals() {
...
int childWidthMeasureSpec = getRootMeasureSpec(mWidth, lp.width);
int childHeightMeasureSpec = getRootMeasureSpec(mHeight, lp.height);
...
//執(zhí)行測量流程
performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
...
//執(zhí)行布局流程
performLayout(lp, desiredWindowWidth, desiredWindowHeight);
...
//執(zhí)行繪制流程
performDraw();
}
onMeasure
每個 View 都有自己的大小闹炉,所以基本自定義 View 的時候都需要重寫 onMeasure() 這個方法,以定制化我們的 View 的寬高
如果不重寫這個方法润樱,我們通常會出現(xiàn) wrap_content 和 match_parent 是一樣的顯示效果
View 默認是會使用 getDefaultSize() 方法進行設(shè)置寬高的渣触,在 AT_MOST和 EXACTLY 兩種情況下都會直接使用測量規(guī)格里面的尺寸
在 ViewGroup 中,并沒有去重寫 View 的 onMeasure() 方法壹若,而這都需要它的子類根據(jù)自己的邏輯去實現(xiàn)
ViewGroup 提供了一個 measureChildren() 方法來依次遍歷每個子 View 對其進行測量嗅钻。
在經(jīng)過 onMeasure() 操作后皂冰,getMeasureWidth() 和 getMeasureHeight() 方法就可以拿到正確的返回值了
由于 View 的 measure 過程和 Activity 的生命周期方法不是同步執(zhí)行的,如果 View 還沒有測量完畢养篓,那么獲得的寬/高就是 0灼擂。所以在 onCreate()、onStart()觉至、onResume() 中均無法正確得到某個 View 的寬高信息剔应。可以通過在 onWindowFocusChanged() 判斷獲取到焦點后進行獲取语御,或者使用 view.post()方式峻贮。
onLayout
onLayout() 方法主要作用是確定子 View 的顯示位置,由于 View 已經(jīng)是最小的層級应闯,所以我們在自定義 View 的時候通常不需要管這個方法纤控,而在自定義 ViewGroup 的時候就不得不注意這個方法
通過確認left,top碉纺,right船万,bottom四個值,確定了view的大小
獲取view的大小骨田,可以在onSizeChange之后
draw
drawBackground()
onDraw()
dispatchDraw()
onDrawScrollbars()
通常我的內(nèi)容就是在draw中耿导,代碼編寫也是主要在這里