首先,支持并感谢@wazrx 的 http://www.jianshu.com/p/45434f73019e和@onevcathttps://onevcat.com/2013/10/vc-transition-in-ios7/ 对于转场动画的讲解和实现,非常详细,本人也是看过他们的文章后受的启发,快速实现了基于本项目需求的转场动画效果。

效果如下:(gif是我们的美术大师帮忙做的演示动效,实际的效果要比这个好,可通过文章最后链接下载我们的app:柠檬跑步,查看效果)

地图切换.gif

我的需求是两个页面push、pop之间的切换,所以自定义了push的转场动画,首先需要个遵循<UIViewControllerAnimatedTransitioning>协议的管理对象,并实现其两个方法:
XWCircleSpreadTransition.h

#import <UIKit/UIKit.h>
typedef NS_ENUM(NSUInteger, XWCircleSpreadTransitionType) {    
   XWCircleSpreadTransitionTypePush = 0,    
   XWCircleSpreadTransitionTypePop
};
@interface XWCircleSpreadTransition : NSObject<UIViewControllerAnimatedTransitioning>
@property (nonatomic, assign) XWCircleSpreadTransitionType type;  
+ (instancetype)transitionWithTransitionType:(XWCircleSpreadTransitionType)type;
- (instancetype)initWithTransitionType:(XWCircleSpreadTransitionType)type;@end

XWCircleSpreadTransition.m

#import "XWCircleSpreadTransition.h"
@implementation XWCircleSpreadTransition
+ (instancetype)transitionWithTransitionType:(XWCircleSpreadTransitionType)type{    
   return [[self alloc] initWithTransitionType:type];
}  
- (instancetype)initWithTransitionType:(XWCircleSpreadTransitionType)type{    
   self = [super init];    
   if (self) {        
        _type = type;    
   }    
   return self;
}  
- (NSTimeInterval)transitionDuration:(id<UIViewControllerContextTransitioning>)transitionContext{    
   return 0.7;
}  
- (void)animateTransition:(id<UIViewControllerContextTransitioning>)transitionContext{    
   switch (_type) {        
       case XWCircleSpreadTransitionTypePush:            
           [self presentAnimation:transitionContext];            
           break;        
       case XWCircleSpreadTransitionTypePop:            
           [self dismissAnimation:transitionContext];            
           break;    
    }
}  
- (void)dismissAnimation:(id<UIViewControllerContextTransitioning>)transitionContext{    
   UIViewController *toVC = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];    
   UIViewController *fromVC = [transitionContext viewControllerForKey:UITransitionContextFromViewControllerKey];    
   UIView *containerView = [transitionContext containerView];    
   [containerView insertSubview:toVC.view atIndex:0];    //画两个圆路径    
   CGFloat radius = sqrtf(containerView.frame.size.height * containerView.frame.size.height + containerView.frame.size.width * containerView.frame.size.width) / 2;    UIBezierPath *startCycle = [UIBezierPath bezierPathWithArcCenter:containerView.center radius:radius startAngle:0 endAngle:M_PI * 2 clockwise:YES];//    UIBezierPath *endCycle =  [UIBezierPath bezierPathWithOvalInRect:temp.mapBtnFrame];    
   UIBezierPath *endCycle =  [UIBezierPath bezierPathWithOvalInRect:CGRectMake(0, 0, 0, 0)];    //创建CAShapeLayer进行遮盖    
   CAShapeLayer *maskLayer = [CAShapeLayer layer];    
   maskLayer.fillColor = [UIColor greenColor].CGColor;    
   maskLayer.path = endCycle.CGPath;    
   fromVC.view.layer.mask = maskLayer;    //创建路径动画    
   CABasicAnimation *maskLayerAnimation = [CABasicAnimation animationWithKeyPath:@"path"];    
   maskLayerAnimation.delegate = self;    
   maskLayerAnimation.fromValue = (__bridge id)(startCycle.CGPath);    
   maskLayerAnimation.toValue = (__bridge id)((endCycle.CGPath));    
   maskLayerAnimation.duration = [self transitionDuration:transitionContext];    
   maskLayerAnimation.delegate = self;    
   maskLayerAnimation.timingFunction = [CAMediaTimingFunction  functionWithName:kCAMediaTimingFunctionEaseInEaseOut];    
   [maskLayerAnimation setValue:transitionContext forKey:@"transitionContext"];    
   [maskLayer addAnimation:maskLayerAnimation forKey:@"path"];
}  
- (void)presentAnimation:(id<UIViewControllerContextTransitioning>)transitionContext{    UIViewController *toVC = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];    UIView *containerView = [transitionContext containerView];    
   [containerView addSubview:toVC.view];    //画两个圆路径//    UIBezierPath *startCycle =  [UIBezierPath bezierPathWithOvalInRect:fromVC.mapBtnFrame];    
   UIBezierPath *startCycle =  [UIBezierPath bezierPathWithOvalInRect:CGRectMake(0, 0, 0, 0)];    CGFloat x = MAX(100, containerView.frame.size.width - 100);    CGFloat y = MAX(100, containerView.frame.size.height - 100);    CGFloat radius = sqrtf(pow(x, 2) + pow(y, 2));    UIBezierPath *endCycle = [UIBezierPath bezierPathWithArcCenter:containerView.center radius:radius startAngle:0 endAngle:M_PI * 2 clockwise:YES];    //创建CAShapeLayer进行遮盖    
   CAShapeLayer *maskLayer = [CAShapeLayer layer];    
   maskLayer.path = endCycle.CGPath;    //将maskLayer作为toVC.View的遮盖    
   toVC.view.layer.mask = maskLayer;    //创建路径动画    
   CABasicAnimation *maskLayerAnimation = [CABasicAnimation animationWithKeyPath:@"path"];    
   maskLayerAnimation.delegate = self;    //动画是加到layer上的,所以必须为CGPath,再将CGPath桥接为OC对象    
   maskLayerAnimation.fromValue = (__bridge id)(startCycle.CGPath);    
   maskLayerAnimation.toValue = (__bridge id)((endCycle.CGPath));    
   maskLayerAnimation.duration = [self transitionDuration:transitionContext];    
   maskLayerAnimation.delegate = self;    
   maskLayerAnimation.timingFunction = [CAMediaTimingFunction  functionWithName:kCAMediaTimingFunctionEaseInEaseOut];      
   [maskLayerAnimation setValue:transitionContext forKey:@"transitionContext"];    
   [maskLayer addAnimation:maskLayerAnimation forKey:@"path"];
}  
- (void)animationDidStop:(CAAnimation *)anim finished:(BOOL)flag{    
   switch (_type) {        
       case XWCircleSpreadTransitionTypePush:{            
           id<UIViewControllerContextTransitioning> transitionContext = [anim valueForKey:@"transitionContext"];             [transitionContext completeTransition:YES];            
           UIView *toView = [transitionContext viewForKey:UITransitionContextToViewKey];            
           toView.layer.mask = nil;            
           UIViewController *vc = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];             vc.view.layer.mask = nil;         }            
           break;        
       case XWCircleSpreadTransitionTypePop:{            
           id<UIViewControllerContextTransitioning> transitionContext = [anim valueForKey:@"transitionContext"];            
           [transitionContext completeTransition:YES];            
           UIViewController *vc = [transitionContext viewControllerForKey:UITransitionContextFromViewControllerKey];            
           vc.view.layer.mask = nil;         }            
           break;    
   }
}
@end

剩下的就是在ViewControllerA中实现代理<UINavigationControllerDelegate>中的animationControllerForOperation:

#pragma mark -
#pragma mark - UINavigation Delegate
-(id<UIViewControllerAnimatedTransitioning>)navigationController:(UINavigationController *)navigationController animationControllerForOperation:(UINavigationControllerOperation)operation fromViewController:(UIViewController *)fromVC toViewController:(UIViewController *)toVC {    
   if (fromVC == self) {        
       return [XWCircleSpreadTransition transitionWithTransitionType:XWCircleSpreadTransitionTypePush];    
   }    
   if (toVC == self) {        
       return [XWCircleSpreadTransition transitionWithTransitionType:XWCircleSpreadTransitionTypePop];    
   }    
   return nil;
}
 

到此,这个简单的转场动画已经实现完毕,而且动画效果可以自定义成自己想要的展现形式。

类似nike+、香蕉打卡的转场动画效果-b的更多相关文章

  1. iOS CATransition 自定义转场动画

    https://www.jianshu.com/p/39c051cfe7dd CATransition CATransition 是CAAnimation的子类(如下图所示),用于控制器和控制器之间的 ...

  2. iOS 动画学习之视图控制器转场动画

    一.概述 1.系统会创建一个转场相关的上下文对象,传递到动画执行器的animateTransition:和transitionDuration:方法,同样,也会传递到交互Controller的star ...

  3. 自己定义转场动画--Swift3.0版本号

    转场动画这事.说简单也简单.能够通过presentViewController:animated:completion:和dismissViewControllerAnimated:completio ...

  4. ios基础动画、关键帧动画、动画组、转场动画等

    概览 在iOS中随处都可以看到绚丽的动画效果,实现这些动画的过程并不复杂,今天将带大家一窥iOS动画全貌.在这里你可以看到iOS中如何使用图层精简非交互式绘图,如何通过核心动画创建基础动画.关键帧动画 ...

  5. iOS自定义转场动画实战讲解

    iOS自定义转场动画实战讲解   转场动画这事,说简单也简单,可以通过presentViewController:animated:completion:和dismissViewControllerA ...

  6. 超酷炫的转场动画?CSS 轻松拿下!

    在 WeGame 的 PC 端官网首页,有着非常多制作精良的基于滚动的动画效果. 这里我简单截取其中 2 个比较有意思的转场动画,大家感受感受.转场动画 1: 转场动画 2: 是不是挺有意思的,整个动 ...

  7. iOS开发UI篇—核心动画(转场动画和组动画)

    转自:http://www.cnblogs.com/wendingding/p/3801454.html iOS开发UI篇—核心动画(转场动画和组动画) 一.转场动画简单介绍 CAAnimation的 ...

  8. CATransition-转场动画

    CAAnimation的子类,用于做转场动画,能够为层提供移出屏幕和移入屏幕的动画效果.iOS比Mac OS X的转场动画效果少一点 UINavigationController就是通过CATrans ...

  9. iOS:核心动画之转场动画CATransition

    转场动画——CATransition CATransition是CAAnimation的子类,用于做转场动画,能够为层提供移出屏幕和移入屏幕的动画效果.iOS比Mac OS X的转场动画效果少一点 U ...

随机推荐

  1. 24小时学通Linux内核之有关Linux文件系统实现的问题

    有时间睡懒觉了,却还是五点多醒了,不过一直躺倒九点多才算起来,昨晚一直在弄飞凌的嵌入式开发板,有些问题没解决,自己电脑系统的问题,虽然Win10发布了,,但我还是好喜欢XP呀,好想回家用用家里的XP来 ...

  2. Rust 学习 0

    安装Rust 后,本地有文档: file:///usr/local/share/doc/rust/html/index.html file:///usr/local/share/doc/rust/ht ...

  3. HTTP - 条件请求

    当 HTTP 请求包含 If-XXX 这种样式的首部时,服务器会对附带的条件进行判断,只有判断指定条件为真,才会执行请求.这样的请求首部有五个,分别是 If-Modified-Since.If-Unm ...

  4. HDOJ2027统计元音

    统计元音 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  5. ios获取当前语言

    上代码: + (NSString*)getPreferredLanguage { NSUserDefaults * defaults = [NSUserDefaults standardUserDef ...

  6. visual studio 2015预览版系统需求

    visual studio 2015预览版的系统需求跟visual studio 2013的一样. 支持visual studio 2015 preview的操作系统:Windows 8.1(x86 ...

  7. 如何设置Win7系统中的上帝模式GodMode(转载)

    如何设置Win7系统中的上帝模式GodMode(转载) NT6系统中隐藏了一个秘密的“GodMode”,字面上译为“上帝模式”.God Mode其实就是一个简单的文件夹窗口,但包含了几乎所有系统的设置 ...

  8. 解决TextView在显示文字的时候,一行还没显示满就跳到下一行

    转载请注明:转自 http://blog.csdn.NET/u011176685/article/details/48295185 一.问题描述: Android的TextView在显示文字的时候,如 ...

  9. Java垃圾回收介绍(译)

    在Java中,对象内存空间的分配与回收是由JVM中的垃圾回收进程自动完成的.与C语言不同的是,在Java中开发者不需要专门为垃圾回收写代码.这是使Java流行的众多特征之一,也帮助了程序员写出了更好的 ...

  10. JqGrid 使用方法详解

    JQGrid JQGrid是一个在jquery基础上做的一个表格控件,以ajax的方式和服务器端通信. JQGrid Demo 是一个在线的演示项目.在这里,可以知道jqgrid可以做什么事情. 下面 ...