Android 6.0 悬浮窗默认关闭解决方案

来源:互联网 发布:见一页而知岁月将暮 编辑:程序博客网 时间:2024/04/30 03:09

在谷歌往Android中加入悬浮窗口功能时就表示希望开发者只用其来做用户通知,修改的悬浮窗功能潜在一定的安全隐患,不过手机厂商可不这么认为,于是本来被用于通知的悬浮窗被改成了其他的功能。我们都知道Android 6.0中,系统新增应用授权机制,还默认禁用了“浮动窗口”权限,所以悬浮窗功能只能当作通知使用。 


直接上错误异常信息

android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$W@60838a -- permission denied for this window type        at android.view.ViewRootImpl.setView(ViewRootImpl.java:591)        at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:310)        at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:85)        at android.app.Dialog.show(Dialog.java:319)        ....

一、遇到问题

  1. Android 6.0 使用悬浮窗崩溃问题 (permission denied for this window type)
  2. 当你使用targetSdkVersion=22 ,而且添加了 
    <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />权限,正常可以使用悬浮窗,然后修改targetSdkVersion=23,重新编译后依然能使用,也就是说只要一次授权了 就不在检测悬浮窗权限了。这是遇到的坑需要注意,卸载重装就好了

二、解决方案

  1. 使用TYPE_TOAST 
    getWindow().setType(WindowManager.LayoutParams.TYPE_TOAST); 
    优点: 不需要权限,都能显示 
    缺点: API level<19 的机器(MIUI除外),要有 android.permission.SYSTEM_ALERT_WINDOW权限并且将 type 设置为 WindowManager.LayoutParams.TYPE_PHONE 或者WindowManager.LayoutParams.TYPE_SYSTEM_ALERT 可以显示但没有交互

为什么TYPE_TOAST就不要权限呢?查看Android源码

public int checkAddPermission(WindowManager.LayoutParams attrs) {    int type = attrs.type;    if (type < WindowManager.LayoutParams.FIRST_SYSTEM_WINDOW            || type > WindowManager.LayoutParams.LAST_SYSTEM_WINDOW) {        return WindowManagerImpl.ADD_OKAY;    }    String permission = null;    switch (type) {        case TYPE_TOAST:            // XXX right now the app process has complete control over            // this...  should introduce a token to let the system            // monitor/control what they are doing.            break;        case TYPE_INPUT_METHOD:        case TYPE_WALLPAPER:            // The window manager will check these.            break;        case TYPE_PHONE:        case TYPE_PRIORITY_PHONE:        case TYPE_SYSTEM_ALERT:        case TYPE_SYSTEM_ERROR:        case TYPE_SYSTEM_OVERLAY:            permission = android.Manifest.permission.SYSTEM_ALERT_WINDOW;            break;        default:            permission = android.Manifest.permission.INTERNAL_SYSTEM_WINDOW;    }    if (permission != null) {        if (mContext.checkCallingOrSelfPermission(permission)                != PackageManager.PERMISSION_GRANTED) {            return WindowManagerImpl.ADD_PERMISSION_DENIED;        }    }    return WindowManagerImpl.ADD_OKAY;}
这个方法是往系统的WindowManager里addView的时候做权限检查用的,除了TYPE_TOAST外,其他的都需要添加权限 。 

2. 先进行权限检测,引导用户开启

private static final int REQUEST_CODE = 1;  private  void requestAlertWindowPermission() {      //new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,Uri.parse("package:" + getPackageName()));    Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION);      intent.setData(Uri.parse("package:" + getPackageName()));      startActivityForResult(intent, REQUEST_CODE);  }    @Override  protected void onActivityResult(int requestCode, int resultCode, Intent data) {      super.onActivityResult(requestCode, resultCode, data);      if (requestCode == REQUEST_CODE) {          if (Settings.canDrawOverlays(this)) {              Log.i(LOGTAG, "onActivityResult success");          }      }  }          


手动设置方式 
(Android 6.0+:设置——应用——右上角齿轮——「在其他应用的上层显示」) 
3. 更改targetSDKVersion=22后,将被继续使用旧有规则,用户在安装的时候不得不接受所有权限,安装后app就有了那些权限,在Manifest里添加<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />权限 也能正常使用了。

如果想了解更详细可以参考 
* http://www.liaohuqiu.net/cn/posts/android-windows-manager/

转自:http://blog.csdn.net/vfush/article/details/51203092