美文网首页Android开发
如何通过getApplicationContext()返回的co

如何通过getApplicationContext()返回的co

作者: 夏广成 | 来源:发表于2017-06-14 09:18 被阅读1050次

android的window一共分为三个级别,包括应用级别的窗口,系统级别的窗口以及子窗口。其中dialog就属于应用级别的窗口。我们在使用dialog的时候,都知道dialog需要传入activity作为构建dialog的context。但是如果传入getApplicationContext,就会报错Unable to add window -- token null is not for an application。那么是不是就只能传入activity了呢?其实也不是。对话框抛出“无法添加窗口 – 令牌null不适用于应用程序”与 getApplication()作为上下文这篇文章一共列出了十种解决方案,我们这里主要就是讨论通过将应用级别的dialog提升为系统级别的window,来解决传入getApplicationContext()的问题。

一:通过window设置setType,提升window级别

系统window的级别从2000-2999不等

Dialog dialog=new Dialog(getApplicationContext());
dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
TextView textView=new TextView(this);
textView.setText("Hello");
dialog.setContentView(textView);
dialog.show();

之后我们还需要在AndroidMenifest.xml中添加一个权限,不然就会出现下面这样的异常。

Caused by: android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$W@d87e9e4 -- permission denied for this window type

需要添加的权限

 <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>

至此我们在其他手机上,就都可以通过传入getApplicationContext来解决弹出dialog的问题了,但是在小米手机上却总是弹不出来。

二:小米手机弹出自定义系统window。

在小米手机上,我们需要手动设置弹出系统window的权限,应用才可以弹出系统级别的window,具体设置方法是设置-->授权管理-->应用权限管理-->选择自己的app-->勾选显示悬浮窗

2017-06-15 09_53_43.gif

经过上面的操作,设置悬浮窗权限成功,就可以正确弹出系统级window。

三:通过运行时判断是否是小米系统解决miui无法弹出系统对话框问题

除了通过上述方法提示用户来设置系统级别window的权限问题,我们还可以通过代码判断是否是miui,这样我们就单独对miui做另外的处理。下面的方法,虽然没有设置悬浮窗权限,但是依然能够弹出系统级别的window。这是因为通过windownManager为dialog设置了setType。

 public void get_dialog(View view){
        Dialog dialog=new Dialog(getApplicationContext());
        TextView textView=new TextView(this);
        if(isMIUIRom()){
            dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_TOAST);
            textView.setText("Hello xiaomi");

        }else{
            dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
            textView.setText("Hello not xiaomi");

        }
        dialog.setContentView(textView);
        dialog.show();
    }

    public  boolean isMIUIRom(){
        String property =getSystemProperty("ro.miui.ui.version.name");
        return !TextUtils.isEmpty(property);
    }

    public  String getSystemProperty(String propName) {
        String line;
        BufferedReader input = null;
        try {
            Process p = Runtime.getRuntime().exec("getprop " + propName);
            input = new BufferedReader(new InputStreamReader(p.getInputStream()), 1024);
            line = input.readLine();
            input.close();
        } catch (IOException ex) {
            return null;
        } finally {
            if (input != null) {
                try {
                    input.close();
                } catch (IOException e) {
                }
            }
        }
        return line;
    }

四:应用级别的dialog为什么离开acitivty的token就活不下去了?

详情可以参考腾讯一位同学写的博客浅析Android的窗口
。文章分析的很详细,这里就不重复了。简要把这篇文章的大意记录一下。
前面我们提到了窗口分为三种:
(1):应用级别窗口,比如Activity。 Dialog 的窗口类型也是是 TYPE _ APPLICATION
(2):子窗口,比如PopupWindow。
(3):系统级别窗口,比如Toast。
其中应用类型窗口,子窗口,以及部分系统窗口,都是需要token的,而其他一些系统窗口则可以不需要token。这里需要token的系统窗口类型如TYPE _ INPUT _ METHOD,TYPE _ VOICE _ INTERACTION,TYPE _ WALLPAPER,TYPE _ DREAM,TYPE _ ACCESSIBILITY _ OVERLAY。

image.png

相关文章

网友评论

    本文标题:如何通过getApplicationContext()返回的co

    本文链接:https://www.haomeiwen.com/subject/uhhsqxtx.html