Swift 修改状态栏颜色/样式(重写 preferredStatusBarStyle 无效的问题)

默认情况下,顶部状态栏(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 属性值来更新状态栏样式。

你可能感兴趣的:(Swift 修改状态栏颜色/样式(重写 preferredStatusBarStyle 无效的问题))