细说CoreAnimation

封面.jpg

前言:

核心动画是iOS中一个渲染和动画的基础设施,你可以用它来为你的应用做基础的动画效果。在核心动画中,你要做的只是配置一些动画参数,接下来的任务就可以交给核心动画来处理了,很方便是吧。下面是核心动画在系统的层次结构:


1.png

CoreAnimation

CAAnimation 是一个抽象的动画类,他提供基本的支持给CAMediaTimingCAAction。给Layer添加动画,你也可以使用子类CABasicAnimation,CAKeyframeAnimation, CAAnimationGroup, 和 CATransition.

2.png

CAAnimation

CAMediaTimingFunction
是一个控制动画节奏的可选的时间函数。它的初始化以下面的初始化方式构造对象:

+ (instancetype)functionWithName:(NSString *)name;
/** Timing function names. **/

CA_EXTERN NSString * const kCAMediaTimingFunctionLinear
    __OSX_AVAILABLE_STARTING (__MAC_10_5, __IPHONE_2_0);
CA_EXTERN NSString * const kCAMediaTimingFunctionEaseIn
    __OSX_AVAILABLE_STARTING (__MAC_10_5, __IPHONE_2_0);
CA_EXTERN NSString * const kCAMediaTimingFunctionEaseOut
    __OSX_AVAILABLE_STARTING (__MAC_10_5, __IPHONE_2_0);
CA_EXTERN NSString * const kCAMediaTimingFunctionEaseInEaseOut
    __OSX_AVAILABLE_STARTING (__MAC_10_5, __IPHONE_2_0);
CA_EXTERN NSString * const kCAMediaTimingFunctionDefault
    __OSX_AVAILABLE_STARTING (__MAC_10_6, __IPHONE_3_0);

具体的效果可以看下面的图。kCAMediaTimingFunctionDefault是系统默认的时间函数。使用这个函数来确保你的动画时间匹配大多数系统的动画。

3.png

你可以对应上图看下面的效果:


4.gif

你可以点击这里是获取CAMediaTimingFunction更多相关内容。你也可以点击这里自己去操作一下相关内容。

CAAnimationDelegate
每个动画都有一个delegate,会回调给你一个动画开始和动画结束的协议。方便为你的应用做相应的处理。在动画结束后你应该为layer使用removeAllAnimations移除相应的动画,减小不必要的开销。

#pragma  mark - CAAnimationDelegate
- (void)animationDidStart:(CAAnimation *)anim{
    NSLog(@"动画开始");
}

- (void)animationDidStop:(CAAnimation *)anim finished:(BOOL)flag {
   
   /*
    因为我们的动画repeatCount属性设置成了1000。所以会很久才结束。你可以手动修改
    你可以根据"key值来判断是哪个layer上的动画"
    比如:[XXX.arrowLayer animationForKey:@"positionX"]拿到动画
    */
   NSLog(@"动画结束");
}

具体相关的内容可以见demo中的CAAnimationViewController

CAPropertyAnimation

CAPropertyAnimationCAAnimation的子类,创造动画的时候以key-path操作相关视图层的属性来完成动画。
完整的key-path请点击这里
具体使用key-path操作可以见demo中的CAPropertyAnimationViewController

属性值
**additive** 决定是否把动画指定的值添加到当前render tree 来产生新的render tree值。

additive的解释还是太过于生硬,你需要知道layer tree(见最下面的layer tree模块)。在动画的时候设置additive YES后会把当前的presentation layer加入到render tree

CABasicAnimation

CABasicAnimation对象为layer提供基本的,单一的frame有关的动画。CABasicAnimation里面需要注意fromValuebyValuetoValue 属性控制

设置预备的参数:

组合参数(不为nil) 区间值
fromValue + toValue [fromValue,toValue]
fromValue + byValue [fromValue,fromValue + byValue]
byValue + toValue [toValue - byValue,toValue]
fromValue [fromValue ,layer当前设置的value]
toValue [layer当前设置的value ,toValue]
byValue [layer当前设置的value ,layer当前设置的value + byValue]
CAKeyframeAnimation

CAKeyframeAnimation对象有能力为layer对象提供关键帧动画。你可以通过指定一个储存关键帧的数组来控制相应的动画时间和动画行为。

属性值
**values**     一个承载动画多个关键帧值的数组
**keyTimes**  一个承载动画时间的数组。数据的每个值都在0-1之间,第一个值为0最后的一个值为1。数组的值对应关键帧数组的值或者是path路劲的属性
**path**  一段CGPathRef路径。layer沿着路劲动画。如果你同时设置了values 和path,path路劲将会被执行。path拥有更高的优先权。

具体使用方法可以见demo中的CAKeyframeAnimationViewController,雪人是按照path路径,社交图标是按照values 效果:

5.gif

CATransaction

layer每一次的修改都是事务的一部分。事物是通过CATransaction类来管理的。CoreAnimation 支持两种类型的事务:隐式事务和显式事务。CATransaction类不是通过alloc init方法来初始化。而是借助于begincommit方法。

method:
**begin**  为当前线程创建一个新的事务
**commit** 提交当前的事务所有的改变项

demoCATransactionViewController类:

- (void)viewDidLoad{
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    
    self.imageView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"圣诞老人"]];
    self.imageView.frame = CGRectMake(0, 0, 100, 100);
    self.imageView.center = self.view.center;
    [self.view addSubview:self.imageView];
    
    self.redLayer = [CALayer layer];
    self.redLayer.frame = CGRectMake(100, 100, 100, 100);
    self.redLayer.delegate = self;
    self.redLayer.backgroundColor = [UIColor redColor].CGColor;
    [self.view.layer addSublayer:self.redLayer];
    
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
{
//    configure the transaction
    [CATransaction begin];
//    [CATransaction setDisableActions:YES];
    [CATransaction setAnimationDuration:4.0];
    [CATransaction setAnimationTimingFunction:[CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseOut]];
    //set the position
    self.redLayer.position  = [[touches anyObject] locationInView:self.view];
    self.imageView.layer.position = [[touches anyObject] locationInView:self.view];
   
    //commit transaction
    [CATransaction commit];
    
    
    NSLog(@"Outside: %@", [self.imageView actionForLayer:self.imageView.layer forKey:@"position"]);
    //begin animation block
    [UIView beginAnimations:nil context:nil];
    //test layer action when inside of animation block
    NSLog(@"Inside: %@", [self.imageView actionForLayer:self.imageView.layer forKey:@"position"]);
    //end animation block
    [UIView commitAnimations];
    
}
6.gif

红色的视图是layer.圣诞老人是UIImageView。当点击了其他区域,你会发现红色的视图有一个缓慢的动画,而圣诞老人马上就到了指定区域。这是因为UIView的隐式动画被关联图层给禁用了。
对于layer图层你还可以利用下面方法禁用隐式动画。

 [CATransaction setDisableActions:YES];

更多信息你可以点击这里去了解。也建议读读喵大翻译的View-Layer 协作.

CAAnimationGroup

CAAnimationGroupCAPropertyAnimation一样是CAAnimation的子类。CAAnimationGroup可以让多个动画组合同时发生。

Important
The delegate and removedOnCompletion properties of animations in the [animations
](https://developer.apple.com/reference/quartzcore/caanimationgroup/1412516-animations?language=objc) array are currently ignored. The CAAnimationGroup
 delegate does receive these messages.

layer tree

核心动画中有三种layer tree。每一种layer tree在你的应用呈现在屏幕中扮演着不同的角色。

  • model layer tree (就是我们应用通常使用最多的layer对象,他们为动画储存着目标值。比如:position,paque等)
  • presentation tree(这些layer对象呈现着一个动画当前值。比如你的layer沿着x轴历时5秒从0到5model layer position会一直显示最后的值5,而presentation随着时间显示0,1,2..5。注意你不应该人为去修改这些值。)
  • render tree(这种layer对象在实际的动画中,但它对核心动画来说是私有的,无法操作的)
7.png

我们来验证下,在例子中的LayerTreeViewController类中加入下面的代码:

- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    self.redView.layer.position = CGPointMake(100, 100);
    [self printMethod];
    
    CABasicAnimation *basicAnimation  = [[CABasicAnimation alloc] init];
    basicAnimation.keyPath = @"position.y";
    basicAnimation.fromValue = @0;
    basicAnimation.toValue = @500;
    basicAnimation.duration = 10;
    [self.redView.layer addAnimation:basicAnimation forKey:@"positionY"];
    
 
    NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0f
                                                      target:self
                                                    selector:@selector(printMethod)
                                                    userInfo:nil
                                                     repeats:YES];
    [timer fire];
    
    
}

- (void)printMethod{
    CALayer *modelLayer = self.redView.layer;
    CALayer *presentationLayer = self.redView.layer.presentationLayer;
    NSLog(@"model layer:%@     presentation layer :%@", NSStringFromCGPoint(modelLayer.position),NSStringFromCGPoint(presentationLayer.position));
}

#pragma mark - get
- (UIView *)redView{
    if (!_redView) {
        _redView = [[UIView alloc] init];
        _redView.frame = CGRectMake(0, 0, 100, 100);
        _redView.backgroundColor = [UIColor redColor];
        _redView.center = self.view.center;
        [self.view addSubview:_redView];
        [self printMethod];
    }
    return _redView;
}

看看打印结果:

3.png

可以看出presentation layer初始值为(0,0)。当我们手动的去改变postion值,presentation layer没有发现变化而model layer发生了改变。在使用动画的时候presentation layer才发生变化,代表在动画中layer实时的值。

更多了解您可以点击Demo

参考资料:

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

推荐阅读更多精彩内容

  • 在iOS中随处都可以看到绚丽的动画效果,实现这些动画的过程并不复杂,今天将带大家一窥ios动画全貌。在这里你可以看...
    每天刷两次牙阅读 8,460评论 6 30
  • 在iOS中随处都可以看到绚丽的动画效果,实现这些动画的过程并不复杂,今天将带大家一窥iOS动画全貌。在这里你可以看...
    F麦子阅读 5,091评论 5 13
  • CAAnimation CAAnimation 是一个抽像类。CAAnimation 也派生出了很多子类,我们使用...
    谢谢生活阅读 1,350评论 0 9
  • 一.CoreAnimation介绍 CoreAnimation是一套图像渲染和动画基础框架,其在iOS和OSX平台...
    AlexCorleone阅读 964评论 0 15
  • 继续关于幸福的讨论。讨论一个问题,如果缺乏定义,推进会很困难的。比如,我问你韩国好不好,你说韩国的泡菜很好,但是我...
    蔚林阅读 183评论 1 1