android.view.WindowManager$BadTokenException: невозможно добавить окно android.view.ViewRootImpl$W@c745883 - разрешение отклонено

Вот моя трассировка стека:

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.


comment
Я думаю, это как-то связано с WindowManager.LayoutParams. Интересно посмотреть, что происходит с: WindowManager.LayoutParams.TYPE_TOAST или WindowManager.LayoutParams.TYPE_APPLICATION_PANEL, а также WindowManager.LayoutParams.TYPE_SYSTEM_ALERT с: <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/> в манифесте   -  person Jon Goodwin    schedule 02.02.2018
comment
Эй, Джон, разрешение существует в AndroidManifest.xml — я попробую ваше предложение. Спасибо.   -  person Dan    schedule 02.02.2018
comment
Привет, Дэн, я готов помочь тебе, так получилось, что я пытаюсь frenagle еще несколько точек из Отладка WebView в приложениях React Native может быть полезна для целей отладки. Или вы можете предложить мне 500 баллов!   -  person Jon Goodwin    schedule 03.02.2018
comment
Я думаю, вы могли бы использовать диалоговое окно замены Ttranslucent Activity для многих телефонов Android, запрещающих разрешение на TYPE_SYSTEM_ALERT   -  person Cyrus    schedule 09.02.2018


Ответы (3)


Я видел эту ошибку некоторое время назад, когда обновлял приложение native&react до SDK 26. Проблема заключается в функции react-native, которая создает красный диалог для разработки. Эти функции используют TYPE_SYSTEM_ALERT в качестве своего типа, поэтому вы не можете использовать уровень SDK выше 25 в своем гибридном приложении с этой версией react-native, если только вы не исправите эту функцию, чтобы она больше не использовала TYPE_SYSTEM_ALERT.

это код реакции в 0.48:

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 уровня;

- Обновите до версии 0.52, где этой функции больше нет, и повторите попытку (после этого библиотеки реакции могут не работать)

-Исправьте функцию в ответной ветке 0.48 и попробуйте исправленную версию. Могут возникнуть некоторые другие проблемы, связанные с SDK (реакция на данный момент все еще компилируется с SDK 22).

Удачного кодирования!

person Fco P.    schedule 07.02.2018

TYPE_SYSTEM_ALERT

Эта константа устарела на уровне API 26. для несистемных приложений. Вместо этого используйте TYPE_APPLICATION_OVERLAY.

person Elad    schedule 11.06.2018

Проверьте разрешение SYSTEM_ALERT_WINDOW или попробуйте предоставить разрешение наложения вручную.

Как сказал @Fco P., вы можете попытаться изменить

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

**## Если вы используете устройство Lenovo. Пожалуйста, попробуйте с другим. Это конкретное разрешение иногда создает проблемы на устройстве Lenovo. ****

person Dipendra Sharma    schedule 08.02.2018