8.3.2 Dialog的Window创建过程
Dialog的Window的创建过程和Activity类似。
1. 创建Window
Dialog中Window的创建同样是通过PolicyManager的makeNewWindow方法来完成的,创建后的对象实际上就是PhoneWindow,这个过程和Activity的Window的创建过程是一致的。
Dialog(Context context, int theme, boolean createContextThemeWrapper) {
if (createContextThemeWrapper) {
if (theme == 0) {
TypedValue outValue = new TypedValue();
context.getTheme().resolveAttribute(com.android.internal.R.attr.dialogTheme,
outValue, true);
theme = outValue.resourceId;
}
mContext = new ContextThemeWrapper(context, theme);
} else {
mContext = context;
}
mWindowManager = (WindowManager)context.getSystemService(Context.WINDOW_SERVICE);
Window w = PolicyManager.makeNewWindow(mContext);
mWindow = w;
w.setCallback(this);
w.setOnWindowDismissedCallback(this);
w.setWindowManager(mWindowManager, null, null);
w.setGravity(Gravity.CENTER);
mListenersHandler = new ListenersHandler(this);
}
2. 初始化DecorView并将Dialog的视图添加到DecorView中
这个过程也和Activity的类似,都是通过Window去添加指定的布局文件。
public void setContentView(int layoutResID) {
mWindow.setContentView(layoutResID);
}
3. 将DecorView添加到Window中并显示
在Dialog的show方法中,会通过WindowManager将DecorView添加到Window中
mWindowManager.addView(mDecor, l);
mShowing = true;
从上面三个步骤可以发现,Dialog的Window创建和Activity的Window创建过程很类似,两者几乎没有什么区别。当Dialog被关闭时,它会通过WindowManager来移除DecorView:mWindowManager.removeViewImmediate(mDecor)。
普通的Dialog有一个特殊之处,那就是必须采用Activity的Context,如果采用Application的Context,那么就会报错。
Dialog dialog = new Dialog(this.getApplicationContext());
TextView textView = new TextView(this);
textView.setText("this is toast");
dialog.setContentView(textView);
dialog.show();
上述代码会报错,错误信息如下:
Caused by: android.view.WindowManager$BadTokenException: Unable to add window -- token null is not valid; is your activity running?
at android.view.ViewRootImpl.setView(ViewRootImpl.java:765)
at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:356)
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:93)
at android.app.Dialog.show(Dialog.java:330)
at com.study.wumeng.practice.MainActivity.onCreate(MainActivity.java:20)
at android.app.Activity.performCreate(Activity.java:7009)
at android.app.Activity.performCreate(Activity.java:7000)
是没有应用token所导致的,而应用token一般只有Activity拥有,所以这里只需要用Activity作为Context来显示对话框即可。另外,系统Window比较特殊,它可以不需要token,因此在上面的例子中,只需要指定对话框的Window为系统类型就可以正常弹出对话框。在本章一开始讲到,WindowManager.LayoutParams中的type表示Window的类型,而系统Window的层级范围是2000~2999,这些层级范围就对应着type参数。系统WIndow的层级有很多值,对于本例来说,可以选择用TYPE_SYSTEM_OVERLAY来指定对话框的Window类型为系统Window。
dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY);
然后别忘了在AndroidManifest文件中声明权限从而可以使用系统Window。
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>