默认情况下,顶部状态栏(statusBar)为 default 样式(即黑色文字和图标),但我们有时会希望可以修改状态栏的样,这样,我们的应用看起来会更加协调。
我们针对不同的情况,所做的处理也不尽相同,这些情况包括:
没有使用导航控制器的情况
使用了系统导航控制器的情况
使用了自定义导航控制器的情况
前提条件:
plist文件里面要设置 View controller-based status bar appearance为YES(系统默认为YES,不要误设为 NO)。
没有使用导航控制器的情况
如果没有使用导航控制器 UINavigationController,那么我们只需要在 UIViewController 中重写 preferredStatusBarStyle 可读属性,在具体实现中返回希望使用的样式即可。然后在需要更新状态栏样式的时候,调用 setNeedsStatusBarAppearanceUpdate() 方法来触发它。
例如:将系统默认的状态栏样式修改为 lightContent,我们可以在控制器的 viewDidLoad() 方法中去触发它:
override var preferredStatusBarStyle: UIStatusBarStyle {
return .lightContent
}
override func viewDidLoad() {
super.viewDidLoad()
// 主动触发来更新状态栏样式
setNeedsStatusBarAppearanceUpdate()
}
使用了系统导航控制器的情况
如果我们使用了系统导航控制器 UINavigationController,那么即使在 UIViewController 中重写了 preferredStatusBarStyle 可读属性,并且主动去调用了 setNeedsStatusBarAppearanceUpdate() 方法,preferredStatusBarStyle 属性也不会被调用,因为此时状态栏的样式是根据导航栏的样式进行自动变换的。
如果我们想要修改状态栏的样式,我们就要主动去修改导航栏的样式。当然,这种情况下我们就不需要去重写 preferredStatusBarStyle 属性了。
例如:修改导航栏样式为 .black,导航栏文字将为白色,状态栏样式将自动变为 .lightContent:
override func viewDidLoad() {
super.viewDidLoad()
navigationController?.navigationBar.barStyle = .black
}
使用了自定义导航控制器的情况
如果我们使用了自定义导航控制器,那么即使我们做了如上述第2种情况的准备工作,状态栏样式也不能跟随导航栏样式进行改变。这种情况下,我们要在自定义的导航控制器中重写 childForStatusBarStyle 可读属性,并返回 topViewController。如下:
class CusstomNavigationController: UINavigationController {
override var childForStatusBarStyle: UIViewController? {
return topViewController
}
}
在想要改变状态栏样式的控制器中,只需要重写 preferredStatusBarStyle 可读属性即可实现状态栏样式改变:
override var preferredStatusBarStyle: UIStatusBarStyle {
return .lightContent
}
这种情况不用修改导航栏样式(不用写 navigationController?.navigationBar.barStyle = .black),也不用主动去调用 setNeedsStatusBarAppearanceUpdate() 方法,因为自定义的 CusstomNavigationController 导航控制器中重写的 childForStatusBarStyle 属性中返回了 topViewController 就是为了告诉系统去调用栈顶的控制器(topViewController)里边的 preferredStatusBarStyle 属性值来更新状态栏样式。