我们在本文讨论navigation controller中的两个view controller之间的转场动画,但是这些做法在 tab bar controller或者任何你自己定义的view controller容器中也是通用的…
尽管从技术角度来讲,使用ios7的api,你可以对自定义容器中的view controller做自定义转场,但是这不是能直接使用的,实现这种效果非常不容易。
请注意我正在讨论的自定义视图控制器容器都是UIViewController的直接子类,而不是UITabBarController或者UINavigationController的子类。
对于你自定义的继承与UIViewController的容器子类,并没有现成可用的Api允许一个任意的动画控制器将一个子视图控制器自动转场到另外一个,不管是可交互的转场还是不可交互式的转场。我甚至都觉得苹果根据不想支持这种方式。苹果支持下面的几种转场方式:
- Navigation Controller推入和推出页面
- Tab bar Controller选择的改变
- Model页面的展示和消失
在本文中,我将向你展示如何自定义视图控制器容器,并且使其支持第三方的动画控制器
预热准备
看到这里,你可能对上文我们说到一些问题犯嘀咕,让我来告诉你答案吧:
为什么我们不直接继承UINavigationController或UITabBarController.并且使用它们提供的功能呢?
有些时候这是你不想要的。可能你想要一个非常特殊的外观或者行为,和这些类能够提供给你的差别非常大,因此你必须使用一些黑客式的手段去达到你想要的结果,同时还要担心系统框架的版本更新后这些黑客式的手段是否还仍然有效。或者,你就是想完全控制你的视图控制器容器,避免不得不支持一些特定的功能。
好吧,那么为什么不实用
transitionFromViewController:toViewController:duration:options:animations:completion :
去实现呢?
这又是一个好问题,你可能想用这种方式去实现,但是或许你对代码的整洁性比较在意,想把这种转场相关的代码封装在内部。那么为什么不实用一个即存的,被良好验证的设计模式呢?这种设计模式可以非常方便的支持第三方的转场动画。
介绍相关的API
在我们开始写代码之前,让我们先花一分钟时间来简单看一下我们需要的组件吧。
ios7自定义视图控制器转场的API基本上都是以协议的方式提供的,这也使其可以非常灵活的使用,因为你可以很简单地讲它们插入到你的类中。最主要的五个组件如下:
- 动画控制器(Animation Controllers)遵从UIViewControllerAnimatedTransitioning协议,并且负责实际执行动画。
- 交互控制器(Interaction Controllers)通过遵从UIViewControllerInteractiveTransitioning协议来控制可交互式的转场
- 转场代理(Transitioning Delegates)根据不同的转场类型方便的提供需要的动画控制器和交互控制器
- 转场上下文(Transitioning Contexts)定义了转场时需要的元数据,比如在转场过程中所参与的视图控制和视图相关属性。转场上下文对象遵从UIViewControllerContextTransitoning协议,并且这是由系统负责生成和提供的。
- 转场协调器(Transition Coordinators)可以在运行转场动画时,并行的运行其他动画。转场协调器遵从UIViewControllerTransitionCoorinator协议。
正如你从其他的阅读材料中得知的那样,转场有不可交互式和可交互式两种方式。在本文红,我们将集中精力于不可交互的转场。这种转场是最简单的转场,也是我们学习的一个好的开始。这意味着我们需要处理上面提到的动画控制器(animation controllers),转场代理(transioning delegates)和转场上下文(transionging contexts)
示例工程
通过三个阶段,我们将要实现一个简单自定义的视图控制器容器,它可以对子视图控制器提供自定义的转场动画的支持。
你可以在这里找到这三个阶段的Xcode工程的源代码
阶段1:基础
我们应用中的核心类是ContainerViewController
,它持有一个UIViewController实例的数组,每个实例是一个普通的ChildViewController。容器视图控制器设置了一个带有可点击图标,并代表每个子视图控制器的私有的子视图:
我们通过点击图标在不同的子视图控制器之间切换,在这一阶段,子视图控制器之间切换时是没有转场动画的。
你可以在这里查看阶段1的源代码
阶段2:转场动画
当我们添加转场动画时,我们想要使用一个遵从 UIViewControllerAnimatedTransitioning协议的动画控制器(animation controllers)。这个协议声明了三个方法,前面的2歌方法是必须实现的:
1 2 3 |
|
通过这些方法,我们可以获得我们所需的所有东西。当我们的视图控制器容器准备执行动画时,我们可以从动画冬至器中获取动画的持续时间,并让其去执行真正的动画。当动画执行完毕后,如果动画控制器实现了可选的 animationEnded:
方法,我们可以调用动画控制器中的 animationEnded: 方法。
但是,首先我们必须把一件事情搞清楚。正如你在上面的方法签名中看到的那样,上面两个必须实现的方法需要一个转场上下文参数,这是一个遵从 UIViewControllerContextTransionging
协议的对象。通常情况下,当我们使用系统内建的类时,系统框架为我们创建了转场上下文对象,并把它传递给动画控制器。但是在我们这种情况下,我们需要自定义转场动画,所以我们需要承担系统框架的责任,自己去创建这个转场上下文对象。
这就是大量使用协议的方便之处。我们可以不用必须复写一个私有类,而复写私有类这种方法是明显不可行的。我们可以定义自己的类,并使其遵从文档中相应的协议就可以了。
尽管在UIViewControllerContextTransioning协议中声明了很多方法,而且他们都是必须要实现的,但是我们现在可以暂时忽略它们中的一些方法,因为我们现在仅仅支持不可交互的转场。
同UIKit类似,我们定义了私有类NSObject <UIViewControllerContextTransitioning>
.在我们的特定例子汇总,这个私有类是 PrivateTransitionContext,它的初始化方法如下实现:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
|
我们把视图的出现和消失时的状态记录了下来,比如初始状态和最终状态的frame.
请注意一点,我们的初始化方法需要我们提供我们是在向右切换还是向左切换。在我们的ContainerViewController
中,按钮是一个接一个水平排列的,转场上下文通过设置每个frame来记录它们之间的位置关系。动画控制器或者说 animator,在生成动画时可以使用这些frame.
我们也可以通过另外的方式去获取这些信息,但是那样的话,就会使animator和ContainerViewController
及其视图控制器耦合在一起了,这是不好的,我们并不想这样。animator应该只关心它自己以及传递给它的上下文,因为这样,在理想的情况下,animator可以在不同的上下文中得到复用。
在下一步实现我们自己的动画控制器时,我们应该时刻记住这一点,现在让我们来实现转场上下文吧。
使用Animator类的实例来做转场动画的核心代码如下所示:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
|
这其中的大部分是对视图控制器容器的操作,计算出我们是在向左切换还是向右切换,做动画的部分基本上只有3行代码:
- 创建Animator
- 创建转场上下文
- 触发动画执行
有了上面的代码,转场效果看起来如下图所示:
非常酷,我们甚至没有写一行动画相关的代码
你可以在阶段2标签下看到这部分代码的变化。
阶段3:封装
我想我们最后要做的一件事情是封装 ContainerViewController,使其能够:
- 提供默认的转场动画
- 提供替换默认动画控制器的代理
这意味着我们需要对Animator类的依赖移除,同时需要创建一个代理协议。
我们如下定义这个协议:
1 2 3 4 5 |
|
containerViewController:didSelectViewController:
方法使ContainerViewController
可以很容易的集成与功能齐全的应用中。
containerViewController:animationControllerForTransitionFromViewController:toViewController:
方法挺有趣的,当然,你可以把它和下面的UIKit中的视图控制器容器的代理协议做对比:
- tabBarController:animationControllerForTransitionFromViewController:toViewController: (UITabBarControllerDelegate)
- navigationController:animationControllerForOperation:fromViewController:toViewController: (UINavigationControllerDelegate)
所有的这些方法都返回一个id
1 2 3 4 5 |
|
如果我们有代理并且它返回一个Animator,那么我们就使用这个 animator.否则,我们使用内部私有类PrivateAnimatedTransition 创建一个默认的 animator.接下来我们将实现 PrivateAnimatedTransition
类。
尽管默认的动画和 Animator有一些不同,但是代码看起来惊人的相似,下面是完整的代码实现:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 |
|
需要注意一点的是,上面的代码没有通过设置视图的frame 来反应它们之间的位置关系,但是代码仍然可以正常工作,只不过转场总是在同一个方向上。因此,这个类也可以被其它的代码库使用。
转场动画看起来像是这样:
在阶段3代码中,app delegate中设置代理的部分被注释掉了,这样就可以看到默认的动画效果了,你可以将其设置回再使用 Animator类,你可能想查看同阶段2相比所有的修改
我们现在有一个自包含的提供了默认转场动画的 ContainerViewController 类,这个默认的转场动画可以被开发者定义的ios7自定义动画控制器(UIViewControllerAnimatedTransitioning)的对象代替,甚至都可以不用关心我们的源代码就可以方便的替换。
结论
在本文中,我们通过使用ios7提供的自定义视图控制器转场的新特性,使得我们自定义的视图控制器容器成为了UIkit的一等公民。
这意味着你可以把自定义的非交互的转场动画应用到自定义的视图控制器容器中。你可以看到我们把7个话题之前使用的转场类直接拿过来使用,而且没有做任何修改。
如果你想让自己的容器视图控制器作为一个类库或者框架,或者仅仅想使你的代码得到更好的复用,这将是非常完美的。
我们现在仅仅支持非交互式的转场,下一步就是对交互式的转场也提供支持。