参考文章:
1、http://chen-wei.me/2016/11/10/android-permission-best-practice/
2、http://blog.csdn.net/lmj623565791/article/details/50709663
3、http://www.jianshu.com/p/bd8a114b4c39
前提介绍:悬浮窗权限Tips:
使用 type 值为 WindowManager.LayoutParams.TYPE_PHONE 或 WindowManager.LayoutParams.TYPE_SYSTEM_ALERT 需要申请 android.permission.SYSTEM_ALERT_WINDOW 权限。
具体分以下几类:
API >=23,需要在manifest中申请权限,并在每次需要用到权限的时候检查是否已有该权限,因为用户随时可以取消掉。具体可以看。
检测方法:调用系统Settings.canDrawOverlays
API >=18,只需要在manifest中申请权限。
对于一般手机,manifest中声明的权限在安装的时候会默认授权,并不需要处理。
对于某些rom,需要用AppOpsManager.checkOp方法检测。
API <18,默认有悬浮窗权限,不需要处理。
type 值为 WindowManager.LayoutParams.TYPE_TOAST 不需要权限。
API >25,TYPE_TOAST 已经被谷歌制裁了,会出现自动消失的情况,具体看 这里
API >= 19 ,可以接收触摸和按键事件。
API <19,无法接收无法接收触摸和按键事件,不需要权限和无法接受触摸事件的源码分析:Android悬浮窗TYPE_TOAST小结源码分析
之前有人做过绕过权限显示悬浮窗,就是用的TYPE_TOAST ,不推荐使用这种方法。
在代码中遇到一个报错:WindowManager$BadTokenException: Unable to add window -- window has already been added。
这个问题出现的原因是:
SDK版本低于7.1.1使用WindowManager.LayoutParams.TYPE_TOAST是不需要授权的,可以像平时用的toast一样展示与任何界面之上,而除了TYPE_TOAST之外都需要申请悬浮窗的权限。
所以7.1.1之后不要再使用TYPE_TOAST,解决方法wei改为使用TYPE_PHONE,并且需要在开启悬浮窗前必须主动申请一下权限:
if (! Settings.canDrawOverlays(MainActivity.this)) {
Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
Uri.parse("package:" + getPackageName()));
startActivityForResult(intent,10);
}
只要使用者允许后,授权dialog就可以在视窗上弹出
如果不同意的话做一些处理:
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == 10) {
if (!Settings.canDrawOverlays(this)) {
// SYSTEM_ALERT_WINDOW permission not granted...
Toast.makeText(MainActivity.this,"not granted",Toast.LENGTH_SHORT);
}
}
}