深入理解iOS的状态栏

时间:2022-11-13 07:33:30

一、状态栏的隐藏

状态栏的隐藏主要有两种方法,下面来一起看看吧。

方法一:通过代码控制

?
1
2
3
4
5
6
7
8
9
10
11
@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这样就可以使用上边的方法了

深入理解iOS的状态栏

注意:添加的view controller-based status bar appearancebool类型,默认为yes,很不幸ios9之后苹果已经不推荐使用这些方法了,这些方法能用但是会报警告。

那么这些方法被禁用以后,如何操作呢?注释里已经提示use -[uiviewcontroller prefersstatusbarhidden]这是ios7之后苹果在uiviewcontroller里添加的新方法,这么做的目的可以让开发者更加灵活的自定义每个viewcontroller的状态栏。

?
1
2
3
- (bool)prefersstatusbarhidden{
  return yes;
}

ios7之后uiviewcontroller中不只提供了这个关于状态栏的设置的函数,还有其他的,后面详细说。

方法二:通过info.plist控制

1,首先我们依然要设置这个(第2步中的两种方式都要设置这个参数)

深入理解iOS的状态栏

2,然后设置(两种方式)

深入理解iOS的状态栏

或者

深入理解iOS的状态栏

两者是等效的!并且两者的状态是同步的。

二、状态栏样式

先看看都有哪些样式(解释看注释)

?
1
2
3
4
5
6
7
8
9
10
11
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;

如何设置状态栏样式

?
1
2
3
4
5
// 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以后这些方法被禁用了,苹果推荐在具体的viewcontrolleruse -[uiviewcontroller preferredstatusbarstyle]

?
1
2
3
- (uistatusbarstyle)preferredstatusbarstyle{
  return uistatusbarstylelightcontent;
}

注意:我们通常使用的viewcontroller都是嵌套在uinavigationcontroller中使用的,此时在viewcontroller中使用- (uistatusbarstyle)preferredstatusbarstyle;函数会发现设置并没有生效。

系统也给我们提供了一个函数- (uiviewcontroller *)childviewcontrollerforstatusbarstyle,也可以解决这个问题,后面会讲。

三、背景色

ios7以后默认情况下状态栏的背景为透明的,一种办法是我们自己写一个uiview作为背景添加到状态栏下面,这样就可以随意设置状态栏的颜色了。

另一种方法就是通过设置navigationbarsetbartintcolor颜色来改变状态栏颜色

uiviewcontroller中其他有关状态栏的函数

preferredstatusbarupdateanimation函数

?
1
2
- (uistatusbaranimation)preferredstatusbarupdateanimation
ns_available_ios(7_0) __tvos_prohibited; // defaults to uistatusbaranimationfade

如果想在当前已经显示的uiviewcontroller中更改状态栏的样式的话,需要调用以上函数。调用该函数后,系统会主动调用preferredstatusbarstyle方法重绘状态栏的样式

childviewcontrollerforstatusbarstyle函数

?
1
2
3
// 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,此时系统就会调用当前viewcontrollerapreferredstatusbarstyle函数;如果返回值是另一个viewcontrollerb那么系统就会调用viewcontrollerbpreferredstatusbarstyle函数。

运用这个函数就可以解决嵌套uinavigationcontroller设置样式无效的问题。

解释一下为什么嵌套uinavigationcontrollerviewcontrollerpreferredstatusbarstyle函数设置无效:

在我们嵌套了uinavigationcontroller的时候,我们的appdelegate.window.rootviewcontroller通常是我们创建的navigationcontroller,这时首先会调用的是navigationcontroller中的childviewcontrollerforstatusbarstyle函数,因为默认返回nil,那么接下来就会调用navigationcontroller本身的preferredstatusbarstyle函数,所以我们在viewcontroller中通过preferredstatusbarstyle函数设置的状态栏样式就不会被调用发现,所以也就无效了。

所以我们要自己创建一个继承于uinavigationcontrollernavigationcontroller,在这个子类中重写childviewcontrollerforstatusbarstyle函数

?
1
2
3
- (uiviewcontroller *)childviewcontrollerforstatusbarstyle{
  return self.topviewcontroller;
}

这样navigationcontroller中的childviewcontrollerforstatusbarstyle函数会返回navigationcontroller中最上层的viewcontroller,那么viewcontroller中的preferredstatusbarstyle函数的设置就会被系统获知

childviewcontrollerforstatusbarhidden函数

?
1
2
3
- (nullable uiviewcontroller *)childviewcontrollerforstatusbarhidden
 
ns_available_ios(7_0) __tvos_prohibited;

childviewcontrollerforstatusbarhidden函数的使用原理同上,不再赘述。

preferredstatusbarupdateanimation函数

?
1
2
3
// 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

动画形式如下

?
1
2
3
4
5
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开发者们有所帮助,如果有疑问大家可以留言交流。