详解在iOS App中自定义和隐藏状态栏的方法

内容摘要
自定义状态栏
有时候,需要在状态栏上显示一些自定义信息,比如新浪微博的官方iOS客户端:告知用户信息处于发送队列、发送成功或者发送失败。



如上图,通过在状态栏显示自定义信
文章正文

自定义状态栏

有时候,需要在状态栏上显示一些自定义信息,比如新浪微博的官方iOS客户端:告知用户信息处于发送队列、发送成功或者发送失败。

如上图,通过在状态栏显示自定义信息,可以给用户友好又不影响软件使用的提示。

为此,我们显得定义一个自定义状态栏类,包含一个显示信息的Label:

复制代码 代码如下:

@interface CustomStatusBar : UIWindow 

    UILabel *_messageLabel; 

 
- (void)showStatusMessage:(NSString *)message; 
- (void)hide; 
 
@end 

接着,设置大小和系统状态栏一致,背景为黑色:
复制代码 代码如下:

self.frame = [UIApplication sharedApplication].statusBarFrame; 
self.backgroundColor = [UIColor blackColor]; 

到这里,为了让自定义的状态栏可以让用户看到,还需要设置它的windowLevel。
在iOS中,windowLevel属性决定了UIWindow的显示层次。默认的windowLevel为UIWindowLevelNormal,即0.0。

系统定义了三个层次如下,

复制代码 代码如下:

const UIWindowLevel UIWindowLevelNormal; 
const UIWindowLevel UIWindowLevelAlert; 
const UIWindowLevel UIWindowLevelStatusBar; 
typedef CGFloat UIWindowLevel; 

为了能够覆盖系统默认的状态栏,我们把自定义的状态栏的windowLevel调高点:
复制代码 代码如下:

self.windowLevel = UIWindowLevelStatusBar + 1.0f;


隐藏状态栏

如果想要隐藏状态栏,有两种做法:

状态栏是否隐藏默认由控制器管理,也就是说,当前状态栏所对应的控制器决定是否隐藏状态栏。

UIViewController中提供了一个prefersStatusBarHidden方法用于查看当前的控制器显示的状态栏是否隐藏,默认这个方法返回的是NO,也就是不隐藏。但是并没有提供相应的方法来设置隐藏状态栏。

这种情况我们可以重写prefersStatusBarHidden方法,通过修改它的返回值达到目的。

复制代码 代码如下:

- (BOOL)prefersStatusBarHidden
{
    return YES;
}

这样状态栏就可以隐藏了。


另一种做法是不让控制器来决定是否隐藏状态栏,而是让[UIApplication sharedApplication]来决定。

复制代码 代码如下:

[UIApplication sharedApplication].statusBarHidden = YES;

运行程序发现状态栏并没有隐藏。查看文档可以发现:
复制代码 代码如下:

Setting statusBarHidden does nothing if your application is using the default UIViewController-based status bar system.

意思是如果使用默认的控制器管理状态栏(第一种做法提到的),那么设置statusBarHidden是没有效果的。想要关闭控制器的管理,需要我们给Info.plist增加一个key:
复制代码 代码如下:

View controller-based status bar appearance

并且设置为NO,这样就可以通过设置statusBarHidden来隐藏状态栏了。(这个key是Info.plist添加Key出现的下拉栏里的最后一个)


当控制器管理状态栏时
复制代码 代码如下:

- (UIViewController *)childViewControllerForStatusBarHidden
- (UIViewController *)childViewControllerForStatusBarStyle

这两个方法可以将状态栏的控制权交给子控制器管理,如果返回nil则表示控制器自己来管理状态栏;如果返回子控制器,则表示子控制器来管理状态栏。比如在navigation controller中实现prefersStatusBarHidden方法可能并不管用,因为它可能默认在childViewControllerForStatusBarHidden中返回的是导航栈顶控制器,此时状态栏由栈顶控制器来决定。如果想让navigation controller来决定,那么可以在类中按下面的方式实现:
复制代码 代码如下:

- (UIViewController *)childViewControllerForStatusBarHidden
{
    return nil;
}

此时,状态栏的管理权在导航控制器。


代码注释

作者:喵哥笔记

IDC笔记

学的不仅是技术,更是梦想!