深入理解iOS的状态栏
一、状态栏的隐藏
状态栏的隐藏主要有两种方法,下面来一起看看吧。
方法一:通过代码控制
@interface UIApplication(UIApplicationDeprecated) // Setting statusBarHidden does nothing if your application is using the default UIViewController-based status bar system. @property(readwrite, nonatomic,getter=isStatusBarHidden) BOOL statusBarHidden NS_DEPRECATED_IOS(2_0, 9_0, "Use -[UIViewController prefersStatusBarHidden]") __TVOS_PROHIBITED; - (void)setStatusBarHidden:(BOOL)hidden animated:(BOOL)animated NS_DEPRECATED_IOS(2_0, 3_2) __TVOS_PROHIBITED; // use -setStatusBarHidden:withAnimation: - (void)setStatusBarHidden:(BOOL)hidden withAnimation:(UIStatusBarAnimation)animation NS_DEPRECATED_IOS(3_2, 9_0, "Use -[UIViewController prefersStatusBarHidden]") __TVOS_PROHIBITED;
注意:让我们先来看看// Setting statusBarHidden does nothing if your application is using the default UIViewController-based status bar system.
这个注释提示,苹果提示开发者如果使用的是系统基础的状态栏样式你的这些设置是不生效的,在接下来要介绍的通过Info.plist隐藏状态栏同样要注意这件事。
在Info.plist
中添加一个View controller-based status bar appearance
设置选项,设置为NO这样就可以使用上边的方法了
注意:添加的View controller-based status bar appearance
是Bool
类型,默认为Yes,很不幸iOS9之后苹果已经不推荐使用这些方法了,这些方法能用但是会报警告。
那么这些方法被禁用以后,如何操作呢?注释里已经提示Use -[UIViewController prefersStatusBarHidden]
这是iOS7之后苹果在UIViewController
里添加的新方法,这么做的目的可以让开发者更加灵活的自定义每个ViewController
的状态栏。
- (BOOL)prefersStatusBarHidden{ return YES; }
iOS7之后UIViewController
中不只提供了这个关于状态栏的设置的函数,还有其他的,后面详细说。
方法二:通过Info.plist控制
1,首先我们依然要设置这个(第2步中的两种方式都要设置这个参数)
2,然后设置(两种方式)
或者
两者是等效的!并且两者的状态是同步的。
二、状态栏样式
先看看都有哪些样式(解释看注释)
typedef NS_ENUM(NSInteger, UIStatusBarStyle) { //默认样式,黑字透明状态栏,适合用于背景色为亮色的页面 UIStatusBarStyleDefault = 0, // Dark content, for use on light backgrounds //白字透明状态栏,适合用于背景色为暗色的页面 UIStatusBarStyleLightContent NS_ENUM_AVAILABLE_IOS(7_0) = 1, // Light content, for use on dark backgrounds // iOS7.0以前黑底白字,iOS7以后跟UIStatusBarStyleLightContent效果一样 UIStatusBarStyleBlackTranslucent NS_ENUM_DEPRECATED_IOS(2_0, 7_0, "Use UIStatusBarStyleLightContent") = 1, // iOS7.0以前启动页为灰底白字,iOS7以后跟UIStatusBarStyleLightContent效果一样 UIStatusBarStyleBlackOpaque NS_ENUM_DEPRECATED_IOS(2_0, 7_0, "Use UIStatusBarStyleLightContent") = 2, } __TVOS_PROHIBITED;
如何设置状态栏样式
// Setting the statusBarStyle does nothing if your application is using the default UIViewController-based status bar system. @property(readwrite, nonatomic) UIStatusBarStyle statusBarStyle NS_DEPRECATED_IOS(2_0, 9_0, "Use -[UIViewController preferredStatusBarStyle]") __TVOS_PROHIBITED; - (void)setStatusBarStyle:(UIStatusBarStyle)statusBarStyle animated:(BOOL)animated NS_DEPRECATED_IOS(2_0, 9_0, "Use -[UIViewController preferredStatusBarStyle]") __TVOS_PROHIBITED;
同样iOS9以后这些方法被禁用了,苹果推荐在具体的viewController
中Use -[UIViewController preferredStatusBarStyle]
- (UIStatusBarStyle)preferredStatusBarStyle{ return UIStatusBarStyleLightContent; }
注意:我们通常使用的viewController
都是嵌套在UINavigationController
中使用的,此时在viewController
中使用- (UIStatusBarStyle)preferredStatusBarStyle;
函数会发现设置并没有生效。
系统也给我们提供了一个函数- (UIViewController *)childViewControllerForStatusBarStyle
,也可以解决这个问题,后面会讲。
三、背景色
iOS7以后默认情况下状态栏的背景为透明的,一种办法是我们自己写一个UIView
作为背景添加到状态栏下面,这样就可以随意设置状态栏的颜色了。
另一种方法就是通过设置navigationBar
的setBarTintColor
颜色来改变状态栏颜色
UIViewController中其他有关状态栏的函数
preferredStatusBarUpdateAnimation函数
- (UIStatusBarAnimation)preferredStatusBarUpdateAnimation NS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED; // Defaults to UIStatusBarAnimationFade
如果想在当前已经显示的UIViewController
中更改状态栏的样式的话,需要调用以上函数。调用该函数后,系统会主动调用preferredStatusBarStyle
方法重绘状态栏的样式
childViewControllerForStatusBarStyle函数
// Override to return a child view controller or nil. If non-nil, that view controller's status bar appearance attributes will be used. If nil, self is used. Whenever the return values from these methods change, -setNeedsUpdatedStatusBarAttributes should be called. - (nullable UIViewController *)childViewControllerForStatusBarStyle NS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED;
这个函数的返回值默认返回nil
,此时系统就会调用当前viewControllerA
的preferredStatusBarStyle
函数;如果返回值是另一个viewControllerB
那么系统就会调用viewControllerB
的preferredStatusBarStyle
函数。
运用这个函数就可以解决嵌套UINavigationController
设置样式无效的问题。
解释一下为什么嵌套UINavigationController
的viewController
的preferredStatusBarStyle
函数设置无效:
在我们嵌套了UINavigationController
的时候,我们的AppDelegate.window.rootViewController
通常是我们创建的navigationController
,这时首先会调用的是navigationController
中的childViewControllerForStatusBarStyle
函数,因为默认返回nil
,那么接下来就会调用navigationController
本身的preferredStatusBarStyle
函数,所以我们在viewController
中通过preferredStatusBarStyle
函数设置的状态栏样式就不会被调用发现,所以也就无效了。
所以我们要自己创建一个继承于UINavigationcontroller
的NavigationController
,在这个子类中重写childViewControllerForStatusBarStyle
函数
- (UIViewController *)childViewControllerForStatusBarStyle{ return self.topViewController; }
这样navigationController
中的childViewControllerForStatusBarStyle
函数会返回navigationController
中最上层的viewController
,那么viewController
中的preferredStatusBarStyle
函数的设置就会被系统获知
childViewControllerForStatusBarHidden函数
- (nullable UIViewController *)childViewControllerForStatusBarHidden NS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED;
childViewControllerForStatusBarHidden
函数的使用原理同上,不再赘述。
preferredStatusBarUpdateAnimation函数
// Override to return the type of animation that should be used for status bar changes for this view controller. This currently only affects changes to prefersStatusBarHidden. - (UIStatusBarAnimation)preferredStatusBarUpdateAnimation NS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED; // Defaults to UIStatusBarAnimationFade
动画形式如下
typedef NS_ENUM(NSInteger, UIStatusBarAnimation) { UIStatusBarAnimationNone, UIStatusBarAnimationFade NS_ENUM_AVAILABLE_IOS(3_2), UIStatusBarAnimationSlide NS_ENUM_AVAILABLE_IOS(3_2), } __TVOS_PROHIBITED;
这个函数返回了动画效果。动画效果只有在prefersStatusBarHidden
函数返回值变化的时候才会展示,同时要通过调用
[self setNeedsStatusBarAppearanceUpdate]
函数来重绘状态栏
四、应用
我们可以通过隐藏系统状态栏,然后自定义UIWindow
通过设置setWindowLevel:UIWindowLevelStatusBar
实现自定义状态栏。
总结
以上就是这篇文章的全部内容了,希望能对各位iOS开发者们有所帮助,如果有疑问大家可以留言交流。