實際開發中都是多控制器的;用一個控制器(父)管理多個控制器(子) ios提供2個特殊的(父)控制器 UINavigationControler 簡介 導航控制器,可以輕鬆完成多個控制器之間的切換,其結構包含導航條(y=20)、棧頂控制器的view、導航控制器的view。導航控制器需要設置一個根控制器 ...
實際開發中都是多控制器的;用一個控制器(父)管理多個控制器(子)
ios提供2個特殊的(父)控制器
UINavigationControler
- 簡介
導航控制器,可以輕鬆完成多個控制器之間的切換,其結構包含導航條(y=20)、棧頂控制器的view、導航控制器的view。導航控制器需要設置一個根控制器,一般是UIViewControler。 - 基本使用
最開始的時候,棧頂控制器的view就是導航控制器的根控制器的view。
1.先去掉Info.plist的倒數第三行的main,以採用代碼方式創建
QQ.png2.在AppDelegate.m中:
其中,先創建視窗,再創建導航控制器(同時創建了一個UIviewControler的控制器,作為其根控制器),然後把該視窗的根控制器設為所創建的導航控制器,最後顯示視窗。
PS:一般地,導航控制器的根控制器來自新建一個UIViewControler類,然後import進來。另外,只要一個控制器是導航控制器的子控制器,那麼導航控制器就會成為這個控制器的一個屬性(可以拿到 )
- 控制器之間的跳轉:
根控制器的導航控制器調用pushViewControler方法,參數為跳轉後的控制器。跳轉後,顯示的是後來這個ViewControler的view,也成了棧頂控制器的view,同時從window上移除了前一個控制器的view,但仍保存於導航控制器的子控制器數組中。
PS:導航控制器是其根控制器的一個屬性。
- 導航控制器的子控制器
一般保存在數組(以棧的形式)(先進後出)中
PS:使用setViewControllers方法可以一次壓入多個控制器,但是只會顯示最後壓入的那個控制器的view(棧頂控制器)
[nav setViewControllers:@[vc1,vc2,vc3] animated:YES];
獲取棧頂控制器:navController.topViewController
獲取當前可見視圖的控制器:.visibleViewController
-
控制器的返回
是跳轉操作的逆向。執行返回操作後,先從window中移除當前控制器的view,同時把該控制器(在棧頂)從棧中移除。此時在棧頂的控制器的view會添加到window,顯示出來,從而實現返回。
-
QQ.png
註意:
要回到指定的控制器,這個指定的控制器必須得是導航控制器的子控制器。
當一個控制器被pop後,控制器記憶體就被釋放了(會調用deinit/dealloc函數。
控制器被銷毀,它的view不一定就銷毀(只要有storng指針),但是當一個控制器被銷毀,那麼它的view的業務邏輯無法處理。
設置導航條的內容
由棧頂控制器的navigationItem屬性來決定導航條的內容,在棧頂控制器的.m的viewDidLoad{ }中設置
QQ.png其中,titleView可以設為各種控制項
QQ.png QQ20170623-094427.png設置navBar的字體大小,顏色
[nav.navigationBar setTitleTextAttributes:@{ NSFontAttributeName:[UIFont systemFontOfSize:19], NSForegroundColorAttributeName:[UIColor whiteColor] }]; navBar的隱藏狀態:navController.navigationBarHidden = YES;
UINavigationBar通過UINavigationItem堆棧,按照如下方式來決定展示在UINavigationBar中的內容
- 位於中間的標題會根據下方順序選擇展示的內容:
如果topItem設置了標題視圖(titleView屬性), 則展示標題視圖
如果topItem設置了標題文字(title屬性), 則展示標題文字
如果以上都未設置, 則展示空白。 - 位於右側的按鈕會根據下方順序選擇展示的內容:
如topItem設置了右側按鈕(rightBarButtonItem屬性)則展示右側按鈕
如果以上都未設置, 則展示空白。 - 位於左側的按鈕會根據下方順序選擇展示的內容:
如topItem設置了左側按鈕(leftBarButtonItem屬性), 則展示左側按鈕
如backItem設置了返回按鈕(backBarButtonItem), 則展示返回按鈕
如backItem設置了標題(title), 則展示用標題文字封裝的返回按鈕 - 如果以上都未設置, 則展示利用文字"Back"封裝的返回按鈕。
設置導航條的外觀
UINavigationBar類中提供了大量屬性/方法用於設置其外觀, 我們可以設置其樣式、背景顏色、色彩顏色、文字屬性,同樣, 我們也可以設置其背景圖片、陰影圖片。
- 樣式:navigationBar.barStyle
該屬性可為UIBarStyleDefault(白底黑字)、UIBarStyleBlack(黑底白字) - 背景色:navigationBar.barTintColor
- 前景色:navigationBar.tintColor
- 設置背景圖片:[navigationBar setBackgroundImage:。。。];
- 陰影圖片:navigationBar.shadowImage = [UIImage imageNamed:@""];
區分關係:
NavigaitonBar是導航欄,位於屏幕上方,管理整個導航控制器的navigationItem,它類似navigationcontroller一樣,提供了一個棧來管理UINavigationItem。在編程時,一般只設置每個控制器的navigationItem屬性。
一個導航控制器管理多個視圖控制器(多個視圖控制器共用一個導航控制器),而一個導航控制器只有一個UINavigationBar,被管理的多個視圖控制器共用這一個UINavigationBar,只要一個視圖控制器改變了UINavigationBar的屬性則影響是全局的。
每個視圖控制器都會有屬於自己的UINavigationItem,系統會以懶載入的方式創建一個UINavigationItem顯示在UINavigationBar中,改變UINavigationItem只會在當前控制器起作用,不會影響其它控制器。
通過storybord跳轉控制器
設根控制器為導航控制器(刪掉預設的控制器,拖入nav,設之為預設即添加箭頭,添加導航控制器的根控制器(viewcontroler),方法是採用拖線的方式,點擊rootcontroler;再拖入一個viewcontroler,實現控制器之間的跳轉,方法是拖線(從一個按鈕拖到下一個控制器,點擊show選項)
設置導航條的內容(可以把其他控制項如item等拖到導航條;可以設導航條標題;
自定義導航控制器
- 設置導航條統一的樣式
+(void)initialize{ // 獲取APP的導航條標識 // UINavigationBar *bar = [UINavigationBar appearance]; //這種寫法有bug,它會代表app中所有的導航條,當app中還有自定義導航條時也會被包含 UINavigationBar *bar = [UINavigationBar appearanceWhenContainedIn:[PDNavigationViewController class],nil]; //(獲取指定某幾個類的導航條標識) //設置nav bar的背景圖片(因無法設置背景顏色) [bar setBackgroundImage:[UIImage imageNamed:@"navBar"] forBarMetrics:UIBarMetricsDefault]; //標題字體大小、顏色 NSMutableDictionary *dict = [NSMutableDictionary dictionary]; dict[NSFontAttributeName] = [UIFont systemFontOfSize:22.0]; dict[NSForegroundColorAttributeName] = [UIColor whiteColor]; [bar setTitleTextAttributes:dict]; }
統一樣式只需設一次即可。因此,在initialize方法中設置,此方法噹噹前類第一次初始化的時候才調用,這樣就實現只設置一次。當然在載入時viewDidLoad方法也只調用一次,但是一個缺點就是,當要創建多個導航控制器時,會調用(設置)多次。
- 設置統一的返回按鈕
重寫push方法來實現
-(void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated{ [super pushViewController:viewController animated:animated]; //判斷當前不是在根控制器下,此時設置leftBarButtonItem if(self.viewControllers.count > 1) viewController.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc] initWithImage:[UIImage imageNamed:@"nearby"] style:0 target:self action:@selector(back)]; } -(void)back{ [self popViewControllerAnimated:YES]; //pop返回上層 }
(此時就系統沒有了滑動返回功能)
- 恢復滑動返回功能
原理:通過清空代理的方式可以恢復
self.interactivePopGestureRecognizer.delegate = nil;
但是存在bug,因為清空之後會帶來後續影響,改善如下:
@interface PDNavigationViewController ()<UINavigationControllerDelegate> @property(nonatomic,strong) id popGesture; //用於暫存 @end -(void)viewDidLoad { [super viewDidLoad]; self.popGesture = self.interactivePopGestureRecognizer.delegate; //暫存代理 self.delegate = self; //將代理設為自身 }
在導航控制器代理方法中清空代理,此方法當控制器顯示完畢時調用,因此能夠拿到當前的控制器
-(void)navigationController:(UINavigationController *)navigationController didShowViewController:(UIViewController *)viewController animated:(BOOL)animated{ //根控制器中不需滑動返回,此時還原代理 if(self.viewControllers[0] == viewController){ //當前是根控制器時 self.interactivePopGestureRecognizer.delegate = self.popGesture; //還原代理 }else{ //非根控制器時,清空代理( 恢復滑動返回功能 ) self.interactivePopGestureRecognizer.delegate = nil; } }
- 實現全屏滑動返回( 系統預設從左側滑動才能返回 )
-(void)viewDidLoad { [super viewDidLoad]; //以下代碼實現全屏滑動返回 id target = self.interactivePopGestureRecognizer.delegate; self.interactivePopGestureRecognizer.enabled = NO; UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:@selector(handleNavigationTransition:)]; [self.view addGestureRecognizer:pan]; pan.delegate = self; }
解決全屏滑動的bug
//當開始滑動就會調用 -(BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer{ //如果是在非根控制器,才允許滑動 BOOL open = self.viewControllers.count > 1; return open; }
作者:Arthur凌
鏈接:https://www.jianshu.com/p/ef6594eeefd6
來源:簡書