写这篇文章的原因是解决了一个APP闪退的问题,闪退的原因是插拔U盘时,注册的广播接收者接收到广播需要弹出一个Dialog询问是否需要打开U盘,这个Dialog设置的是系统级别悬浮窗,没有这个权限,报错导致闪退,下面从具体报错开始看。注:我的这个APP是文件管理器,通过Android.mk预置到系统中,为系统应用不可卸载。
一、分析报错
报错:
Unable to add window android.view.ViewRootImpl$W@5b87261 -- permission denied for window type 2003
涉及到的代码块:
if (null == usbDialog) {
usbDialog = new UsbDialog(context, R.style.Dialog);
usbDialog.setCancelable(true);
usbDialog.setCanceledOnTouchOutside(true);
usbDialog.getWindow().setType(
WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
usbDialog.setCloseCallBack(new UsbDialog.CloseCallBack() {
@Override
public void onDismiss() {
closeusbDialog();
}
});
usbDialog.show();
}
WindowManager.LayoutParams.TYPE_SYSTEM_ALERT 这里设置了系统级悬浮窗类型,即该悬浮窗是悬浮在所有应用之上的包括launcher,开发过悬浮窗的小伙伴应该会比较熟悉,悬浮分为APP内悬浮和系统级悬浮,感兴趣的同学可以看看我的另一个篇文章 悬浮球入门教学。
关于这个报错网上的大部分解决方式如下:
总结一下意思就是在用户使用的时候弹窗给用户授权,我的文件管理器不能出现弹窗授权这种情况,所以该怎么默认授予应用 android.permission.SYSTEM_ALERT_WINDOW 这个权限呢?APP层面的修改肯定不行了,需要修改framework的代码。
二、解决问题
需要修改的代码如下:
代码路径 frameworks\base\services\core\java\com\android\server\pm\permission\DefaultPermissionGrantPolicy.java
新增下面的代码: 假设你的应用包名为 com.rpg.rebuild
private static final Set<String> WINDOW_PERMISSIONS = new ArraySet<>();
static {
WINDOW_PERMISSIONS.add(Manifest.permission.SYSTEM_ALERT_WINDOW);
}
. . . . . .
. . . . . .
. . . . . .
grantSystemFixedPermissionsToSystemPackage(pm,
getDefaultProviderAuthorityPackage("com.rpg.rebuild", userId),
userId,WINDOW_PERMISSIONS);
注:需要预制成系统应用,上述代码才生效。
写在最后:如果你是一个三方APP开发者,你的APP是上架应用商店的,目前是没有办法实现不弹窗给用户授予,就获得系统级悬浮窗口权限的,除非就是让手机厂商把你的三方APP预制到系统中像上面一样授予默认权限。