Activity -> PhoneWindow-> DecorView-> TitleView(ActionBarContainer->ActionBar)、ContentView(FrameLayout、RelativeLayout)
PhoneWindow這個類是Framework為我們提供的Android窗口概念的具體實現(xiàn)。調用setContentView()方法設置Activity的用戶界面時,實際上就完成了對所關聯(lián)的PhoneWindow的ViewTree的設置。
每個應用程序窗口的decorView都有一個與之關聯(lián)的ViewRoot對象,這種關聯(lián)關系是由WindowManager來維護的。decorView與ViewRoot的關聯(lián)關系是Activity啟動時,在ActivityThread.handleResumeActivity()方法中建立的。
繪制起點:requestLayout
繪制流程的三個階段:
測量 measure: 判斷是否需要重新計算View的大小,需要的話則計算
布局 layout: 判斷是否需要重新計算View的位置,需要的話則計算
繪制 draw: 判斷是否需要重新繪制View,需要的話則重繪制。
一、測量 measure
1.MeasureSpec 的理解
MeasureSpec中的值是一個整型(32位),其中高兩位是mode,后面30位存的是size。
一個MeasureSpec封裝了從父容器傳遞給子容器的布局要求,是由父View的MeasureSpec和子View的LayoutParams通過簡單的計算得出一個針對子View的測量要求。
MeasureSpec一共有三種模式:
UPSPECIFIED : 對于子容器沒有任何限制,通常是不確定的值,需要對子容器進一步測量才能確定。
EXACTLY: 為View和子容器設置了精確尺寸。
AT_MOST:View和子容器可以是該限定大小內的任意值。
2.測量流程的理解
如果我們在xml 的layout_width或者layout_height 把值都寫死,那么測量完全就不需要了。
之所以要測量,是因為 match_parent 就是充滿父容器,wrap_content 就是自己多大就多大。
這個測量過程,就是計算出來的父View的MeasureSpec不斷往子View傳遞,結合子View的LayoutParams 一起再算出子View的MeasureSpec,子View有了MeasureSpec才能測量自己和自己的子View。
2.1 父View的MeasureSpec 是EXACTLY時
說明父View的大小是確切的,那么它的size 是多大,最后展示到屏幕就是多大。
1、如果子View的大小是MATCH_PARENT,那么子View的大小肯定也是確切的,子View的size=父View的size,mode=EXACTLY。
2、如果子View 的layout_xxxx是WRAP_CONTENT,大小是根據(jù)自己的content 來決定的,但是子View畢竟不能超過父View的大小。因此子View MeasureSpec mode的應該是AT_MOST,而size 暫定父View的 size。
3、如果子View 的layout_xxxx是確定的值例如200dp,那么控件最后展示就是200dp,不管父View有多大,也不管自己的content 有多大。MeasureSpec 的mode = EXACTLY 大小size=layout_xxxx 設置的值。
2.2 父View的MeasureSpec 是AT_MOST時
父View的大小是不確定,最大的值是MeasureSpec 的size。
1、如果子View 的layout_xxxx是MATCH_PARENT,由于父View的大小不確定(只限制了最大值),那么子View即使充滿父容器,大小肯定也不能確定的。所以子View的mode=AT_MOST,size=父View的size。
子View最終大小由父View測量結果決定。
2、如果子View 的layout_xxxx是WRAP_CONTENT,父View的大小是不確定(只知道最大限定值),那么在子View的Content沒算出大小之前,子View MeasureSpec mode的就是AT_MOST,而size 暫定父View的 size。
子View最終大小由計算出的content決定。
2.3 父View的MeasureSpec 是UNSPECIFIED(未指定)時
表示沒有任何束縛和約束,不像AT_MOST表示最大只能多大,不也像EXACTLY表示父View確定的大小,子View可以得到任意想要的大小,不受約束。
對于子View來說無論是WRAP_CONTENT還是MATCH_PARENT,子View也是沒有任何束縛的,size的值沒有任何意義了,所以一般都直接設置成0。
測量工作都是在measure方法中調用onMeasure()做的,measure方法是final的所以這個方法也不可重寫,如果想自定義View的測量,應該重寫onMeasure()方法。