第八章(4)---Dialog的Window创建过程

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"/>
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 夜已当空,星光不遐,当我抬头望着深邃的星空的时候,就好比是一个冰激凌,被太阳的火焰骤然包裹,在窒息中融化,最终升华...
    玻璃上的生理盐水阅读 298评论 0 1
  • 这是我第一次接触简书这个软件,还是从一本励志书籍中听到的一个词,然后就下载来看看,不过现在已经是夜晚11点多了,已...
    石大雪花阅读 233评论 0 0