今天在写一个悬浮框的案例时,偶然发现了一个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上之后的版本,就会报一下错误
提示我没有权限, WTF!!!
后来在官方文档上终于找到了答案:
https://developer.android.com/reference/android/provider/Settings.html#canDrawOverlays(android.content.Context)
上图中提到的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
}
}