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-->勾选显示悬浮窗
经过上面的操作,设置悬浮窗权限成功,就可以正确弹出系统级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。