4.1 初識(shí)ViewRoot和DocerView
在正式介紹View的三大流程之前寇僧,我們必須先介紹一些基本概念允华,這樣才能更好地理解View的measure压汪、layout和draw過(guò)程丽蝎,本節(jié)主要介紹ViewRoot和DecorView的概念飒房。
ViewRoot對(duì)應(yīng)于ViewRootImpl類呻拌,它是連接WindowManager和DecorView的紐帶葱轩,View的三大流程均是通過(guò)ViewRoot來(lái)完成的。在ActivityThread中,當(dāng)Activity對(duì)象被創(chuàng)建完畢后靴拱,會(huì)將DeecorView添加到Window中复亏,同時(shí)會(huì)創(chuàng)建ViewRootImpl對(duì)象,并將ViewRootImpl對(duì)象和DecorView建立關(guān)聯(lián)缭嫡,這個(gè)過(guò)程可參看如下源碼:
root = new ViewRootImpl(view.getContext(), display);
root.setView(view, wparams, panelParentView);
View的繪制流程是從ViewRoot的performTraversals方法開始的缔御,它經(jīng)過(guò)measure、layout和draw三個(gè)過(guò)程才能最終將一個(gè)View繪制出來(lái)妇蛀,其中measure用來(lái)測(cè)量View的寬和高耕突,layout用來(lái)確定View在父容器中的放置位置,而draw則負(fù)責(zé)將View繪制在屏幕上评架,針對(duì)performTraversals的大致流程眷茁,可以用流程圖4-1來(lái)表示。
圖4-1 performTraversals的工作流程圖
如圖4-1所示纵诞,performTraversals會(huì)依次調(diào)用performMeasure上祈、performLayout和performDraw三個(gè)方法,這三個(gè)方法分別完成頂級(jí)View的measure浙芙、layout和draw這三大流程登刺,其中在performMeasure中會(huì)調(diào)用measure方法,在measure方法中又會(huì)調(diào)用onMeasure方法嗡呼,在onMeasure方法中則會(huì)對(duì)所有的子元素進(jìn)行measure過(guò)程纸俭,這個(gè)時(shí)候measure流程就從父容器傳遞到子元素中了,這樣就完成了一次measure過(guò)程南窗。接著子元素會(huì)重復(fù)父容器的measure過(guò)程揍很,如此反復(fù)就完成了整個(gè)View樹的遍歷。同理万伤,performLayout和performDraw的傳遞流程和performMeasure是類似的窒悔,唯一不同的是,performDraw的傳遞過(guò)程是在draw方法中通過(guò)dispatchDraw來(lái)實(shí)現(xiàn)的敌买,不過(guò)這并沒有本質(zhì)區(qū)別简珠。
measure過(guò)程決定了View的寬/高,Measure完成以后放妈,可以通過(guò)getMeasuredWidth和getMeasuredHeight方法來(lái)獲取到View測(cè)量后的寬/高北救,在幾乎所有的情況下它都等同于View最終的寬/高荐操,但是特殊情況除外芜抒,這點(diǎn)在本章后面會(huì)進(jìn)行說(shuō)明。Layout過(guò)程決定了View的四個(gè)頂點(diǎn)的坐標(biāo)和實(shí)際View的寬/高托启,完成以后宅倒,可以通過(guò)getTop、getBottom屯耸、getLeft和getRight來(lái)拿到View的四個(gè)頂點(diǎn)的位置拐迁,并可以通過(guò)getWidth和getHeight方法來(lái)拿到View的最終寬/高蹭劈。Draw過(guò)程則決定了View的顯示,只有draw方法完成以后View的內(nèi)容才能呈現(xiàn)在屏幕上线召。
如圖4-2所示铺韧,DecorView作為頂級(jí)View,一般情況下它內(nèi)部會(huì)包含一個(gè)豎直方向的LinearLayot缓淹,在這個(gè)LinearLayout里面有上下兩個(gè)部分(具體情況和Android版本及主題有關(guān))哈打,上面是標(biāo)題欄,下面是內(nèi)容欄讯壶。在Android中我們通過(guò)setCOntentView所設(shè)置的布局文件其實(shí)就是被加到內(nèi)容欄之中的料仗,而內(nèi)容欄的id是content,因此可以理解為Activity指定布局的方法不叫setView而叫setContentView伏蚊,因?yàn)槲覀兊牟季值拇_加到了id為content的FrameLayout中立轧。如何得到content呢?可以這樣:ViewGroup content = findViewyId(R.android.id.content)躏吊。如何得到我們?cè)O(shè)置的View呢氛改?可以這樣:content.getChildAt(0)。同時(shí)比伏,通過(guò)源碼我們可以知道平窘,DecorView其實(shí)是一個(gè)FrameLayout,View層的事件都先經(jīng)過(guò)DecorView凳怨,然后才傳遞給我們的View瑰艘。
圖4-2 頂級(jí)View:DecorView的結(jié)構(gòu)