iOS視圖控制器使用學習筆記

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 *)#>];

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章