Запретить строку состояния для появления Android (изменено)

Я реализую приложение в режиме киоска, и я успешно сделал приложение полноэкранным без появления строки состояния после 4.3, но не могу скрыть строку состояния в 4.3 и 4.4, так как строка состояния появляется, когда мы проводим вниз в верхней части экрана.

Я попытался сделать это на весь экран

  • указание полноэкранной темы в манифесте
  • окно настроек Flags т.е. setFlags
  • setSystemUiVisibility

Возможно дублирование, но конкретное решение не найдено

Навсегда скрыть строку состояния Android

Наконец, что я хочу, так это как постоянно скрывать строку состояния в активности? в андроиде 4.3,4.4,5,6 версии

2 ответа

Решение

Мы не могли предотвратить появление статуса в полноэкранном режиме на устройствах kitkat, поэтому сделали хак, который все еще удовлетворяет требованию, т.е. блокирует расширение строки состояния.

Чтобы это работало, приложение не было полноэкранным. Мы наложили на строку состояния и использовали все входные события. Это препятствовало расширению статуса.

нота:

  • customViewGroup - это пользовательский класс, который расширяет любой макет (фрейм, относительный макет и т. д.) и использует событие касания.
  • для использования события касания переопределяет метод onInterceptTouchEvent группы представлений и возвращает true

обновленный

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

реализация customViewGroup

Код:

WindowManager manager = ((WindowManager) getApplicationContext()
            .getSystemService(Context.WINDOW_SERVICE));

WindowManager.LayoutParams localLayoutParams = new WindowManager.LayoutParams();
localLayoutParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_ERROR;
localLayoutParams.gravity = Gravity.TOP;
localLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE|

            // this is to enable the notification to recieve touch events
            WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL |

            // Draws over status bar
            WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN;

    localLayoutParams.width = WindowManager.LayoutParams.MATCH_PARENT;
    localLayoutParams.height = (int) (50 * getResources()
            .getDisplayMetrics().scaledDensity);
    localLayoutParams.format = PixelFormat.TRANSPARENT;

    customViewGroup view = new customViewGroup(this);

    manager.addView(view, localLayoutParams);

В Android M вам нужно получить дополнительное разрешение на наложение. android.permission.SYSTEM_ALERT_WINDOW недостаточно! Поэтому я использовал код из ответа Abhimaan в disableStatusBar() и должен был сделать намерение открыть правильный диалог настроек. Я также добавил удаление просмотра в onDestroy() для того, чтобы включить строку состояния, когда приложение выходит. Я также уменьшил высоту наложения до 40, так как этого достаточно. Код работает с 5.1 и 6.0 здесь.

public static final int OVERLAY_PERMISSION_REQ_CODE = 4545;
protected CustomViewGroup blockingView = null;

@Override
protected void onDestroy() {

    super.onDestroy();

    if (blockingView!=null) {
        WindowManager manager = ((WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE));
        manager.removeView(blockingView);
    }
}


@Override
protected void onCreate(Bundle savedInstanceState) {

        if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {

            if (!Settings.canDrawOverlays(this)) {
                Toast.makeText(this, "Please give my app this permission!", Toast.LENGTH_SHORT).show();
                Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,Uri.parse("package:" + getPackageName()));
                startActivityForResult(intent, OVERLAY_PERMISSION_REQ_CODE);
            } else {
                disableStatusBar();
            }
        }
        else {
            disableStatusBar();
        }
}


@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {

    super.onActivityResult(requestCode, resultCode, data);

    if (requestCode == OVERLAY_PERMISSION_REQ_CODE) {
        if (!Settings.canDrawOverlays(this)) {
            Toast.makeText(this, "User can access system settings without this permission!", Toast.LENGTH_SHORT).show();
        }
        else
        { disableStatusBar();
        }
    }
}

protected void disableStatusBar() {

    WindowManager manager = ((WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE));

    WindowManager.LayoutParams localLayoutParams = new WindowManager.LayoutParams();
    localLayoutParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_ERROR;
    localLayoutParams.gravity = Gravity.TOP;
    localLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE |

            // this is to enable the notification to receive touch events
            WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL |

            // Draws over status bar
            WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN;

    localLayoutParams.width = WindowManager.LayoutParams.MATCH_PARENT;
    localLayoutParams.height = (int) (40 * getResources().getDisplayMetrics().scaledDensity);
    localLayoutParams.format = PixelFormat.TRANSPARENT;

    blockingView = new CustomViewGroup(this);
    manager.addView(blockingView, localLayoutParams);
}

Для проекта, над которым я работал, я нашел решение для этого, но это заняло много времени. Различные темы на Stackru и в других местах помогли мне придумать это. Это был обходной путь на Android M, но он работал отлично. Как кто-то просил об этом, так что я подумал, что я должен опубликовать это здесь, если это может кому-нибудь помочь.

Теперь, когда прошло некоторое время, я не помню всех деталей, но CustomViewGroup - это класс, который переопределяет основную ViewGroup и обнаруживает, что пользователь провел пальцем сверху, чтобы показать строку состояния. Но мы не хотели показывать это, поэтому перехват пользователя был обнаружен, и любые дальнейшие действия были проигнорированы, то есть ОС Android не получит сигнал, чтобы открыть скрытую строку состояния.

А затем включены методы для отображения и скрытия строки состояния, которые вы можете копировать / вставлять как есть в вашем коде, где вы хотите показать / скрыть строку состояния.

/**
 * This class creates the overlay on the status bar which stops it from expanding.
 */
public static class CustomViewGroup extends ViewGroup {

    public CustomViewGroup(Context context) {
        super(context);
    }

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
    }

    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        Log.v("customViewGroup", "********** Status bar swipe intercepted");
        return true;
    }
}

public static void allowStatusBarExpansion(Context context) {
    CustomViewGroup view = new CustomViewGroup(context);
    WindowManager manager = ((WindowManager) context.getApplicationContext()
            .getSystemService(Context.WINDOW_SERVICE));
    manager.removeView(view);
}

// Stop expansion of the status bar on swipe down.
public static void preventStatusBarExpansion(Context context) {
    WindowManager manager = ((WindowManager) context.getApplicationContext()
            .getSystemService(Context.WINDOW_SERVICE));

    Activity activity = (Activity) context;
    WindowManager.LayoutParams localLayoutParams = new WindowManager.LayoutParams();
    localLayoutParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_ERROR;
    localLayoutParams.gravity = Gravity.TOP;
    localLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE |

            // this is to enable the notification to receive touch events
            WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL |

            // Draws over status bar
            WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN;

    localLayoutParams.width = WindowManager.LayoutParams.MATCH_PARENT;
    //http://stackru.com/questions/1016896/get-screen-dimensions-in-pixels
    int resId = activity.getResources().getIdentifier("status_bar_height", "dimen", "android");
    int result = 0;
    if (resId > 0) {
        result = activity.getResources().getDimensionPixelSize(resId);
    }

    localLayoutParams.height = result;

    localLayoutParams.format = PixelFormat.TRANSPARENT;

    CustomViewGroup view = new CustomViewGroup(context);

    manager.addView(view, localLayoutParams);
}
Другие вопросы по тегам