iOS状态栏详解
JamelThatch
8年前
<h2>状态栏的隐藏</h2> <p>状态栏的隐藏主要有两种方法:</p> <p><strong>方法一:通过代码控制</strong></p> <pre> <code class="language-objectivec">@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;</code></pre> <p>注意:让我们先来看看 // Setting statusBarHidden does nothing if your application is using the default UIViewController-based status bar system. 这个注释提示,苹果提示开发者 如果使用的是系统基础的状态栏样式你的这些设置是不生效的 ,在接下来要介绍的通过 Info.plist 隐藏状态栏同样要注意这件事。</p> <p>在 Info.plist 中添加一个 View controller-based status bar appearance 设置选项,设置为 NO 这样就可以使用上边的方法了</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/2515631371f2eda03f9ce60a01d065a6.png"></p> <p>注意:添加的View controller-based status bar appearance是Bool类型,默认为Yes</p> <p>很不幸iOS9之后苹果已经不推荐使用这些方法了,这些方法能用但是会报警告。</p> <p>那么这些方法被禁用以后,如何操作呢?注释里已经提示 Use -[UIViewController prefersStatusBarHidden] 这是iOS7之后苹果在 UIViewController 里添加的新方法,这么做的目的可以让开发者更加灵活的自定义每个 ViewController 的状态栏。</p> <pre> <code class="language-objectivec">- (BOOL)prefersStatusBarHidden{ return YES; }</code></pre> <p>iOS7之后 UIViewController 中不只提供了这个关于状态栏的设置的函数,还有其他的,后面详细说。</p> <p><strong>方法二:通过 Info.plist 控制</strong></p> <p>1,首先我们依然要设置这个(第2步中的两种方式都要设置这个参数)</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/2515631371f2eda03f9ce60a01d065a6.png" alt="iOS状态栏详解" width="1040" height="134"></p> <p>2,然后设置(两种方式)</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/f837f7412eae5becbf823e5e02744777.png"></p> <p>或者</p> <p><img src="https://simg.open-open.com/show/8a8ff4b7e5806ab8ad79e7563058c1e8.png"></p> <p>两者是等效的!并且两者的状态是同步的。</p> <h2>状态栏样式</h2> <p>先看看都有哪些样式(解释看注释)</p> <pre> <code class="language-objectivec">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;</code></pre> <p>如何设置状态栏样式</p> <pre> <code class="language-objectivec">// 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;</code></pre> <p>同样iOS9以后这些方法被禁用了,苹果推荐在具体的 viewController 中 Use -[UIViewController preferredStatusBarStyle]</p> <pre> <code class="language-objectivec">- (UIStatusBarStyle)preferredStatusBarStyle{ return UIStatusBarStyleLightContent; }</code></pre> <p>注意:我们通常使用的 viewController 都是嵌套在 UINavigationController 中使用的,此时在 viewController 中使用</p> <p>- (UIStatusBarStyle)preferredStatusBarStyle; 函数会发现设置并没有生效。</p> <p>除了以上博客中的方法外,系统也给我们提供了一个函数 - (UIViewController *)childViewControllerForStatusBarStyle ,也可以解决这个问题,后面会讲。</p> <h2>背景色</h2> <p>iOS7以后默认情况下状态栏的背景为透明的,一种办法是我们自己写一个UIView作为背景添加到状态栏下面,这样就可以随意设置状态栏的颜色了。</p> <p>另一种方法就是通过设置 navigationBar 的 setBarTintColor 颜色来改变状态栏颜色</p> <h2>UIViewController中其他有关状态栏的函数</h2> <ul> <li><strong>preferredStatusBarUpdateAnimation函数</strong></li> </ul> <pre> <code class="language-objectivec">- (UIStatusBarAnimation)preferredStatusBarUpdateAnimation NS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED; // Defaults to UIStatusBarAnimationFade</code></pre> <p>该函数是用来设置状态栏显示、消失时的动画类型,动画类型有:</p> <pre> <code class="language-objectivec">typedef NS_ENUM(NSInteger, UIStatusBarAnimation) { UIStatusBarAnimationNone, UIStatusBarAnimationFade NS_ENUM_AVAILABLE_IOS(3_2), UIStatusBarAnimationSlide NS_ENUM_AVAILABLE_IOS(3_2), } __TVOS_PROHIBITED;</code></pre> <p>默认为 UIStatusBarAnimationFade ,当状态栏隐藏、显示状态发生改变的时候,该函数的返回值就会发挥作用。</p> <ul> <li><strong>childViewControllerForStatusBarStyle函数</strong></li> </ul> <pre> <code class="language-objectivec">// 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;</code></pre> <p>这个函数的返回值默认返回 nil ,此时系统就会调用当前 viewControllerA 的 preferredStatusBarStyle 函数;如果返回值是另一个 viewControllerB 那么系统就会调用 viewControllerB 的 preferredStatusBarStyle 函数。</p> <p>运用这个函数就可以解决嵌套 UINavigationController 设置样式无效的问题。</p> <p>解释一下为什么嵌套 UINavigationController 的 viewController 的 preferredStatusBarStyle 函数设置无效:</p> <p>在我们嵌套了 UINavigationController 的时候,我们的</p> <p>AppDelegate.window.rootViewController</p> <p>通常是我们创建的 navigationController ,这时首先会调用的是 navigationController 中的 childViewControllerForStatusBarStyle 函数,因为默认返回 nil ,那么接下来就会调用 navigationController 本身的 preferredStatusBarStyle 函数,所以我们在 viewController 中通过 preferredStatusBarStyle 函数设置的状态栏样式就不会被调用发现,所以也就无效了。</p> <p>所以我们要自己创建一个继承于 UINavigationcontroller 的</p> <p>NavigationController ,在这个子类中重写</p> <p>childViewControllerForStatusBarStyle 函数</p> <pre> <code class="language-objectivec">- (UIViewController *)childViewControllerForStatusBarStyle{ return self.topViewController; }</code></pre> <p>这样 navigationController 中的 childViewControllerForStatusBarStyle 函数会返回 navigationController 中最上层的 viewController ,那么 viewController 中的 preferredStatusBarStyle 函数的设置就会被系统获知</p> <ul> <li><strong>childViewControllerForStatusBarHidden函数</strong></li> </ul> <pre> <code class="language-objectivec">- (nullable UIViewController *)childViewControllerForStatusBarHidden NS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED;</code></pre> <p>childViewControllerForStatusBarHidden 函数的使用原理同上,不再赘述。</p> <ul> <li><strong>preferredStatusBarUpdateAnimation函数</strong></li> </ul> <pre> <code class="language-objectivec">// 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</code></pre> <p>动画形式如下</p> <pre> <code class="language-objectivec">typedef NS_ENUM(NSInteger, UIStatusBarAnimation) { UIStatusBarAnimationNone, UIStatusBarAnimationFade NS_ENUM_AVAILABLE_IOS(3_2), UIStatusBarAnimationSlide NS_ENUM_AVAILABLE_IOS(3_2), } __TVOS_PROHIBITED;</code></pre> <p>这个函数返回了动画效果。动画效果只有在 prefersStatusBarHidden</p> <p>函数返回值变化的时候才会展示,同时要通过调用</p> <p>[self setNeedsStatusBarAppearanceUpdate] 函数来重绘状态栏</p> <h2>应用</h2> <p>我们可以通过隐藏系统状态栏,然后自定义 UIWindow<br> 通过设置 setWindowLevel:UIWindowLevelStatusBar 实现自定义状态栏。</p> <p> </p> <p> </p> <p>来自:http://www.jianshu.com/p/4196d7cf95f4</p> <p> </p>