View Controller 转场

UIViewController类内置支持呈现其他视图控制器。你可以从任何视图控制器呈现其他视图控制器,但UIKit可能会将请求重新路由到不同视图控制器。源视图控制器(称为presenting view controller)和要显示的视图控制器(称为presented view controller)之间构成了视图控制器层次结构的一部分。

有以下几种方式呈现视图控制器:

  • 使用segue自动显示视图控制器,segue会使用你在Interface Builder中配置的信息实例化并呈现视图控制器。下面是segue的几种类型:

    • Show:此segue调用showViewController: sender:方法显示新内容。对于大多数视图控制器,show segue 在源视图控制器上以modal方式呈现新内容。但UISplitViewControllerUINavigationController类会重写showViewController: sender:方法,以根据自身设计处理呈现方式。如在UINavigationController中,视图控制器会被push到其导航堆栈。

    • Show Detail:此segue调用showDetailViewController: sender:方法显示新内容。Show Detail segue仅与嵌入在UISplitViewController对象内的视图控制器相关,此时分割视图用新内容替换detail controller。在其他视图控制器中,show detail会以modal形式呈现新内容。

    • Present Modally:此segue使用指定presentation styel和transition style以modal形式呈现新内容。

    • Present as Popover:在horizontally regular environment,视图控制器显示在弹出窗口中;在horizontally compact environment,视图控制器使用全屏模式显示。

      Push、Modal、Popover、Replace这四种segue已经不推荐使用。

  • 使用showViewController:sender:showDetailViewController:sender:方法呈现视图控制器。该方法为视图控制器提供了自适应(adaptive)、灵活的呈现方式。这些方法让presenting view controller决定如何呈现视图控制器。例如:容器视图控制器会以子视图控制器的形式呈现新的控制器,而非默认的modal形式。

  • 使用presentViewController:animated:completion:方法总是以modal形式呈现视图控制器。调用该方法的视图控制器可能不是最终处理呈现过程的视图控制器。例如:必须由全屏控制器呈现全屏控制器,如果当前presenting view controller不是全屏,UIKit将遍历视图层次结构,直到找到合适视图控制器。呈现完成后,UIKit更新presentingViewControllerpresentedViewController属性。

1. Showing View Controllers

使用showViewController:sender:showDetailViewController:sender:方法呈现视图控制器步骤如下:

  1. 创建并设置要呈现的视图控制器。
  2. 设置刚创建的视图控制器的modalPresentationStyle属性,指定显示样式。最终显示时,可能不采用该属性。
  3. 设置刚创建的视图控制器的modalTransitionStyle属性,指定界面切换样式。最终显示时,可能不采用该属性。
  4. 调用showViewController:sender:showDetailViewController:sender:方法。

创建Single View App模板的demo,demo名称为DisplayViewController,创建两个继承自UIViewController的控制器,名称分别为FirstVCSecondVC。完成后project navigator如下:

ProjectNavigator.png

进入FirstVC.m文件,添加如下代码:

- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 初始化showVCButton。
    UIButton *showVCButton = [UIButton buttonWithType:UIButtonTypeSystem];
    showVCButton.frame = CGRectMake(0, 0, self.view.bounds.size.width, 30);
    showVCButton.center = self.view.center;
    [showVCButton setTitle:@"showViewController:sender:" forState:UIControlStateNormal];
    [showVCButton addTarget:self action:@selector(showVCButtonClicked:) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:showVCButton];
}

- (void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:animated];
    NSLog(@"FirstVC did appear");
}

- (void)viewDidDisappear:(BOOL)animated {
    [super viewDidDisappear:animated];
    NSLog(@"FirstVC did disappear");
}

- (void)showVCButtonClicked:(UIButton *)sender {
    // 初始化presented view controller。
    SecondVC *secVC = [[SecondVC alloc] init];
    secVC.color = [UIColor lightGrayColor];
    
    // show presented view controller.
    [self showViewController:secVC sender:sender];
}

进入SecondVC.h文件,声明color属性,并在SecondVC.m文件viewDidLoad方法中设置视图背景为color

@property (strong, nonatomic) UIColor *color;

运行demo,点击showViewController:sender:按钮,SecondVC自下而上弹出,覆盖整个视图控制器。

现在,无法从presented view controller返回到presenting view controller,更新SecondVC.m代码如下:

- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 设置背景颜色。
    self.view.backgroundColor = self.color;
    
    // 添加退出当前视图控制器按钮。
    UIButton *dismissVCButton = [UIButton buttonWithType:UIButtonTypeSystem];
    dismissVCButton.frame = CGRectMake(0, 0, self.view.bounds.size.width, 30);
    dismissVCButton.center = self.view.center;
    [dismissVCButton setTitle:@"dismissViewControllerAnimated:completion:" forState:UIControlStateNormal];
    [dismissVCButton addTarget:self action:@selector(dismissVCButtonClicked:) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:dismissVCButton];
}

- (void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:animated];
    
    NSLog(@"SecondVC did appear");
}

- (void)viewDidDisappear:(BOOL)animated {
    [super viewDidDisappear:animated];
    
    NSLog(@"SecondVC did disappear");
}

- (void)dismissVCButtonClicked:(UIButton *)sender {
    // 退出当前控制器。
    [self dismissViewControllerAnimated:YES completion:^{
        NSLog(@"Dismiss SecondVC Completion");
    }];
}

运行demo,如下所示:

showViewController:sender: & dismissViewControllerAnimated:completion:

可以看到:呈现SecondVC时,视图自下而上弹出;退出SecondVC时,视图自上而下退出。

另外,通过控制台可以看到dismissViewController:completion:方法的完成处理程序会在SecondVC调用viewDidDisappear:方法后执行。

showViewController:sender:方法还可以在UINavigationController中使用。更新FirstVC.m文件中的showVCButtonClicked:方法如下:

- (void)showVCButtonClicked:(UIButton *)sender {
    ...
    // show presented view controller.
//    [self showViewController:secVC sender:sender];
    [self.navigationController showViewController:secVC sender:sender];
}

因为当前视图控制器是UIViewController实例,self.navigationControllernil,所以不执行跳转操作。

打开Main.storyboard,将FirstVC嵌入到导航视图控制器。运行demo:

showViewController:sender:

可以看到,现在点击showViewController:sender:按钮时,SecondVC自右向左压入导航堆栈视图。点击SecondVCdismissViewControllerAnimated:completion:按钮,没有执行任何操作。这是因为dismissViewControllerAnimated:completion:方法只能退出以modal呈现的视图控制器。在导航堆栈视图,一般通过点击左上角返回按钮,或触发UIScreenEdgePanGestureRecognizer左侧手势返回。也可以调用popViewControllerAnimated:方法弹出导航视图控制器顶部视图,更新SecondVC.m后如下:

- (void)viewDidLoad {
    ...
    // 添加弹出当前堆栈视图控制器顶部控制器按钮。
    UIButton *popVCButton = [UIButton buttonWithType:UIButtonTypeSystem];
    popVCButton.frame = CGRectMake(0, 0, self.view.bounds.size.width, 30);
    popVCButton.center = CGPointMake(self.view.center.x, self.view.center.y + 50);
    [popVCButton setTitle:@"popViewControllerAnimated:" forState:UIControlStateNormal];
    [popVCButton addTarget:self action:@selector(popVCButtonClicked:) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:popVCButton];
}

- (void)popVCButtonClicked:(UIButton *)sender {
    // 弹出顶部视图控制器。
    [self.navigationController popViewControllerAnimated:YES];
}

运行demo,如下所示:

popViewControllerAnimated:

showViewController:sender:showDetailViewController:sender:方法是呈现视图控制器的首选方式。视图控制器可以在不知道其层次结构时调用该方法,如果处于堆栈视图中,该方法会将新视图压入堆栈中;如果处于UIViewController实例中,该方法会以modal形式呈现视图控制器。

所以,上面呈现SecondVC控制器的代码可以使用[self showViewController:secVC sender:sender]方法,这样可以更灵活的在app不同部分重用视图控制器,而不需添加各种判断。

弹出视图控制器方法如下:

  • popViewControllerAnimated::该方法弹出并返回导航视图控制器堆栈最顶层视图控制器。如果最顶层视图控制器是根视图控制器,则该方法不执行任何操作,即不能弹出导航控制器堆栈最后一个控制器。
  • popToRootViewControllerAnimated::只保留根视图控制器(即最后一个视图控制器),移除堆栈中其他视图控制器。该方法会返回一个包含所有被弹出视图控制器的数组。
  • popToViewController:animated::移除视图控制器,直到显示指定视图控制器。该方法会返回一个包含所有被弹出视图控制器的数组。需要注意的是,指定的视图控制器必须已经存在于堆栈中,否则程序会崩溃。

2. pushViewController:animated:

除使用iOS 8中增加的showViewController: sender:方法将视图控制器压入堆栈,还可以使用iOS 2中的pushViewController:animated:方法。

pushViewController:animated:方法只能用于导航堆栈控制器中,不能用在UIViewController实例。showViewController: sender:showDetailViewController: sender:为自适应方法,可以根据当前视图控制器层级结构处理。

show和push用在导航堆栈视图控制器时没有区别。在storyboar中,push segue已不推荐使用。所以,如果你不需要支持iOS 7,应当使用自适应性强、Apple推荐的showViewController: sender:showDetailViewController: sender:方法。

3. presentViewController:animated:completion:

在水平常规时(in a horizontally regular environment),presentViewController:animated:completion:方法使用modalPresentationStyle属性指定的值呈现视图控制器;在水平紧凑时(in a horizontally compact environment),默认使用全屏模式呈现视图控制器。

  1. 创建并设置要呈现的视图控制器。
  2. 设置刚创建视图控制器的modalPresentationStyle属性,指定显示样式。最终显示时,可能不采用该属性。如果modalPresentationStyleUIModalPresentationPopover,还需要配置以下属性:
    • 将presented view controller的preferredContentSize属性设置为所需大小。
    • 使用UIPopoverPresentationController对象设置anchor,可以通过获取presented view controller的popoverPresentationController属性获得该对象,并设置下列选项之一:
      • 设置UIPopoverPresentationController对象的barButtonItem属性为指定UIBarButtonItem
      • 设置UIPopoverPresentationController对象的sourceView属性、sourceRect属性为视图上指定区域。
  3. 设置刚创建视图控制器的modalTransitionStyle属性,指定界面切换样式。最终显示时,可能不采用该属性。
  4. 调用presentViewController:animated:completion:方法。

更新FirstVC.m文件中的showVCButtonClicked:方法:

- (void)showVCButtonClicked:(UIButton *)sender {
    // 初始化presented view controller。
    SecondVC *secVC = [[SecondVC alloc] init];
    secVC.color = [UIColor lightGrayColor];
    
    // 设置modalPresentationStyle、大小。
    secVC.modalPresentationStyle = UIModalPresentationPopover;
    secVC.preferredContentSize = CGSizeMake(self.view.bounds.size.height/3, self.view.bounds.size.width/2);
    
    UIPopoverPresentationController *popoverVC = secVC.popoverPresentationController;
    if (popoverVC) {
        // 设置sourceView为点击的按钮,sourceRect为按钮边框,arrow方向为上。
        popoverVC.sourceView = sender;
        popoverVC.sourceRect = sender.bounds;
        popoverVC.permittedArrowDirections = UIPopoverArrowDirectionUp;
    }
    
    [self presentViewController:secVC animated:YES completion:^{
        NSLog(@"Present sevVC completion");
    }];
}

如果在iPhone中运行demo,其会以modal形式自下而上弹出视图控制器;如果在iPad中运行,则以弹框形式呈现:

UIModalPresentationPopover

modalPresentationStyle属性值还可被设置为UIModalPresentationFullScreenUIModalPresentationPageSheet等。

4. modalTransitionStyle

呈现视图控制器时过度样式有以下四种:

  • UIModalTransitionStyleCoverVertical:当视图控制器呈现时,其视图从屏幕底部向上滑动;退出时,其视图从顶部向底部滑动。这也是默认的transition style。
  • UIModalTransitionStyleFlipHorizontal:呈现视图控制器时,当前视图控制器使用从右向左水平3D翻转,产生一种presented view controller在presenting view controller背面的效果。在退出时,从左向右翻转返回到presenting view controller。
  • UIModalTransitionStyleCrossDissolve:当呈现视图控制器时,presenting view controller溶解淡出,与此同时presented view controller淡入。退出时使用类似的交叉溶解。
  • UIModalTransitionStylePartialCurl:呈现视图控制器时,presenting view controller视图一角卷起,显露出presented view controller视图。退出时动画与卷起时相反。该效果只能在modalPresentationStyel属性为UIModalPresentationStyleFullScreen时才可以使用,否则会触发异常。

下面GIF图显示了四种切换方式区别:

modalTransitionStyle

上面GIF使用了storyboard segue进行跳转,如果你不熟悉如何使用,可以下载demo查看。

Demo名称:DisplayViewController
源码地址:https://github.com/pro648/BasicDemos-iOS

参考资料:

  1. Presenting a View Controller
  2. showViewController v.s pushViewController

欢迎更多指正:https://github.com/pro648/tips/wiki

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容