【问题标题】:android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$W@c745883 - permission deniedandroid.view.WindowManager$BadTokenException: 无法添加窗口 android.view.ViewRootImpl$W@c745883 - 权限被拒绝
【发布时间】:2018-07-09 12:51:29
【问题描述】:

这是我的堆栈跟踪:

01-30 15:11:41.037 13010-13010/project.app E/AndroidRuntime: FATAL EXCEPTION: main
 Process: project.app, PID: 13010
 android.view.WindowManager$BadTokenException:
   Unable to add window android.view.ViewRootImpl$W@c745883 -- permission denied for window type 2003
   at android.view.ViewRootImpl.setView(ViewRootImpl.java:789)
   at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:356)
   at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:93)
   at android.app.Dialog.show(Dialog.java:330)
   at com.facebook.react.devsupport.DevSupportManagerImpl$4.run(DevSupportManagerImpl.java:344)
   at android.os.Handler.handleCallback(Handler.java:790)
   at android.os.Handler.dispatchMessage(Handler.java:99)
   at android.os.Looper.loop(Looper.java:164)
   at android.app.ActivityThread.main(ActivityThread.java:6494)
   at java.lang.reflect.Method.invoke(Native Method)
   at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:438)
   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:807)

我找到了一个关于 TYPE_SYSTEM_ERROR 在 Android Oreo (8) 中被弃用的答案,所以我实现了以下方法,我也发现:

public void fixAndroid() {
    WindowManager.LayoutParams params;
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
      params = new WindowManager.LayoutParams(
              WindowManager.LayoutParams.MATCH_PARENT,
              WindowManager.LayoutParams.MATCH_PARENT,
              WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY,
              WindowManager.LayoutParams.FLAG_FULLSCREEN,
              PixelFormat.TRANSLUCENT);
    } else {
      params = new WindowManager.LayoutParams(
              WindowManager.LayoutParams.MATCH_PARENT,
              WindowManager.LayoutParams.MATCH_PARENT,
              WindowManager.LayoutParams.TYPE_SYSTEM_ERROR,
              WindowManager.LayoutParams.FLAG_FULLSCREEN,
              PixelFormat.TRANSLUCENT);
    }
  }

在我的onCreate() 方法中,我有:

  @Override
  public void onCreate (Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    // Checking permissions on init
    fixAndroid();
  }

我仍然得到错误。

我正在使用 Expo SDK 21,React Native 0.48。该应用程序已与 ExpoKit 分离。

每次更改后,我都会清理我的项目,然后通过 Android Studio 在模拟器上运行它。

编辑:我在 Nexus 5X 模拟器上运行它,在 API 27 上运行。

【问题讨论】:

  • 我认为这与WindowManager.LayoutParams有关。看看会发生什么很有趣:WindowManager.LayoutParams.TYPE_TOASTWindowManager.LayoutParams.TYPE_APPLICATION_PANEL 以及 WindowManager.LayoutParams.TYPE_SYSTEM_ALERT 与:<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/> 在清单中
  • 嗨,Jon,权限存在于 AndroidManifest.xml 中 - 我会尝试您的建议。谢谢。
  • 嗨,Dan,我很乐意为您提供帮助,碰巧我试图 frenagle 来自Debugging WebView in React Native apps 的更多点可能值得一看以用于调试目的。或者你可以给我 500 分!
  • 我认为你可以使用 Ttranslucent Activity 替换对话框来处理很多安卓手机拒绝关于 TYPE_SYSTEM_ALERT 的权限

标签: android react-native android-manifest react-native-android android-permissions


【解决方案1】:

我前段时间看到了这个错误,当我将一个 native&react 应用程序更新到 SDK 26 时。问题是 react-native 函数为开发创建了红色对话框。该函数使用 TYPE_SYSTEM_ALERT 作为其类型,因此您不能在具有该版本的 react-native 的混合应用程序中使用大于 25 的 SDK 级别,除非您修补该函数使其不再使用 TYPE_SYSTEM_ALERT。

这是 0.48 中的 react-native 代码:

private void showNewError(
      final String message,
      final StackFrame[] stack,
      final int errorCookie,
      final ErrorType errorType) {
    UiThreadUtil.runOnUiThread(
        new Runnable() {
          @Override
          public void run() {
            if (mRedBoxDialog == null) {
              mRedBoxDialog = new RedBoxDialog(mApplicationContext, DevSupportManagerImpl.this, mRedBoxHandler);
              mRedBoxDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
            }
            if (mRedBoxDialog.isShowing()) {
              // Sometimes errors cause multiple errors to be thrown in JS in quick succession. Only
              // show the first and most actionable one.
              return;
            }
            mRedBoxDialog.setExceptionDetails(message, stack);
            updateLastErrorInfo(message, stack, errorCookie, errorType);
            // Only report native errors here. JS errors are reported
            // inside {@link #updateJSError} after source mapping.
            if (mRedBoxHandler != null && errorType == ErrorType.NATIVE) {
              mRedBoxHandler.handleRedbox(message, stack, RedBoxHandler.ErrorType.NATIVE);
              mRedBoxDialog.resetReporting(true);
            } else {
              mRedBoxDialog.resetReporting(false);
            }
            mRedBoxDialog.show();
          }
        });
  } 

您需要更改mRedBoxDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT); 调用其他类型。但是,这并不能保证您能够使用 SDK 26 运行您的应用程序,当我尝试使用 SDK 26 编译反应时,项目的其他部分“反应”爆炸,所以这不太可能在短期内工作期限(您可能需要开始解决其他问题)。所以你的选择是:

-将应用降级到25级;

-升级到 react 0.52,该功能不再存在,然后重试(在此之后,react 库可能无法工作)

-修补 react-native 0.48 分支中的函数并尝试修补版本。可能会出现与 SDK 相关的其他一些问题(此时的 React 仍在使用 SDK 22 进行编译)

编码愉快!

【讨论】:

    【解决方案2】:
    TYPE_SYSTEM_ALERT
    

    此常量在 API 级别 26 中已弃用。对于非系统应用。请改用TYPE_APPLICATION_OVERLAY

    【讨论】:

      【解决方案3】:

      验证 SYSTEM_ALERT_WINDOW 权限或尝试手动提供覆盖权限。

      正如@Fco P.所说,你可以尝试改变

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

      **## 如果您使用的是联想设备。请尝试另一个。此特定权限有时会在 Lenovo 设备中出现问题。****

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 2013-09-10
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 2013-08-26
        相关资源
        最近更新 更多