很多场景下UI设计会要求状态栏文字展示为白色
重写 preferredStatusBarStyle 方法为什么有的地方生效,有的地方却不生效呢 ?
接下来我们做一下简单的分析和介绍 :
首先,控制器分为以下几种情况:
- 没有使用导航控制器的情况
- 使用了系统导航控制器的情况
- 使用了自定义导航控制器的情况
前提条件:
plist文件里面要设置 View controller-based status bar appearance为YES(系统默认为YES,不要误设为 NO)
1.没有使用导航控制器的情况
如果没有使用导航控制器 UINavigationController,那么我们只需要在 UIViewController 中重写 preferredStatusBarStyle 可读属性,在具体实现中返回希望使用的样式即可。然后在需要更新状态栏样式的时候,调用 setNeedsStatusBarAppearanceUpdate() 方法来触发它。
例如:将系统默认的状态栏样式修改为 lightContent,我们可以在控制器的 viewDidLoad() 方法中去触发它:
override func viewDidLoad() {
super.viewDidLoad()
// 主动触发来更新状态栏样式
setNeedsStatusBarAppearanceUpdate()
}
override var preferredStatusBarStyle: UIStatusBarStyle {
return .lightContent
}
2.使用了系统导航控制器的情况
如果我们使用了系统导航控制器 UINavigationController,那么即使在 UIViewController 中重写了 preferredStatusBarStyle 可读属性,并且主动去调用了 setNeedsStatusBarAppearanceUpdate() 方法,preferredStatusBarStyle 属性也不会被调用,因为此时状态栏的样式是根据导航栏的样式进行自动变换的。
如果我们想要修改状态栏的样式,我们就要主动去修改导航栏的样式。当然,这种情况下我们就不需要去重写 preferredStatusBarStyle 属性了。
例如:修改导航栏样式为 .black,导航栏文字将为白色,状态栏样式将自动变为 .lightContent:
override func viewDidLoad() {
super.viewDidLoad()
navigationController?.navigationBar.barStyle = .black
}
3.使用了自定义导航控制器的情况
如果我们使用了自定义导航控制器,那么即使我们做了如上述第2种情况的准备工作,状态栏样式也不能跟随导航栏样式进行改变。这种情况下,我们要在自定义的导航控制器中重写 childForStatusBarStyle 可读属性,并返回 topViewController。如下:
class CusstomNavigationController: UINavigationController {
//重写 childForStatusBarStyle 方法
override var childForStatusBarStyle: UIViewController? {
return topViewController
}
}
在想要改变状态栏样式的控制器中,只需要重写 preferredStatusBarStyle 可读属性即可实现状态栏样式改变:
override var preferredStatusBarStyle: UIStatusBarStyle {
return .lightContent
}
这种情况不用修改导航栏样式(不用写 navigationController?.navigationBar.barStyle = .black),也不用主动去调用 setNeedsStatusBarAppearanceUpdate() 方法,因为自定义的 CusstomNavigationController 导航控制器中重写的 childForStatusBarStyle 属性中返回了 topViewController 就是为了告诉系统去调用栈顶的控制器(topViewController)里边的 preferredStatusBarStyle 属性值来更新状态栏样式。