仅在横向模式下使用沉浸式模式
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.
如果有人对带有操作栏的应用程序有解决方案,我会很高兴知道解决方案。
我尝试制作将在横向(没有状态栏和导航栏)上进入沉浸式模式并在移动模式(有 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.
如果有人对带有操作栏的应用程序有解决方案,我会很高兴知道解决方案。