項(xiàng)目設(shè)置
iOS新建工程后可以在General
標(biāo)簽頁對(duì)APP進(jìn)行基本的配置矫限,在Deployment Info
下的Status Bar Style
中可以進(jìn)行狀態(tài)欄的樣式設(shè)置。
這里有三個(gè)樣式可以設(shè)置:
- 第一個(gè)是整體樣式,默認(rèn)是Default扎唾,是黑色的樣式戚丸,還有一種是Light,是白色的樣式劲妙。
- 第二個(gè)是
Hide status bar
,勾上之后啟動(dòng)的時(shí)候自動(dòng)隱藏狀態(tài)欄儒喊,后面可以在UIApplication或者UIViewController中設(shè)置顯示狀態(tài)欄镣奋。 - 第三個(gè)是
Require full screen
,網(wǎng)上查資料怀愧,這個(gè)是針對(duì)分屏任務(wù)的侨颈,勾上這個(gè)后就表示這個(gè)App需要全屏運(yùn)行,不支持分屏任務(wù)了芯义。
三個(gè)樣式修改后也可以在Info.plist
中找到相應(yīng)的設(shè)置
程序控制
UIApplication
在UIApplication中可以通過一些方法和屬性設(shè)置狀態(tài)欄樣式哈垢。
@interface UIApplication(UIApplicationDeprecated)
@property(nonatomic,getter=isProximitySensingEnabled) BOOL proximitySensingEnabled NS_DEPRECATED_IOS(2_0, 3_0) __TVOS_PROHIBITED; // default is NO. see UIDevice for replacement
- (void)setStatusBarHidden:(BOOL)hidden animated:(BOOL)animated NS_DEPRECATED_IOS(2_0, 3_2) __TVOS_PROHIBITED; // use -setStatusBarHidden:withAnimation:
// Explicit setting of the status bar orientation is more limited in iOS 6.0 and later.
@property(readwrite, nonatomic) UIInterfaceOrientation statusBarOrientation NS_DEPRECATED_IOS(2_0, 9_0, "Explicit setting of the status bar orientation is more limited in iOS 6.0 and later") __TVOS_PROHIBITED;
- (void)setStatusBarOrientation:(UIInterfaceOrientation)interfaceOrientation animated:(BOOL)animated NS_DEPRECATED_IOS(2_0, 9_0, "Explicit setting of the status bar orientation is more limited in iOS 6.0 and later") __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;
// 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 withAnimation:(UIStatusBarAnimation)animation NS_DEPRECATED_IOS(3_2, 9_0, "Use -[UIViewController prefersStatusBarHidden]") __TVOS_PROHIBITED;
- (BOOL)setKeepAliveTimeout:(NSTimeInterval)timeout handler:(void(^ __nullable)(void))keepAliveHandler NS_DEPRECATED_IOS(4_0, 9_0, "Please use PushKit for VoIP applications instead of calling this method") __TVOS_PROHIBITED;
- (void)clearKeepAliveTimeout NS_DEPRECATED_IOS(4_0, 9_0, "Please use PushKit for VoIP applications instead of calling this method") __TVOS_PROHIBITED;
@end
可以設(shè)置status bar hidden和status bar style,并且支持動(dòng)畫扛拨。不過從這個(gè)Category的名字就可以看到耘分,這些方法已經(jīng)被標(biāo)識(shí)Deprecated,不建議使用了绑警。
View controller-based status bar appearance
可能你通過上面的UIApplication進(jìn)行設(shè)置并沒有什么效果求泰,這是因?yàn)檫€有一個(gè)Info.plist中的屬性需要配置,View controller-based status bar appearance
计盒,這個(gè)屬性需要一個(gè)Boolean的值渴频,默認(rèn)是YES,表示狀態(tài)欄的樣式由每個(gè)ViewController控制章郁,所以你針對(duì)UIApplication進(jìn)行的設(shè)置就沒有作用了枉氮。也因?yàn)閁IApplication針對(duì)狀態(tài)欄的樣式設(shè)置被標(biāo)識(shí)成Deprecated志衍,所以這個(gè)屬性默認(rèn)是YES,開發(fā)者可以在每個(gè)頁面自行控制狀態(tài)欄的樣式聊替。
UIViewController
一個(gè)新的工程里默認(rèn)會(huì)創(chuàng)建一個(gè)ViewController楼肪,可以覆蓋一些方法控制狀態(tài)欄樣式。
- (BOOL)prefersStatusBarHidden {
return NO;
}
- (UIStatusBarAnimation)preferredStatusBarUpdateAnimation {
return UIStatusBarAnimationSlide;
}
- (UIStatusBarStyle)preferredStatusBarStyle {
return UIStatusBarStyleLightContent;
}
這三個(gè)方法的命名也很好理解惹悄,可以修改返回值觀察變化春叫。
setNeedsStatusBarAppearanceUpdate
我有幾個(gè)按鈕,需要?jiǎng)討B(tài)修改狀態(tài)欄樣式泣港,比如有一個(gè)按鈕是控制style暂殖,一個(gè)是控制hidden,在按鈕的事件處理中修改一個(gè)成員變量当纱,在上面的返回函數(shù)中使用這個(gè)成員變量呛每。還需要調(diào)用setNeedsStatusBarAppearanceUpdate更新狀態(tài)欄樣式。
- (IBAction)toggleStatusHidden:(id)sender {
_statusBarHidden = !_statusBarHidden;
// 直接調(diào)用setNeedsStatusBarAppearanceUpdate沒有動(dòng)畫效果坡氯,所以用UIView的動(dòng)畫包了一下
[UIView animateWithDuration:1 animations:^{
[self setNeedsStatusBarAppearanceUpdate];
}];
}
- (IBAction)toggleStatusBarStyle:(id)sender {
// UIStatusBarStyle 枚舉對(duì)應(yīng)的兩個(gè)style正好是 0 和 1晨横,這里可以直接取反操作
_style = !_style;
// preferredStatusBarUpdateAnimation 只會(huì)影響hidden屬性,style的animation固定是漸變動(dòng)畫
[UIView animateWithDuration:1 animations:^{
[self setNeedsStatusBarAppearanceUpdate];
}];
}
這里為了支持動(dòng)畫效果箫柳,使用了UIView的animation動(dòng)畫手形,可以把duration設(shè)置的長一點(diǎn),觀察Slide和Fade兩種動(dòng)畫的區(qū)別悯恍。但是只影響prefersStatusBarHidden
库糠,這點(diǎn)在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
使用UIViewController還沒找到style切換動(dòng)畫的方法涮毫,如果有同學(xué)知道怎么實(shí)現(xiàn)瞬欧,還望在評(píng)論中告訴一下。
UINavigationController
上面的例子全部都在一個(gè)新建的空工程中實(shí)踐的窒百,只有一個(gè)ViewController黍判,然而我們大多數(shù)情況都需要UINavigationController豫尽,我們?cè)赟toryboard中選中ViewController篙梢,選擇菜單中的Editor->Embed In->Navigation Controller,加入一個(gè)導(dǎo)航控制器美旧,再運(yùn)行渤滞,我們發(fā)現(xiàn)style的修改不起作用了,hidden和animation還是有效果的榴嗅。猜測(cè)是因?yàn)閁INavigationController的preferesStatusBarStyle方法沒有轉(zhuǎn)發(fā)給當(dāng)前ViewController妄呕。于是自定義一個(gè)MyNavigationController,在Storyboard中設(shè)置Class嗽测,重寫MyNavigationController的preferesStatusBarStyle
方法:
- (UIStatusBarStyle)preferredStatusBarStyle {
return self.topViewController.preferredStatusBarStyle;
}
再運(yùn)行绪励,切換Style也可以了肿孵。
總結(jié)
- 從iOS9開始,UIApplication對(duì)statusbarstyle的修改方法已經(jīng)廢棄疏魏,建議使用UIViewController中相關(guān)是設(shè)置方法停做,同時(shí)View controller-based status bar appearance默認(rèn)為YES。
- preferredStatusBarUpdateAnimation默認(rèn)值是Fade大莫,只對(duì)顯示/隱藏狀態(tài)欄起作用蛉腌。
- 當(dāng)嵌入在一個(gè)UINavigationController中時(shí)需要重寫preferredStatusBarStyle。
- Genaral中三個(gè)設(shè)置項(xiàng)的意義只厘。