Android 窗口管理
在整個控件樹的最頂端,是一個邏輯的樹頂,ViewParent,在源碼中的實現(xiàn)是ViewRoot(ViewRoot extends Handler implements ViewParent)。它是整個控件樹和WindowManager之間的事件信息的翻譯者。WindowManager是Android中一個重要的服務(wù)。它將用戶的操作,翻譯成為指令,發(fā)送給呈現(xiàn)在界面上的各個Window。Activity會將頂級的控件注冊到WindowManager中,當(dāng)用戶真是觸碰屏幕或鍵盤的時候,WindowManager就會通知到,而當(dāng)控件有一些請求產(chǎn)生,也會經(jīng)由ViewParent送回到WindowManager中。從而完成整個通信流程。
事件分發(fā)流程
用戶在觸發(fā)一個時間后產(chǎn)生一個消息,消息先被window manager采集到,然后分發(fā)給client。
window manager通過IWindow分發(fā)給client,我們知道ViewRoot使用IWindowSession與server通訊,使用IWindow接收消息,所以第一步消息到了ViewRoot這里來了
也就是window manager->IWindow->ViewRoot
到了ViewRoot后它把消息轉(zhuǎn)發(fā)給Looper(樓主可以看ViewRoot.java看它是怎么獲取當(dāng)前線程的Looper的)
Looper->ViewRoot$RootHandler().dispatch(),有轉(zhuǎn)發(fā)會給ViewRoot,然后就通過decor view形成的樹狀結(jié)構(gòu)依次分發(fā)下去。
window manager proxy中維護了view, ViewRoot, layout param三元組。
每次調(diào)用window manager proxy的addView都會新增一個三元組。一般程序中都是調(diào)用addView(decor,...),也就是只對decor view調(diào)用addView
*****************\
在Activity在performLaunchActivity時,會使用Activity.attach()建立一個PhoneWindow主窗口。這個主窗口的建立并不是一個重點。handleResumeActivity真正要啟動一個Activity時候,將主窗口加入到WindowManager,當(dāng)然并不是將主窗口本身,而是將主窗口的DecorView加入到WindowManager中。
• performLaunchActivity@ActivityThread.java
attach()@Activity.javaSession.java
addWindow()@WindowManager
mWindow = PolicyManager.makeNewWindow(this); --實例化一個activity或者dialog或者widget的地方才會make new window
• handleResumeActivity()@ActivityThread.java
addView()@WindowManagerImpl.java --添加DecorView到WindowManager中
setView()@ViewRoot.java
add()@IwindowSession.java
| --這里通過AIDL調(diào)用
add()@WindowManagerService$Service.java
關(guān)于PhoneWindow:
而PhoneWindow只是做了一個具體跟手機功能相關(guān)的公用事件的處理,所以在Android中PhoneWindow并不是一個抽象的純正概念,而是一個跟手機系統(tǒng)相關(guān)的一個特別窗口概念,例如按鍵的默認(rèn)動作處理,按鍵音的發(fā)出等等。
一個View對應(yīng)一個ViewRoot
問題:WindowMangerService如果通過AIDL將事件派發(fā)到客戶端的?