仅在横向模式下使用沉浸式模式

Using immersive mode only on landscape mode

我尝试制作将在横向(没有状态栏和导航栏)上进入沉浸式模式并在移动模式(有 return 状态栏和导航栏)上禁用沉浸式模式的应用。无需重新启动 activity(onCreate 仅调用一次)。 我写了一个小示例应用程序来展示我的问题。

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="test.com.immersivetest">
    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity
            android:name=".MainActivity"
            android:configChanges="orientation|screenSize">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
</manifest>

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true"
    android:padding="0dp"
    tools:context="test.com.immersivetest.MainActivity">

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="#ffff0000"
         />
</RelativeLayout>

MainActivity.java

package test.com.immersivetest;

import android.content.res.Configuration;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;

public class MainActivity extends AppCompatActivity {

    private View activity;
    private View decorView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        activity = findViewById(R.id.activity_main);
        decorView = this.getWindow().getDecorView();
        System.out.println("On Crate");
    }

    @Override
    public void onConfigurationChanged(Configuration newConfig) {
        super.onConfigurationChanged(newConfig);
        updateForOrientation(newConfig.orientation);
    }

    public void updateForOrientation(int orientation) {
        if (orientation == Configuration.ORIENTATION_LANDSCAPE) {
            hideSystemUI();
        } else if (orientation == Configuration.ORIENTATION_PORTRAIT) {
            showSystemUI();
        }
    }



    private void hideSystemUI() {
        System.out.println("Hide UI");
        activity.setFitsSystemWindows(false);
        decorView.setSystemUiVisibility(
                View.SYSTEM_UI_FLAG_LAYOUT_STABLE
                        | View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
                        | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
                        | View.SYSTEM_UI_FLAG_HIDE_NAVIGATION // hide nav bar
                        | View.SYSTEM_UI_FLAG_FULLSCREEN // hide status bar
                        | View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY);

    }

    private void showSystemUI() {
        activity.setFitsSystemWindows(true);
        decorView.setSystemUiVisibility(
                View.SYSTEM_UI_FLAG_LAYOUT_STABLE
                        | View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
                        | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN);
    }


}

查看我的问题的步骤:

1) 启动应用程序(纵向模式),结果我在全屏上看到 child 带有红色背景的 RelativeLayout,带有标题栏、状态栏和导航栏。这是正确的。截图

2) 旋转到横向。我再次看到正确的结果:Child 在没有标题栏、状态栏、导航栏的沉浸式模式下具有红色背景的 RelativeLayout。

3) 旋转到纵向。并查看正确的结果:就像第一步的屏幕截图一样。

4) 再次旋转到横向。而且我看到不正确的结果。 Child RelativeLayout 未拉伸到 parent 大小。我们在屏幕顶部和底部看到空白字段。有问题的截图

如何解决?为什么所有旋转到横向而不是第一个不正确调整大小 child RelativeLayout?

P.s。最小 SDK 版本 19.

我自己找到了解决方案 ;) 我有来自 https://developer.android.com/training/system-ui/immersive.html#nonsticky 的片段。它不能正常工作。我更改了 showSystemUI() 中的标志,并且一切正常!

private void showSystemUI() {
    activity.setFitsSystemWindows(true);
    decorView.setSystemUiVisibility(View.SYSTEM_UI_FLAG_VISIBLE);
}

但它在我的主应用程序中有效,因为我不使用操作栏。如果启用操作栏,则显示的操作栏不正确。我不知道为什么。无论如何,我在启动应用程序时检查了标志,它只有 SYSTEM_UI_FLAG_VISIBLE.

如果有人对带有操作栏的应用程序有解决方案,我会很高兴知道解决方案。