屏幕覆盖检测到块 Android 权限

Screen overlay detected blocks Android permissions

我注意到我的 Android 应用程序在我的新 phone 上有一个奇怪的问题。 SDK 23 权限弹出窗口(如外部存储)被下面附加的警告阻止。我最初认为这与我的 phone 有关,但它似乎没有影响我安装的任何其他应用程序。

这个问题可能与安装调试版本有关,还是我的权限处理有问题?我认为它可能与我正在使用的其中一个广告平台有关,但我尝试禁用它们但它仍然出现

我在下面粘贴了生成此权限请求的图像保存功能。我正在使用 Dexter 来节省编写一大堆丑陋的样板代码

public static void saveToExternalStorageIfAllowed(final Context context, final Bitmap bitmapImage, final String title) {
    final Tracker t = ((LoLHistory) context.getApplicationContext()).getTracker(LoLHistory.TrackerName.APP_TRACKER);

    // saving to publicly visible/accessible folder. Requires write permission
    int permissionCheck = ContextCompat.checkSelfPermission(context, Manifest.permission.WRITE_EXTERNAL_STORAGE);
    if (permissionCheck != PackageManager.PERMISSION_GRANTED) {
        // do not have permissions to write, request
        t.send(new HitBuilders.EventBuilder()
                .setCategory("FILE")
                .setAction("PermissionMissing")
                .setLabel("WRITE_EXTERNAL")
                .build());
        Dexter.checkPermission(new PermissionListener() {
            @Override
            public void onPermissionGranted(PermissionGrantedResponse response) {
                t.send(new HitBuilders.EventBuilder()
                        .setCategory("FILE")
                        .setAction("PermissionGranted")
                        .setLabel("WRITE_EXTERNAL")
                        .build());

                saveToExternalStorage(context, bitmapImage, title);
            }

            @Override
            public void onPermissionDenied(PermissionDeniedResponse response) {
                t.send(new HitBuilders.EventBuilder()
                        .setCategory("FILE")
                        .setAction("PermissionDenied")
                        .setLabel("WRITE_EXTERNAL")
                        .build());
            }

            @Override
            public void onPermissionRationaleShouldBeShown(PermissionRequest permission, PermissionToken token) {/* ... */}
        }, Manifest.permission.WRITE_EXTERNAL_STORAGE);
    } else {
        saveToExternalStorage(context, bitmapImage, title);
    }
}

private static void saveToExternalStorage(Context context, Bitmap bitmapImage, String title) {
    Tracker t = ((LoLHistory) context.getApplicationContext()).getTracker(LoLHistory.TrackerName.APP_TRACKER);

    // create image folder if does not exist
    File imagesFolder = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES), context.getString(R.string.app_name));
    if (!imagesFolder.mkdirs() && !imagesFolder.isDirectory()) {
        String state = Environment.getExternalStorageState();
        if (Environment.MEDIA_MOUNTED.equals(state)) {
            // failed to create and is not a directory. Something went wrong...
            t.send(new HitBuilders.EventBuilder()
                    .setCategory("FILE")
                    .setAction("CreateDirFailed")
                    .setLabel(imagesFolder.getPath())
                    .build());
        } else {
            t.send(new HitBuilders.EventBuilder()
                    .setCategory("FILE")
                    .setAction("CreateDirFailedMediaNotMounted")
                    .setLabel(imagesFolder.getPath())
                    .build());
        }
    }

    // delete image if already exists so FOS can create a new one
    File image = new File(imagesFolder, title + ".jpg");
    if (image.exists()) {
        // image already exists, deleting to start from clean state
        if (!image.delete()) {
            // failed to delete
            t.send(new HitBuilders.EventBuilder()
                    .setCategory("FILE")
                    .setAction("DeleteFailed")
                    .setLabel(image.getPath())
                    .build());
        }
    }

    // compress bitmap and write to file stream. FOS creates file if does not exist
    FileOutputStream out = null;
    try {
        out = new FileOutputStream(image);
        bitmapImage.compress(Bitmap.CompressFormat.JPEG, 50, out);
        out.flush();
    } catch (Exception e) {
        e.printStackTrace();
        t.send(new HitBuilders.ExceptionBuilder()
                .setDescription(e.getLocalizedMessage())
                .setFatal(true)
                .build());
    } finally {
        try {
            if (out != null) {
                out.close();
            }
        } catch (IOException e) {
            e.printStackTrace();
            t.send(new HitBuilders.ExceptionBuilder()
                    .setDescription(e.getLocalizedMessage())
                    .setFatal(true)
                    .build());
        }
    }

    // get Uri from saved image
    Uri uriSavedImage = Uri.fromFile(image);

    // media scan the new file so it shows up in the gallery
    Intent mediaScanIntent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
    mediaScanIntent.setData(uriSavedImage);
    context.sendBroadcast(mediaScanIntent);
}

更新:因为很多人都在提到它,所以如前所述,这个问题不是因为安装了覆盖应用程序。在 Draw over other apps 菜单下,我有以下应用程序:Google Play Music、Google Play services、Photos、TalkBack、Twitch、Twitter。 所有这些都设置为否

此外,我已经测试了其他应用程序,如 Google Hangouts 和 Twitter,它们也有需要危险权限的操作,我能够提供这些权限而不会出现此问题。


解决方案: 我已将 R. Zagorski 的答案标记为解决方案,因为它包含很多一般情况。对我来说,实际上是 Toast 打破了我的权限流程。这个弹出窗口让我走上了完全错误的道路,浪费了很多时间...

这是 Toast 在权限弹出窗口出现后的前几秒我看到的:

此弹出窗口成为 Android Marshmallow 中包含权限管理器的问题。如果您 phone 上安装了任何有权在屏幕上覆盖的应用程序,则有许多权限设置,包括文件访问权限,您必须先禁用这些屏幕覆盖权限才能更改这些设置。对我来说,我的文本应用程序和 Facebook Messenger 应用程序是罪魁祸首。每次我想为任何其他应用程序提供请求的权限时,我必须单击该弹出窗口中的“打开设置”选项并撤销上述两个应用程序的屏幕覆盖访问权限,然后重新打开有问题的应用程序以再次获得权限提示。然后,如果我想要消息弹出窗口或聊天头像,我必须再次重新启用覆盖权限。这真的很烦人。我认为您的应用程序很好,android 的权限管理只是一团糟。

当检测到屏幕覆盖时,系统将不允许您继续为应用授予权限。因为屏幕覆盖可以接收用户的输入。 因此,在授予您的应用程序权限之前,您必须禁用所有 "screen overlay" 项。 这不是您的应用程序的问题。所有具有目标 SDK 23 的应用程序都会遇到此问题。

此弹出窗口是由清单声明的 manifest.PERMISSION.SYSTEM_ALERT_WINDOW 权限引起的。 开发人员必须了解 3 类权限。:

  1. 普通权限 - 什么都不做,只需在清单中声明
  2. Vulnerable permissions - 在Manifest中声明并第一时间请求许可。它们可以通过系统设置进行更改
  3. 以上危险权限:SYSTEM_ALERT_WINDOW and WRITE_SETTINGS属于此类。它们必须被授予,但在系统设置中不可见。要请求它,您不使用标准方式 (int checkSelfPermission (String permission)),但您必须适当地检查 Settings.canDrawOverlays()Settings.System.canWrite()

如果您没有SYSTEM_ALERT_WINDOW权限:

  1. 检查与权限弹出窗口交互时是否可见 Toast 虽然检测到覆盖弹出窗口没有提及,但 Toast也算作叠加
  2. 检查您的任何依赖项是否需要它

如果您不确定自己是否在使用此权限,可以执行几个测试用例:

  1. 自行申请权限:

    public class MainActivity extends AppCompatActivity {
    
        public final static int REQUEST_CODE = 10101;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            if (checkDrawOverlayPermission()) {
                startService(new Intent(this, PowerButtonService.class));
            }
        }
    
        public boolean checkDrawOverlayPermission() {
            if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
                return true;
            }
            if (!Settings.canDrawOverlays(this)) {
                Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
                    Uri.parse("package:" + getPackageName()));
                startActivityForResult(intent, REQUEST_CODE);
                return false;
            } else {
                return true;
            }
        }
    
        @Override
        @TargetApi(Build.VERSION_CODES.M)
        protected void onActivityResult(int requestCode, int resultCode, Intent data) {
            if (requestCode == REQUEST_CODE) {
                if (Settings.canDrawOverlays(this)) {
                    startService(new Intent(this, PowerButtonService.class));
                }
            }
        }
    }
    
  2. 检查 this or this 是否不是你的情况

  3. 查看注意,当通过Play商店安装应用程序时,此权限是自动授予的(我没有检查过,因此无法确认)

权限模型可以理解,只是有时候需要多挖掘。

这不是问题,您只需安装一个覆盖应用程序,最常见的是

  • Facebook 即时通
  • 启用弹出窗口的 Whatsapp
  • Seebye 聊天头
  • Viber
  • 任何在屏幕上显示菜单或其他内容的应用都是叠加应用。

关闭它,然后尝试。

通常,这是由于使用了浮动应用程序造成的。

三星 Android Phone 用户:

Open the Settings
Then Applications > Application manager
Press on More > Apps that can appear on top

如果您发现有气泡的应用程序,请显示屏幕叠加层。如果某个应用调整了 phone 的亮度,请同时禁用这些应用上的屏幕覆盖选项。关键是,你必须找出罪魁祸首应用程序并禁用屏幕覆盖。

来源:Fix Screen Overlay Detected Error On Android

我在谷歌上搜索了很多解决方案,但没有任何效果。然后我尝试了设置中的所有选项:)

最后我可以给你明确的方法来禁用覆盖 -

转到设置 > 应用程序 > 默认应用程序 > 设备帮助 > "Turn of Read Text on screen" 选项

我已经通过安装 Alert Window Checker: https://play.google.com/store/apps/details?id=jp.sfapps.alertwindowchecker 解决了这个问题。它指示您当前哪些应用程序正在使用屏幕覆盖功能。

就我而言,我必须在应用程序 ES FileExplorer 中禁用滑动检测器才能解决此问题

重新启动 phone 对我有用。