@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
中不只提供了这个关于状态栏的设置的函数,还有其他的,后面详细说。
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
,也可以解决这个问题,后面会讲。UIView
作为背景添加到状态栏下面,这样就可以随意设置状态栏的颜色了。navigationBar
的setBarTintColor
颜色来改变状态栏颜色- (UIStatusBarAnimation)preferredStatusBarUpdateAnimation NS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED; // Defaults to UIStatusBarAnimationFade如果想在当前已经显示的
UIViewController
中更改状态栏的样式的话,需要调用以上函数。调用该函数后,系统会主动调用preferredStatusBarStyle
方法重绘状态栏的样式// 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
函数的设置就会被系统获知- (nullable UIViewController *)childViewControllerForStatusBarHidden NS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED;
childViewControllerForStatusBarHidden
函数的使用原理同上,不再赘述。// 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
实现自定义状态栏。