即使在设置中启用权限后,Settings.canDrawOverlays仍返回false

时间:2016-10-04 12:38:25

标签: android overlay system-alert-window

我正在尝试从this回答浪潮代码来检查是否启用了权限。但即使从设置启用了权限,它也会返回false。

public static boolean canDrawOverlayViews(Context con){
    if(Build.VERSION.SDK_INT< Build.VERSION_CODES.LOLLIPOP){return true;}
    try {
        return Settings.canDrawOverlays(con);
    }
    catch(NoSuchMethodError e){
        return canDrawOverlaysUsingReflection(con);
    }
}


public static boolean canDrawOverlaysUsingReflection(Context context) {

    try {

        AppOpsManager manager = (AppOpsManager) context.getSystemService(Context.APP_OPS_SERVICE);
        Class clazz = AppOpsManager.class;
        Method dispatchMethod = clazz.getMethod("checkOp", new Class[] { int.class, int.class, String.class });
        //AppOpsManager.OP_SYSTEM_ALERT_WINDOW = 24
        int mode = (Integer) dispatchMethod.invoke(manager, new Object[] { 24, Binder.getCallingUid(), context.getApplicationContext().getPackageName() });

        return AppOpsManager.MODE_ALLOWED == mode;

    } catch (Exception e) {  return false;  }

}

5 个答案:

答案 0 :(得分:5)

最近我也遇到了同样的问题并得到了以下解决方法。 参考 https://code.google.com/p/android/issues/detail?id=198671#c7

public boolean getWindoOverLayAddedOrNot2() {

   String sClassName = "android.provider.Settings";
   try {
       Class classToInvestigate = Class.forName(sClassName);
       if (context == null)
           context = activity;
       Method method = classToInvestigate.getDeclaredMethod("isCallingPackageAllowedToDrawOverlays", Context.class, int.class, String.class, boolean.class);
       Object value = method.invoke(null, context, Process.myUid(), context.getPackageName(), false);

       Log.i("Tag", value.toString());
       // Dynamically do stuff with this class
       // List constructors, fields, methods, etc.

   } catch (ClassNotFoundException e) {
       // Class not found!
   } catch (Exception e) {
       // Unknown exception
       e.printStackTrace();
   }

   return false;
}

答案 1 :(得分:1)

检查是否涉及设备管理员? 我在禁用设备管理员时遇到此问题,我在DeviceAdminReceiver-&gt; onDisabled()和某些设备上检查了此权限,并且canDrawOverlays返回false,尽管我已获得许可。

上述答案有时帮助但不是所有时间。在检查之前做的工作是Thread.sleep。

try {
    Thread.sleep(20);
} catch (InterruptedException e) {
    // some exception here
}

对我有用的最短时间是20毫升。比canDrawOverlays返回true

注意:这不是一个好习惯,但这是唯一对我有用的东西

答案 2 :(得分:0)

根据BennyP的回答,我使Runnable在500毫秒后运行所需的代码,并且效果很好。反馈有些延迟,但是用户甚至不会注意到延迟。

这是我添加到onResume()

中的代码
Handler handler = new Handler();
    handler.postDelayed(new Runnable() {
        @Override
        public void run() {
            if(!Settings.canDrawOverlays(ControllerActivity.this)){
                //Handle overlay permission not given here
            }
            else{
                //Handle overlay permission given here
            }
        }
    }, 500);

希望有帮助!

答案 3 :(得分:0)

在用户访问设置后,我尝试重新启动活动。这是代码:

public static void restartActivity(Activity act){

Intent intent = getIntent();

finish();

startActivity(intent);

}

答案 4 :(得分:-1)

首先,我对

这种奇怪的行为感到非常惊讶
    Settings.canDrawOverlays(this);        

我的使用也面临同样的问题,即使已经分配了权限,它也会返回false。 我注意到了,我在onStart()方法中使用了这个检查,它正在创建这种有线行为。为了解决这个问题,我在互联网上搜索,没有任何结果可以满足我和我可以使用的那个。

解决方案

     if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {

        Log.e("Overlay Permission", "" + Settings.canDrawOverlays(this));
        if (!Settings.canDrawOverlays(this)) {
            MyPreferences.saveBoolean(HomeScreen.this, "showOverlayPermissionDialog", true);
        } else {
            MyPreferences.saveBoolean(HomeScreen.this, "showOverlayPermissionDialog", false);
        }
    }

我在onCreate()中做了一些湖。在这里,我在SharedPreferences中相应地保存了值,并根据这些共享首选项值,我创建了一个检查,用于在我的onStart()中显示叠加对话框。这很好用!

您可以尝试此解决方案,并在问题解决后将此答案标记为有用。

由于