Android 8.0系统上使用WindowManager添加view的一个问题

6,617 阅读1分钟
原文链接: blog.csdn.net

这里写图片描述

如图,app有个功能,用户设置一个定时,到时后会弹窗提示。实现很简单,注册SYSTEM_ALERT_WINDOW权限,使用WindowManager对象调用addView(view, para)添加view即可。

WindowManager wm = (WindowManager) context.getApplicationContext().getSystemService(Context.WINDOW_SERVICE);
para.type = WindowManager.LayoutParams.TYPE_PHONE;
...
(6.0以上需要动态检查是否被授予了SYSTEM_ALERT_WINDOW权限)
...
wm.addView(view, para);

之前一直好使,最近把测试机升级到8.0,却发现直接崩溃了:

android.view.WindowManager$BadTokenException: Unable to add window -- token null is not valid; is your activity running?

但是这时候我的app确实是在前台运行的,经过一番搜索之后发现,在8.0上,对于TYPE_PHONE等几种种窗口类型,使用SYSTEM_ALERT_WINDOW权限不能在其他应用和系统窗口上方显示提醒窗口,而需要使用名为 TYPE_APPLICATION_OVERLAY 的新窗口类型。

官方文档具体如下:

使用 SYSTEM_ALERT_WINDOW 权限的应用无法再使用以下窗口类型来在其他应用和系统窗口上方显示提醒窗口:

TYPE_PHONE
TYPE_PRIORITY_PHONE
TYPE_SYSTEM_ALERT
TYPE_SYSTEM_OVERLAY
TYPE_SYSTEM_ERROR
相反,应用必须使用名为 TYPE_APPLICATION_OVERLAY 的新窗口类型。

使用 TYPE_APPLICATION_OVERLAY 窗口类型显示应用的提醒窗口时,请记住新窗口类型的以下特性:

应用的提醒窗口始终显示在状态栏和输入法等关键系统窗口的下面。
系统可以移动使用 TYPE_APPLICATION_OVERLAY 窗口类型的窗口或调整其大小,以改善屏幕显示效果。
通过打开通知栏,用户可以访问设置来阻止应用显示使用 TYPE_APPLICATION_OVERLAY 窗口类型显示的提醒窗口。

所以在设置type的时候增加判断即可:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            para.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
        } else {
            para.type = WindowManager.LayoutParams.TYPE_PHONE;
        }