Unable to add window android.view.ViewRootImpl -- permission denied for this window type

今天在写一个悬浮框的案例时,偶然发现了一个6.0存在的一个问题。 先上代码:

private void addFlowLayoutOnWindow() {
        //获取WindowManager对象
        windowManager = ((WindowManager) getSystemService(WINDOW_SERVICE));
        //创建WindowManager的布局参数对象
        wmParams = new WindowManager.LayoutParams();
        //设置添加View的类型
        wmParams.type = WindowManager.LayoutParams.TYPE_PHONE;
        //设置添加View的标识
        wmParams.flags = WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS |
                WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
        wmParams.gravity = Gravity.LEFT | Gravity.TOP;
        //设置添加View的默认坐标值
        wmParams.x = 50;
        wmParams.y = 50;
        //设置添加View的宽高
        wmParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
        wmParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
        wmParams.format = 1;


        //最终调用WindowManger.addView方法,将View添加到窗口上
        windowManager.addView(flowLayout, wmParams);
    }

以上代码就是调用系统的WindowManager添加了一个TYPE_PHONE的View,并显示到屏幕上,但是这段代码如果想正常运行需要在AndroidManifest.xml中申请以下权限

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

添加权限申请之后,代码可以正常运行。

但是一旦以上代码运行到6.0上之后的版本,就会报一下错误

Unable to add window android.view.ViewRootImpl -- permission denied for this window type_第1张图片

提示我没有权限, WTF!!!
后来在官方文档上终于找到了答案:
https://developer.android.com/reference/android/provider/Settings.html#canDrawOverlays(android.content.Context)

Unable to add window android.view.ViewRootImpl -- permission denied for this window type_第2张图片

上图中提到的canDrawOverlays方法就是用来判断一个App是否能够在其它App之上显示视图。正常来讲一个App是不能在其它App之上显示视图的,除非在AndroidManifest.xml文件中声明了SYSTEM_ALERT_WINDOW权限,6.0之后添加了动态权限申请机制,因此如果是在6.0之后的版本上还需要用户主动的允许权限申请。而为了让系统弹出提示用户允许权限申请操作的界面,需要程序员在代码中发送ACTION_MANAGE_OVERLAY_PERMISSION。具体代码可以参照如下代码:

if (Build.VERSION.SDK_INT >= 23) {
            if (!Settings.canDrawOverlays(this)) {
                Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION);
                intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                startActivityForResult(intent, 1);
            } else {
                //TODO do something you need
            }
        }

你可能感兴趣的:(互联网开发经验)