在日常開(kāi)發(fā)中缅茉,UINavigationController是我們常用的一種視圖控制器,也許平時(shí)我們也有碰到一些問(wèn)題并且查找了解決方法男摧,但是并不清楚它的具體實(shí)現(xiàn)的方式和原理蔬墩,相信了解后,會(huì)對(duì)之后的開(kāi)發(fā)很有幫助耗拓。
接來(lái)我會(huì)按點(diǎn)分析:
1.UINavigationControllrt結(jié)構(gòu)
創(chuàng)建一個(gè)UINavigationController并設(shè)置一個(gè)rootViewController拇颅,查看它的層級(jí):
可以看到它整個(gè),整個(gè)內(nèi)容在UILayoutContainerView上乔询,在這之上有三部分組成轉(zhuǎn)場(chǎng)視圖UINacigationTransitionView樟插、導(dǎo)航視圖UINavigationBar和工具欄視圖UIToolBar(在最底部默認(rèn)是不在,通過(guò)toolbarHidden屬性添加上去)竿刁。
而在轉(zhuǎn)場(chǎng)視圖中的UIViewControllerWrapperView則是真正控制顯示的ViewController黄锤。
2.導(dǎo)航區(qū)navigationBar對(duì)于內(nèi)容區(qū)域view的影響
@property(nonatomic,assign,getter=isTranslucent) BOOL translucent NS_AVAILABLE_IOS(3_0) UI_APPEARANCE_SELECTOR;
// Default is NO on iOS 6 and earlier. Always YES if barStyle is set to UIBarStyleBlackTranslucent
在navigationBar上的這個(gè)屬性是控制導(dǎo)航區(qū)是否半透明。
在iOS7之前食拜,蘋(píng)果是擬物化風(fēng)格鸵熟,導(dǎo)航區(qū)默認(rèn)是不透明的,內(nèi)容區(qū)的view都是從導(dǎo)航區(qū)下方開(kāi)始的(64)监婶,而iOS7之后旅赢,轉(zhuǎn)為扁平化,默認(rèn)為半透明惑惶。
當(dāng)我們把該屬性為NO煮盼,也就是不透明時(shí),它的表現(xiàn)和之前一樣带污,所有放在內(nèi)容區(qū)ViewController上的View都是從導(dǎo)航區(qū)下方開(kāi)始為起點(diǎn)(即64僵控,在iPhoneX等全面屏機(jī)型上為84,為方便之后文中都用64代表狀態(tài)欄+導(dǎo)航欄的高度)鱼冀。
而當(dāng)為默認(rèn)半透明時(shí)报破,放在ViewController上的View起點(diǎn)都是0。
這里有一點(diǎn)需要注意千绪,當(dāng)view是放在UIScrollView或其子類上時(shí)充易,這個(gè)view會(huì)和導(dǎo)航區(qū)沒(méi)有設(shè)置透明的表現(xiàn)一樣,在導(dǎo)航區(qū)下方開(kāi)始荸型。經(jīng)過(guò)研究盹靴,打印scrollView的bounds會(huì)發(fā)現(xiàn)它的y值為-64。在UIViewController上有一個(gè)屬性
@property(nonatomic,assign) BOOL automaticallyAdjustsScrollViewInsets API_DEPRECATED("Use UIScrollView's contentInsetAdjustmentBehavior instead", ios(7.0,11.0),tvos(7.0,11.0));
// Defaults to YES
字面意思理解就是自動(dòng)調(diào)節(jié)scrollViewInset瑞妇,默認(rèn)YES稿静,目的是不讓導(dǎo)航欄遮住在scrollView上的內(nèi)容,在iOS11被棄用辕狰,用UIScrollView上的contentInsetAdjustmentBehavior屬性代理改备。
3.關(guān)于popViewControllerAnimated方法
在navigationController上有一個(gè)viewControllers的數(shù)組,里面裝著在NationController中所有push進(jìn)來(lái)的ViewController蔓倍,我們依次把5個(gè)vc給push進(jìn)來(lái)悬钳,查看navigationController. viewControllers會(huì)發(fā)現(xiàn)這5個(gè)vc依次在數(shù)組中,我們把這5個(gè)vc依次叫1-5vc偶翅,這時(shí)候我們移除數(shù)組中最上面兩個(gè)vc他去,并調(diào)用popViewControllerAnimated方法:
NSMutableArray *array = [NSMutableArray arrayWithArray:self.navigationController.viewControllers];
[array removeLastObject];
[array removeLastObject];
self.navigationController.viewControllers = array;
[self.navigationController popViewControllerAnimated:YES];
我們會(huì)發(fā)現(xiàn)這時(shí)候會(huì)返回到3vc,由此倒堕,我們可以知道popViewControllerAnimated的邏輯是移除當(dāng)前顯示控制器灾测,若沒(méi)有則不移除viewControllers數(shù)組中的vc跳到棧頂控制器。
4.關(guān)于navigationBar
(1)navigationBar只有一個(gè)垦巴,通過(guò)當(dāng)前控制器的navigationItem顯示內(nèi)容媳搪,如果有需要而直接在navigationBar上添加view,必須在不需要的時(shí)候進(jìn)行移除(如退出當(dāng)前vc或進(jìn)入下一vc時(shí))骤宣,否則會(huì)一直存在秦爆。
(2)關(guān)于設(shè)置導(dǎo)航區(qū)的透明,首先需要把半透明關(guān)閉
self.navigationController.navigationBar.translucent = NO;
然后有兩種方式憔披,一種是設(shè)置圖片方法:
- (void)transluentStyle{
[self.navigationController.navigationBar setBackgroundImage:self.image forBarMetrics:UIBarMetricsDefault];
}
- (UIImage*)image{
UIGraphicsBeginImageContext(CGSizeMake(100, 100));
[[[UIColor whiteColor] colorWithAlphaComponent:0] setFill];
UIRectFill(CGRectMake(0, 0, 100, 100));
UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();
return image;
}
一種是找到背景顯示的view并把它設(shè)置為透明顏色:
- (void)transluentTwoStyle{
NSArray *ary = [self.navigationController.navigationBar subviews];
UIColor *alphaColor = [[UIColor whiteColor] colorWithAlphaComponent:0];
for (int i = 0; i < ary.count; i++) {
UIView *view = ary[i];
view.backgroundColor = alphaColor;
for (int j = 0; j < view.subviews.count; j++) {
UIView *subView = view.subviews[j];
subView.backgroundColor = alphaColor;
for (int k = 0; k < subView.subviews.count; k++) {
UIView *subsubView = subView.subviews[k];
subsubView.backgroundColor = alphaColor;
}
}
if([view isKindOfClass:NSClassFromString(@"_UIBarBackground")]){
view.backgroundColor = alphaColor;
}
}
}
(3)導(dǎo)航區(qū)下方的黑線去除等限,因?yàn)樵趯蛹?jí)圖上找到黑線的view爸吮,發(fā)現(xiàn)它的y值是64,說(shuō)明高于導(dǎo)航區(qū)而又在導(dǎo)航區(qū)上望门,這時(shí)候只要設(shè)置下導(dǎo)航區(qū)超過(guò)不顯示即可:
self.navigationController.navigationBar.clipsToBounds = YES;
(4)當(dāng)需要在navigationBar上添加多個(gè)按鈕并需要有一點(diǎn)間隔時(shí)可以這樣做:
NSMutableArray *barItems = [NSMutableArray array];
UIBarButtonItem *barItem = [[UIBarButtonItem alloc] initWithTitle:@"Nav" style:UIBarButtonItemStylePlain target:self action:@selector(showNav)];
UIBarButtonItem *barItemSpace = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemFixedSpace target:nil action:nil];
barItemSpace.width = 60;
UIBarButtonItem *barItemT = [[UIBarButtonItem alloc] initWithTitle:@"view" style:UIBarButtonItemStylePlain target:self action:@selector(showNavTwo)];
[barItems addObject:barItem];
[barItems addObject:barItemSpace];
[barItems addObject:barItemT];
self.navigationItem.rightBarButtonItems = barItems;
(5)關(guān)于navigationbarHidden方法形娇,它是通過(guò)remove的方法將navigationBar隱藏的,所以如果要顯示筹误,不能用navigationbar.hidden = NO;
的方式桐早。
如果覺(jué)得有用,請(qǐng)隨手給一個(gè)贊厨剪,謝謝:)