
当然,demo中为了不影响其他演示,使用了自定义导航控制器TransparentNavigationController并且在其中设置隐藏导航栏背景,实际开发中如果有必要你可以通过appearance统一隐藏所有的导航栏背景。
方案2效果相对来说已经比较完美了,不足的地方是如果要实现毛玻璃效果就需要自己处理。微信在这点上做了更细节的处理:在转场过程中使用自定义导航栏背景,转场切换完成以后替换成系统的导航栏效果,实现了对方案2的细节改进。如果你想实现半透明效果而又不想自己实现,可以考虑这种方式,否则方案2应该可以说是比较完美的方案。
切换动画
其实多数App都采用的前面提到的解决方案,但是类似于手QQ进入QQ空间的切换就没有这么处理(考虑可能是出于QQ空间界面的导航栏上拉还需要导航栏渐显而有意这么处理的),手QQ要实现的效果是从动态到空间导航栏从蓝色逐渐消失,而从空间回来则相反。实现这个过程其实最主要的是考虑手势返回时透明度的设置和返回进度保持一致,好在苹果已经提供了现成的api供我们调用,这就是self.transitionCoordinator.animateAlongsideTransition(in view: UIView?, animation: ((UIViewControllerTransitionCoordinatorContext) -> Swift.Void)?, completion: ((UIViewControllerTransitionCoordinatorContext) -> Swift.Void)? = nil),默认转场时会通过这个动画回调计算出block的中间插值,无论手势停留在哪一时刻都能确保动画插值保持一致,从而达到类似于QQ空间的手势切换效果。demo4演示了这一实现:

当然,如果你愿意,可以通过demo4配合demo3实现更好的切换效果。
此外,也有人通过swizzle UINavigationController的_updateInteractiveTransition:方法在手势切换过程中动态修改导航栏的透明度等来达到完美的过渡,但相比较于上面的公开方法并没有太明显的优势。而且这只适用于从无到有的过渡,对于不同颜色的导航栏过渡也显得无能为力。类似的还有使用iOS7新加入的UIViewControllerInteractiveTransitioning自定义转场动画,这跟swizzle _updateInteractiveTransition:并没有明显的区别。
扩展:全屏返回手势
和导航栏相关的另一个问题就是侧滑手势返回。前面可以看到方案2相对方案1其中一个优点就是不用想办法维护返回手势失效的问题,但是不管哪种方案都不能全屏手势返回。尽管这个需求如果苹果从系统级设计实现将是易如反掌,但显然苹果觉得从边缘侧滑才是合理的,不过全屏侧滑基本已经是目前国内app的标配了。










