5

我正在尝试使用答案中的代码来检查权限是否已启用。但即使从设置中启用了权限,它也会返回 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;  }

}
4

5 回答 5

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;
}
于 2016-11-25T15:05:49.587 回答
0

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

公共静态无效restartActivity(活动行为){

意图意图 = getIntent();

结束();

开始活动(意图);

}

于 2019-04-24T07:12:55.943 回答
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);

希望能帮助到你!

于 2018-11-29T13:28:22.950 回答
0

检查是否涉及设备管理员?我在禁用设备管理员时遇到了这个问题,我在 DeviceAdminReceiver->onDisabled() 和某些设备上检查了这个权限,尽管我有权限,但 canDrawOverlays 返回 false。

上述答案有时会有所帮助,但并非总是如此。起作用的是检查前的 Thread.sleep 。

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

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

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

于 2017-02-09T11:01:09.863 回答
-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 中相应地保存了这些值,并根据这些 Shared Preference 值,我创建了一个检查以在我的 onStart() 中显示一个覆盖对话框。这很好用!

您可以尝试此解决方案,如果您的问题得到解决,请将此答案标记为有用。

谢谢

于 2017-03-27T10:50:23.583 回答