几年前就从网上抄袭过addChildViewController的用法,但没花过心思去理解它。而且,很多抄袭来的代码,无法理解为什么要那么写!(因为有些本来就写得烂,也误导了自己..)
今天又重新搞了一遍,(以前很少用,所以耦合度有点高..),参考其官方注释和其他人的代码。
先看几个属性、方法:
- 子控制器数组,不包含任何当前展示的子控制器
open var childViewControllers: [UIViewController] { get }
- 如果该子控制器有另一个父控制器,会先通过执行removeFromParentViewController方法把该子控制器
从其当前父控制器移除。如果重载该方法,必须调用super方法。
open func addChildViewController(_ childController: UIViewController)
- 把自身从父控制器的childViewControllers数组中移除。如果重载,必须调用super方法。
open func removeFromParentViewController()
- 该方法可用于切换兄弟控制器。方法调用者是它们共同的父控制器。(用 [UIViewController addChildViewController:] 建立父子关系)
该方法将把【‘toViewController’的view】添加到【‘fromViewController’的view的父视图view】上;
当切换完成后,【‘fromViewController’的view】将从其父视图上移除。,...,
该方法调用者不能是iOS容器类控制器及其子类(tabbar,navigation controller等)...。
open func transition(from fromViewController: UIViewController, to toViewController: UIViewController,
duration: TimeInterval, options: UIViewAnimationOptions = [],
animations: (() -> Swift.Void)?,
completion: ((Bool) -> Swift.Void)? = nil)
根据这几个方法说明,我认为,childViewControllers是配合addChildViewController一起用的,当然
内部肯定还有remove相关的操作,暂时不用关心。addChildViewController没有任何特别的操作,就是往数组里添加元素,以此来确认双方父子关系。父控制器则通过childViewControllers辨认哪些是自己的儿子,儿子们相互间就是兄弟。
关键方法transition(from,to,duration,options,animations,completion)则是用于兄弟控制器的切换。有些人在每次调用切换方法之前,还调用addChildViewController,完了还调用removeFromParentViewController。纯粹是脱裤子放屁!完全没必要,因为transition方法并不会移除子控制器。
所以正确的写法应该是:
- 在父控制器中:
1.添加子控制器
func addChildVCs() {
//直接导致firstVC执行willMove(toParentViewController parent: UIViewController?)
self.addChildViewController(firstVC)
//直接导致secondVC执行willMove(toParentViewController parent: UIViewController?)
self.addChildViewController(secondVC)
firstVC.view.frame = self.view.bounds
self.view.addSubview(firstVC.view)
firstVC.didMove(toParentViewController: self) //手动调用
currentVC = localPackageVC
}
2.子控制器间的切换(不需要再写addChildViewController!也不要写oldVC.removeFromParentViewController())
func changeControllerFrom(oldVC: UIViewController, to newVC: UIViewController) {
//oldVC.willMove(toParentViewController: nil) //手动调用
//newVC.willMove(toParentViewController: self) //手动调用
newVC.view.frame = self.view.bounds
self.transition(from: oldVC, to: newVC, duration: 0.5, options: UIViewAnimationOptions.transitionCrossDissolve, animations: {}, completion: { (isfinished) in
if isfinished {
oldVC.didMove(toParentViewController: nil) //手动调用
newVC.didMove(toParentViewController: self) //手动调用
self.currentVC = newVC
}
})
- 子控制器中:
监听切换状态:
//移除,则parent == nil
override func willMove(toParentViewController parent: UIViewController?) {
debugPrint("will move..parent=" + ((parent == nil) ? "nil" : "parent"))
}
//移除,则parent == nil
override func didMove(toParentViewController parent: UIViewController?) {
debugPrint("did move..parent=" + ((parent == nil) ? "nil" : "parent"))
}