一、Window、WindowManager、WindowManagerService的關(guān)系
Window是一個抽象類槐雾,具體的實現(xiàn)類是PhoneWindow對View進(jìn)行管理。WindowManager是一個接口類幅狮,繼承自ViewManager募强,從名稱上就知道是用來管理window的,它的實現(xiàn)類是WindowManagerImpl彪笼。如果我們想要對 window (View)
Window 包含了 View 井對 View 進(jìn)行管理泵肄, Window 用虛線來表示是因為 Window是個抽象概念捆交,用來描述一個窗口,并不是真實存在的腐巢, Window 的實體其實也是 View品追,WindowManager 用來管理 Window ,而WindowManager 所提供的功能最終會由 WMS 進(jìn)行處理冯丙。
二肉瓦、WindowManager 的關(guān)聯(lián)類
Window Manager 個接口類,繼承自接口 ViewManager, View Manager 中定義了個方怯胃惜,分別用來添加泞莉、更新和刪除 View ,如下所示:
public interface ViewManager
{
public void addView(View view, ViewGroup.LayoutParams params);
public void updateViewLayout(View view, ViewGroup.LayoutParams params);
public void removeView(View view);
}
Window Manager 也繼承了這些方法船殉,而這些方怯傳人的參數(shù)都是 View 類型鲫趁,說明Window 是以 View 的形式存在的。 WindowManager 在繼承 ViewManager 的同時利虫,又加入很多功能 挨厚,包括 Window 類型和層級相關(guān)的常量、內(nèi)部類以及一些方泣糠惫,其中有兩個方法是根據(jù) Window 的特性加入的疫剃,如下所示:<br />
<br />
public Display getDefaultDsplay ();
public void removeViewimmedate(View view) ;
getDefaultDisplay 方怯能夠得知這個 WindowManager 例將 Window 添加到哪個屏上了,換句話說硼讽,就是得到 WindowManager 所管理的屏幕 Display removeViewImmediate方法則規(guī)定在這個方法返回前要立即執(zhí)行 View.onDetachedFromWindow()慌申,來完成傳入的View 相關(guān)的銷毀工作。
Window類是一個抽象類理郑,具體的實現(xiàn)類是phonewindow蹄溉,PhoneWindow是在Activity創(chuàng)建時調(diào)用attach方法時創(chuàng)建的,代碼如下:
final void attach(Context context, ActivityThread aThread,
Instrumentation instr, IBinder token, int ident,
Application application, Intent intent, ActivityInfo info,
CharSequence title, Activity parent, String id,
NonConfigurationInstances lastNonConfigurationInstances,
Configuration config, String referrer, IVoiceInteractor voiceInteractor,
Window window, ActivityConfigCallback activityConfigCallback) {
attachBaseContext(context);
mFragments.attachHost(null /*parent*/);
mWindow = new PhoneWindow(this, window, activityConfigCallback);
mWindow.setWindowControllerCallback(this);
mWindow.setCallback(this);
mWindow.setOnWindowDismissedCallback(this);
mWindow.getLayoutInflater().setPrivateFactory(this);
if (info.softInputMode != WindowManager.LayoutParams.SOFT_INPUT_STATE_UNSPECIFIED) {
mWindow.setSoftInputMode(info.softInputMode);
}
if (info.uiOptions != 0) {
mWindow.setUiOptions(info.uiOptions);
}
mUiThread = Thread.currentThread();
.........
mWindow.setWindowManager(
(WindowManager)context.getSystemService(Context.WINDOW_SERVICE),
mToken, mComponent.flattenToString(),
(info.flags & ActivityInfo.FLAG_HARDWARE_ACCELERATED) != 0);
if (mParent != null) {
mWindow.setContainer(mParent.getWindow());
}
mWindowManager = mWindow.getWindowManager();
mCurrentConfig = config;
mWindow.setColorMode(info.colorMode);
setAutofillCompatibilityEnabled(application.isAutofillCompatibilityEnabled());
enableAutofillCompatibilityIfNeeded();
}
在第十行代碼mWindow = new PhoneWindow(this, window, activityConfigCallback);處創(chuàng)建了PhoneWindow您炉,然后在23行代碼調(diào)用了PhoneWindow的setWindowManager方法柒爵,代碼如下:
public void setWindowManager(WindowManager wm, IBinder appToken, String appName,
boolean hardwareAccelerated) {
mAppToken = appToken;
mAppName = appName;
mHardwareAccelerated = hardwareAccelerated
|| SystemProperties.getBoolean(PROPERTY_HARDWARE_UI, false);
if (wm == null) {
wm = (WindowManager)mContext.getSystemService(Context.WINDOW_SERVICE);
}
mWindowManager = ((WindowManagerImpl)wm).createLocalWindowManager(this);
}
wm為null時,通過getSystemService貨物到WindowManager赚爵,看第十行代碼里面有個強(qiáng)制轉(zhuǎn)換類型為WindowManagerImpl棉胀,我們來看下createLocalWindowManager里面做了什么:
public WindowManagerImpl createLocalWindowManager(Window parentWindow) {
return new WindowManagerImpl(mContext, parentWindow);
}
創(chuàng)建了一個WindowManagerImpl的實例,所以我們可以知道(WindowManager)mContext.getSystemService(Context.WINDOW_SERVICE)對應(yīng)的就是一個WindowManagerImpl冀膝。將window對象作為參數(shù)傳進(jìn)了唁奢,這樣WindowManagerImpl就持有了window的引用,可以對window進(jìn)行操作窝剖,比如添加view麻掸,在WindowManagerImpl的addView方法里面我們看一下:
@Override
public void addView(@NonNull View view, @NonNull ViewGroup.LayoutParams params) {
applyDefaultToken(params);
mGlobal.addView(view, params, mContext.getDisplay(), mParentWindow);
}
處調(diào)用 WindowManagerGlobal的 ddView 方法,其中最后一個參數(shù)mParentWindow就是上面提到的Window 赐纱,可以看 WindowManagerlmpl 雖然是WindowManager 的實現(xiàn)類但是沒有實現(xiàn)什么功能脊奋,而是將功委托給WindowManagerGlobal 熬北,這里用到的是橋接模式。關(guān)于WindowManagerGlobal的addView
方法會在 后面中進(jìn)行介紹 诚隙。我 來查看在 indowManagerImpl 中是如何定義WindowManagerGlobal 的讶隐,如下所示:
public final class WindowManagerImpl implements WindowManager {
//單例模式
private final WindowManagerGlobal mGlobal = WindowManagerGlobal.getInstance();
private final Context mContext;
private final Window mParentWindow;
private IBinder mDefaultToken;
public WindowManagerImpl(Context context) {
this(context, null);
}
private WindowManagerImpl(Context context, Window parentWindow) {
mContext = context;
mParentWindow = parentWindow;
}
public WindowManagerImpl createLocalWindowManager(Window parentWindow) {
return new WindowManagerImpl(mContext, parentWindow);
}
public WindowManagerImpl createPresentationWindowManager(Context displayContext) {
return new WindowManagerImpl(displayContext, mParentWindow);
}
}
通過上面的源碼分析,我們客知道WindowManager的關(guān)聯(lián)類如下圖所示:<br />PhoneWindow 繼承自 Window, Window 通過 setWindow Manager
方住與 WindowManager 發(fā)生關(guān)聯(lián)久又。WindowManager 繼承自接口 ViewManager,Window ManagerImp 是WindowMar er 接口的 實現(xiàn)類巫延,但是具體 的功能都會委托給WindowManagerG!obal 來實現(xiàn)