UIViewControl是所有視圖控制類的父類,所有子類都繼承於它。
視圖控制器是爲iPhone應用程序提供了基礎的視圖控制模型,用戶可以通過視圖控制器來管理視圖的繼承關係。所以使用視圖控制器可以方便地管理視圖中的子視圖的子視圖,如不使用視圖控制器來操作視圖的話,那麼所有的視圖的視圖必須有繼承管理。(Android中的Activity的作用也是用於添加布局控件及控制控件之間的調用操作的關係,把view和control集成在Activity中。如這樣就會是Activity裏面的代碼極重,所以需要用MVC/MVP/MVVC的設計模式把view/control/model分離開來)。
先創建一個視圖控制器實例,並把實例設置爲根視圖控制器。當視圖控制器實例創建好之後,其自身會有一個view,接下來就可以在view上創建自己想要創建的各種視圖,並通過視圖控制器來管理這些新創建的視圖。
1.UIViewController視圖控制器:
1.1 視圖控制器的創建:
視圖控制器的創建有兩種方法:一種是代碼創建,另一種是用XIB來創建。
- 代碼創建視圖控制器:
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
NSLog(@"AppDelegate添加didFinishLaunchingWithOptions");
self.window = [[[UIWindow alloc]initWithFrame:[[UIScreen mainScreen]bounds]] autorelease];
self.window.backgroundColor = [UIColor whiteColor];
UIViewController *vc = [[UIViewController alloc]init];
self.window.rootViewController = vc; //設置視圖控制器實例爲根控制器
vc.view.backgroundColor = [UIColor purpleColor];
[self.window makeKeyAndVisible];
return YES;
}
- XIB方法創建:
1.2 視圖控制器的生命週期:
視圖控制器用於管理與之關聯的多個視圖,它自身也會協調與其他視圖控制器的通信。在管理與之關聯的視圖時,系統規定只有當應用需要時纔會加載視圖,而不在需要則卸載視圖,以節省系統資源。
- 視圖控制器視圖的加載:
視圖控制器自身有一個view屬性,是在控制器的生命週期裏,所以在對它進行內存管理時,它必須在視圖控制器釋放前release。如沒有定義view,則系統會首先調用[self loadView]方法,返回系統的view。也可自己重寫視圖中的loadView方法。
子類重寫loadView方法,則會先使用用戶自定義方法創建視圖。在重寫loadView方法中,要創建一個view視圖賦值給視圖控制器的view屬性。
如子類調用父類的loadView方法[spuer loadView],可能有三種創建視圖的情形:1.通過StroyBoard;2.通過XIB文件;3.直接創建一個空視圖。
視圖控制器是判斷用戶在子類中重寫loadView方法,而不是判斷調用loadView方法後視圖控制器中的view是否爲空。
在調用loadDidView方法前,視圖控制器的view爲空,這是因爲視圖控制器view使用時,會調用view的getter方法。它會判斷view是否創建,如沒創建則會調用loadView方法來創建view。當loadView方法執行完後接着是執行loadDidView,這是view已經創建完成。
loadView方法是用來創建自定義視圖的,沒有必要通過視圖控制器的實例來調用loadView方法。
- 視圖控制器視圖的顯示與消失:
viewWillAppear:當視圖控制器對象的視圖即將加入窗口;
viewDidAppear:當視圖控制器對象的視圖已經加入窗口;
viewWillDisappear:當視圖控制器對象的視圖即將消失;
viewDidDisappear:當視圖控制器對象的視圖已經消失。
- 視圖控制器視圖的卸載:
當系統發出警告或者調用didReveiveMemoryWarning時,系統會判斷當前是否有視圖及視圖是否能被卸載。如能則卸載,調用viewWillUnload方法後釋放調當前view,再調用viewDidUnload方法。
1.3 模態視圖:
當彈出模態視圖時,系統會中段程序正常執行流程,如UIAlertView。
視圖控制器通過presentModalViewController方法彈出模態視圖,主要用於以下幾種情形:
- 需要收集用戶信息;
- 臨時呈現內容或改變工作狀態;
- 改變設備的方向;
- 顯示一個新的view層級。
模態視圖常用的屬性設置和方法主要有以下四種:
- presentViewController:(UIViewController*)animated:(BOOL)completion:(void)completion:該方法的作用就是彈出一個模態視圖。
- modalPresentationStyle :該屬性是用來設置彈出的模態視圖的風格,是一個枚舉類;
- modalTransitionStyle:是用來設置彈出和消失模態視圖時視圖之間切換的動畫效果的;
- dismissViewControllerAnimated:(BOOL)completion:(void)completion:是使模態視圖消失的方法。
代碼示例:
- 創建一個自己定義的MyModalViewController添加到ViewController中:
-(void)presentModal
{
MyModalViewController *myModalviewController = [[MyModalViewController alloc]init];
myModalviewController.modalPresentationStyle = UIModalTransitionStyleCoverVertical;
myModalviewController.modalTransitionStyle = UIModalTransitionStylePartialCurl;
[self presentViewController:myModalviewController animated:YES
completion:^{
NSLog(@"顯示模態視圖");
}];
- 在自己定義的ModalViewController中添加消失時的事件處理:
-(void)ModalDismiss
{
[self.delegate ChangeView:@"傳給附着view的值"];
[self dismissViewControllerAnimated:YES completion:^{
NSLog(@"模態視圖消失!");
}];
}
模態視圖的設計方法:
在模態視圖中,需要將模態視圖設置爲代理,從而委託ViewController去實現相應的協議的方法。
- 在MyModalViewController添加協議,用於值的傳遞,並用propert屬性創建協議的存取方法:
@protocol ModalViewControllerDelegate <NSObject>
@optional
-(void)ChangeView:(NSString *)text;
@end
@interface MyModalViewController : UIViewController
{
}
@property (nonatomic, assign) id<ModalViewControllerDelegate> delegate;
@end
- 在ViewController中添加協議:
@interface ViewController : UIViewController<ModalViewControllerDelegate>
{
}
@end
- ViewController在協議的實現方法ChageView中獲取到從MyModalViewController視圖傳來的值:
- (void)ChangeView:(NSString *)text
{
NSLog(@"ViewController ChangeView=%d",text);
}
- 設置模態控制視圖的實例爲代理:
myModalviewController.delegate = self;
2.UINavigationController導航控制器:
它的功能用於構建多層次的應用程序,管理多個視圖切換。
導航控制器時視圖控制器的一個子類,而導航控制器下面還有UIImagePickerController和UIVideoEditorController兩個子類。
- NavigationBar:主要來負責視圖之間的切換,位於整個導航控制器的最上方;
- Custom content:用來顯示內容的視圖,自定義視圖的內容將會顯示在這裏;
- Navigation toolbar:是導航控制器的輔助工具欄視圖。
導航控制是以棧的形式來實現的。
2.1 導航控制器的創建:
UINavigationController *navigation = [[UINavigationController alloc]initWithRootViewController:vc];
[vc setTitle:@"首頁"];
[vc release];
self.window.rootViewController = navigation;
self.window.rootViewController = navigation;
[navigation release];
2.2 toolbar:
toolbar默認是隱藏的所以在viewDidAppear設置顯示,toolbar應該由當前視圖進行控制:
-(void)viewDidAppear:(BOOL)animated
{
[super viewDidAppear:animated];
[self.navigationController setToolbarHidden:NO animated:YES];
UIBarButtonItem *btn = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemAdd target:self action:nil];
UIBarButtonItem *btn1 = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemReply target:self action:nil];
NSArray *item = @[btn,btn1];
[self setToolbarItems:item animated:YES];
[btn release];
[btn1 release];
}
2.3 導航欄:
一個NavigationBar由:LeftBarButtonItem/RightBarButtonItem/BackBarButtonItem/Title/TitleView,TitleView可以是用戶自己定義視圖。
2.4 自定義的原則:
- LeftBarButtonItem:如當前ViewController設置了LeftBarButtonItem,那麼就顯示用戶設置的LeftBarButtonItem;如當前ViewController沒有設置LeftBarButtonItem,而且當前ViewController不是根視圖控制器時,則顯示前一層的視圖控制器的BackBarButton,前一層的視圖控制器沒有顯示地指定BackBarButton系統將會根據前一視圖控制器的title屬性自動生成一個“back”按鈕顯示;如當前視圖是根視圖,且沒有設置相應的LeftBarButtonItem,那麼不顯示任何內容。
- title:如定義一個視圖,將顯示當前視圖的TitleView設置成自定義視圖,那麼title上就會顯示用戶自定義的視圖;如沒有設置TitleView系統就會根據當前視圖的navigationControlle.title的值創建一個UILabel顯示其內容;
- RightBarButtonItem:如當前的視圖控制器設置了RightBarButtonItem,就顯示設置內容;如沒有設置則不顯示任何內容。
2.5 導航控制器實現視圖之間的切換:
主要以下視圖切換的方法:
- (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated; // Uses a horizontal slide transition. Has no effect if the view controller is already in the stack.
- (nullable UIViewController *)popViewControllerAnimated:(BOOL)animated; // Returns the popped controller.
- (nullable NSArray<__kindof UIViewController *> *)popToViewController:(UIViewController *)viewController animated:(BOOL)animated; // Pops view controllers until the one specified is on top. Returns the popped controllers.
- (nullable NSArray<__kindof UIViewController *> *)popToRootViewControllerAnimated:(BOOL)animated; // Pops until there's only a single view controller left on the stack. Returns the popped controllers.
2.6 UIImagePickerController:
UIImagePickerController是模態視圖,用來選擇相片。其中可以使用UINavigationControllerDetegate和UIImagePickerControllerDetegate代理方法做具體的事件操作。
//設定sourceType爲相機,然後判斷是否可用。(iPod)沒相機,不可用則sourceType設置爲相片庫
UIImagePickerControllerSourceType *sourceType = UIImagePickerControllerSourceTypeCamera;
if (![UIImagePickerController isSourceTypeAvailable:UIImagePickerControllerSourceTypeCamera]) {
sourceType = UIImagePickerControllerSourceTypePhotoLibrary;
}
UIImagePickerController *pick = [[UIImagePickerController alloc]init];
pick.delegate = self;
pick.allowsEditing = YES;
pick.modalPresentationStyle = UIModalPresentationPageSheet;
pick.modalTransitionStyle = UIModalTransitionStylePartialCurl;
[self presentViewController:pick animated:YES completion:^{
}];
[pick release];
3.UITableBarController分欄控制器(類似於Android中的ViewPager+Fragment的效果):
UITableBarController也是用來控制和管理UIViewController的類,UINavigationController的管理是通過棧的方式(類似於Android activity啓動模式和任務棧)會有一層層的層級關係,出棧後當前視圖會被卸載。而UITableBarController是以數組的形式將分欄信息添加到手機屏幕上,而視圖之間的關係是平級的,並沒有層級之分,切換之後視圖不會被移除(Android中的Fragment之間也是類似的概念)。
4. 視圖間數據傳遞方式:
- 導航控制器屬性傳值方法:通過導航控制的push方法來實現;
- 協議傳值方式;
- 通知傳值方法:NSNotificationCenter(類Android的中EventBus消息傳值)就像一個發射站,這樣可以進行兩者之間值的傳遞;
//註冊通知發出事件
[[NSNotificationCenter defaultCenter]addObserver:<#(nonnull id)#> selector:<#(nonnull SEL)#> name:<#(nullable NSNotificationName)#> object:<#(nullable id)#>];
//z註冊通知收到事件
[[NSNotificationCenter defaultCenter]postNotificationName:<#(nonnull NSNotificationName)#> object:<#(nullable id)#>];
- NSUserDefaults傳值方法(類Android中的SharePreferences):
NSUserDefaults是NSObject類提供的一個偏好存儲自定義類,它是一個用於輕量級存儲數據的類。
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
[defaults setValue:<#(nullable id)#> forKey:<#(nonnull NSString *)#>];
[defaults valueForKey:<#(nonnull NSString *)#>];