2

我正在修复一个不是我开发的 android 应用程序,几天前客户要求我将 targetSdkVersion 设置为 26(之前是 14),之后我注意到应用程序在某些情况下崩溃,例如崩溃时显示一个 ProgressDialog。这是存在此问题的代码部分:

mProgressDialog.setTitle(getString(R.string.downloading_data));
mProgressDialog.setMessage(mAlertMsg);
Drawable icon = getActivity().getDrawable(android.R.drawable.ic_dialog_info);
icon.setColorFilter(Color.parseColor("#00cbac"), PorterDuff.Mode.SRC_ATOP);
mProgressDialog.setIcon(icon);
mProgressDialog.setIndeterminate(true);
mProgressDialog.setCancelable(false);
mProgressDialog.setButton(getString(R.string.cancel), loadingButtonListener);
mProgressDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
mProgressDialog.show();

执行最后一行时,应用程序崩溃并在 Logcat 中出现此错误:

android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$W@b784d3a -- permission denied for window type 2003

当应用程序在具有 Android 8 和 9 的设备上运行时会出现此问题。我寻找类似问题的解决方案,我发现使用WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY而不是更好TYPE_SYSTEM_ALERT,然后我修改了我编写的代码的倒数第二行在这篇文章中但它没有改变任何东西,应用程序崩溃相同并且在日志中出现此错误:

android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$W@b784d3a -- permission denied for window type 2038

在清单中,我得到了许可:

<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>

我还从设备设置中启用了应用程序的所有权限。我怎么解决这个问题?

4

1 回答 1

10

如您所见,Android Oreo 及更高版本严重限制了您可以使用的覆盖类型。Oreo 引入了新的TYPE_APPLICATION_OVERLAY常量供应用程序使用,同时弃用和禁止旧常量。

但是,TYPE_APPLICATION_OVERLAY在 Nougat 及以下版本中不存在,因此尝试在那里使用它会导致 SecurityException。

代替

mProgressDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);

mProgressDialog.getWindow().setType(
        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O)
            WindowManager.LayoutParams.TYPE_SYSTEM_ALERT
        else 
            WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY);

至于为什么它仍然在 Oreo 及更高版本上崩溃TYPE_APPLICATION_OVERLAY,该SYSTEM_ALERT_WINDOW权限不是正常的运行时权限,并且不会显示在应用程序信息的“权限”部分下。它的面向用户的名称是“在其他应用程序上绘制”(通常),它位于通用应用程序信息页面或设置>>应用程序>>特殊访问权限下。

您不能像使用运行时权限那样以编程方式授予它,您可以通过一个简单的对话框来授予它。相反,您需要检查是否授予了权限,如果没有,则启动设置页面以允许用户授予它:How to programmatically grant the "draw over other apps" permission in android?


但是,我真的不明白为什么一个简单的进度对话框需要成为一个完整的覆盖。事实上,ProgressDialog已被完全弃用。您应该考虑迁移到简单的 ProgressBar,或者,如果您想保留 ProgressDialog,请删除该setType()行。

于 2018-10-21T16:34:02.247 回答