View的繪制流程

Activity -> PhoneWindow-> DecorView-> TitleView(ActionBarContainer->ActionBar)、ContentView(FrameLayout、RelativeLayout)

PhoneWindow這個(gè)類是Framework為我們提供的Android窗口概念的具體實(shí)現(xiàn)。調(diào)用setContentView()方法設(shè)置Activity的用戶界面時(shí),實(shí)際上就完成了對(duì)所關(guān)聯(lián)的PhoneWindow的ViewTree的設(shè)置。

每個(gè)應(yīng)用程序窗口的decorView都有一個(gè)與之關(guān)聯(lián)的ViewRoot對(duì)象,這種關(guān)聯(lián)關(guān)系是由WindowManager來維護(hù)的。decorView與ViewRoot的關(guān)聯(lián)關(guān)系是Activity啟動(dòng)時(shí),在ActivityThread.handleResumeActivity()方法中建立的。

繪制起點(diǎn):requestLayout

繪制流程的三個(gè)階段:

測(cè)量 measure: 判斷是否需要重新計(jì)算View的大小,需要的話則計(jì)算

布局 layout: 判斷是否需要重新計(jì)算View的位置,需要的話則計(jì)算

繪制 draw: 判斷是否需要重新繪制View,需要的話則重繪制。

一、測(cè)量 measure

1.MeasureSpec 的理解

MeasureSpec中的值是一個(gè)整型(32位),其中高兩位是mode,后面30位存的是size。

一個(gè)MeasureSpec封裝了從父容器傳遞給子容器的布局要求,是由父View的MeasureSpec和子View的LayoutParams通過簡(jiǎn)單的計(jì)算得出一個(gè)針對(duì)子View的測(cè)量要求。

MeasureSpec一共有三種模式:

UPSPECIFIED : 對(duì)于子容器沒有任何限制,通常是不確定的值,需要對(duì)子容器進(jìn)一步測(cè)量才能確定。

EXACTLY: 為View和子容器設(shè)置了精確尺寸。

AT_MOST:View和子容器可以是該限定大小內(nèi)的任意值。

2.測(cè)量流程的理解

如果我們?cè)趚ml 的layout_width或者layout_height 把值都寫死,那么測(cè)量完全就不需要了。

之所以要測(cè)量,是因?yàn)?match_parent 就是充滿父容器,wrap_content 就是自己多大就多大。

這個(gè)測(cè)量過程,就是計(jì)算出來的父View的MeasureSpec不斷往子View傳遞,結(jié)合子View的LayoutParams 一起再算出子View的MeasureSpec,子View有了MeasureSpec才能測(cè)量自己和自己的子View。

2.1 父View的MeasureSpec 是EXACTLY時(shí)

說明父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的應(yīng)該是AT_MOST,而size 暫定父View的 size。

3、如果子View 的layout_xxxx是確定的值例如200dp,那么控件最后展示就是200dp,不管父View有多大,也不管自己的content 有多大。MeasureSpec 的mode = EXACTLY 大小size=layout_xxxx 設(shè)置的值。

2.2 父View的MeasureSpec 是AT_MOST時(shí)

父View的大小是不確定,最大的值是MeasureSpec 的size。

1、如果子View 的layout_xxxx是MATCH_PARENT,由于父View的大小不確定(只限制了最大值),那么子View即使充滿父容器,大小肯定也不能確定的。所以子View的mode=AT_MOST,size=父View的size。

子View最終大小由父View測(cè)量結(jié)果決定。

2、如果子View 的layout_xxxx是WRAP_CONTENT,父View的大小是不確定(只知道最大限定值),那么在子View的Content沒算出大小之前,子View MeasureSpec mode的就是AT_MOST,而size 暫定父View的 size。

子View最終大小由計(jì)算出的content決定。

2.3 父View的MeasureSpec 是UNSPECIFIED(未指定)時(shí)

表示沒有任何束縛和約束,不像AT_MOST表示最大只能多大,不也像EXACTLY表示父View確定的大小,子View可以得到任意想要的大小,不受約束。

對(duì)于子View來說無論是WRAP_CONTENT還是MATCH_PARENT,子View也是沒有任何束縛的,size的值沒有任何意義了,所以一般都直接設(shè)置成0。

測(cè)量工作都是在measure方法中調(diào)用onMeasure()做的,measure方法是final的所以這個(gè)方法也不可重寫,如果想自定義View的測(cè)量,應(yīng)該重寫onMeasure()方法。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容