隐藏状态栏

本课程介绍如何在不同版本的 Android 上隐藏状态栏。隐藏状态栏(以及可选的导航栏)可以让内容使用更多的显示空间,从而提供更沉浸式的用户体验。

图 1 显示了一个带有可见状态栏的应用

system bars

图 1. 可见状态栏。

图 2 显示了一个隐藏状态栏的应用。请注意,操作栏也隐藏了。您永远不应该在没有状态栏的情况下显示操作栏。

system bars

图 2. 隐藏状态栏。

在 Android 4.0 及更低版本上隐藏状态栏

您可以通过设置 WindowManager 标志来隐藏 Android 4.0(API 级别 14)及更低版本上的状态栏。您可以通过编程方式或通过在应用的清单文件中设置活动主题来执行此操作。如果状态栏应始终在应用中保持隐藏状态,则建议在应用的清单文件中设置活动主题(尽管严格来说,如果需要,您可以通过编程方式覆盖主题)。例如

<application
    ...
    android:theme="@android:style/Theme.Holo.NoActionBar.Fullscreen" >
    ...
</application>

使用活动主题的优点如下

  • 与通过编程方式设置标志相比,它更易于维护且错误更少。
  • 它会导致更流畅的 UI 过渡,因为系统在实例化应用的主活动之前拥有渲染 UI 所需的信息。

或者,您可以通过编程方式设置 WindowManager 标志。此方法可以更轻松地在用户与应用交互时隐藏和显示状态栏

Kotlin

class MainActivity : Activity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        // If the Android version is lower than Jellybean, use this call to hide
        // the status bar.
        if (Build.VERSION.SDK_INT < 16) {
            window.setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
                    WindowManager.LayoutParams.FLAG_FULLSCREEN)
        }
        setContentView(R.layout.activity_main)
    }
    ...
}

Java

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        // If the Android version is lower than Jellybean, use this call to hide
        // the status bar.
        if (Build.VERSION.SDK_INT < 16) {
            getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
                    WindowManager.LayoutParams.FLAG_FULLSCREEN);
        }
        setContentView(R.layout.activity_main);
    }
    ...
}

当您设置 WindowManager 标志(无论是通过活动主题还是通过编程方式)时,这些标志将保持有效,除非您的应用清除它们。

您可以使用 FLAG_LAYOUT_IN_SCREEN 将活动布局设置为使用启用 FLAG_FULLSCREEN 时可用的相同屏幕区域。这可以防止在状态栏隐藏和显示时内容调整大小。

在 Android 4.1 及更高版本上隐藏状态栏

您可以使用 setSystemUiVisibility() 来隐藏 Android 4.1(API 级别 16)及更高版本上的状态栏。setSystemUiVisibility() 在单个视图级别设置 UI 标志;这些设置会聚合到窗口级别。使用 setSystemUiVisibility() 设置 UI 标志可以让您比使用 WindowManager 标志更精细地控制系统栏。此代码段隐藏状态栏

Kotlin

// Hide the status bar.
window.decorView.systemUiVisibility = View.SYSTEM_UI_FLAG_FULLSCREEN
// Remember that you should never show the action bar if the
// status bar is hidden, so hide that too if necessary.
actionBar?.hide()

Java

View decorView = getWindow().getDecorView();
// Hide the status bar.
int uiOptions = View.SYSTEM_UI_FLAG_FULLSCREEN;
decorView.setSystemUiVisibility(uiOptions);
// Remember that you should never show the action bar if the
// status bar is hidden, so hide that too if necessary.
ActionBar actionBar = getActionBar();
actionBar.hide();

请注意以下事项

  • 一旦 UI 标志被清除(例如,通过从活动导航离开),如果要再次隐藏栏,您的应用需要重置它们。有关如何侦听 UI 可见性更改以便应用可以相应地做出响应的讨论,请参阅 响应 UI 可见性更改
  • 设置 UI 标志的位置会产生影响。如果您在活动的 onCreate() 方法中隐藏系统栏,并且用户按下了“主页”,则系统栏将重新出现。当用户重新打开活动时,onCreate() 不会被调用,因此系统栏将保持可见。如果希望系统 UI 更改在用户在活动中进出导航时保持不变,请在 onResume()onWindowFocusChanged() 中设置 UI 标志。
  • setSystemUiVisibility() 方法仅在您从中调用的视图可见时才有效。
  • 从视图导航离开会导致使用 setSystemUiVisibility() 设置的标志被清除。

使内容显示在状态栏后面

在 Android 4.1 及更高版本中,您可以设置应用程序的内容显示在状态栏后面,这样内容在状态栏隐藏和显示时就不会调整大小。为此,请使用 SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN。您可能还需要使用 SYSTEM_UI_FLAG_LAYOUT_STABLE 来帮助您的应用保持稳定的布局。

当您使用此方法时,您需要确保应用程序 UI 的关键部分(例如,地图应用程序中的内置控件)不会被系统栏覆盖。这可能会导致您的应用程序无法使用。在大多数情况下,您可以通过在 XML 布局文件中添加 android:fitsSystemWindows 属性并将其设置为 true 来解决此问题。这会调整父 ViewGroup 的填充以留出系统窗口的空间。这对于大多数应用程序来说已经足够了。

但是,在某些情况下,您可能需要修改默认填充以获得应用程序所需的布局。要直接操作内容相对于系统栏(占用窗口“内容插入”空间)的布局方式,请覆盖 fitSystemWindows(Rect insets)。当窗口的内容插入发生更改时,视图层次结构会调用 fitSystemWindows() 方法,以允许窗口相应地调整其内容。通过覆盖此方法,您可以根据需要处理插入(以及应用程序的布局)。