iOS中有很多方法可以实现动画,我们可以用CAKeyframeAnimation, CABasicAnimation,CASpringAnimation(iOS9.0中添加的,实现弹簧的效果),也可以用UIView中为我们提供的动画接口,不管用哪种方法,目的就是实现动画效果,这里重点讲解CAKeyframeAnimation和CABasicAnimation。

在学习动画之前,我们需要了解一些知识,大家都知道动画是作用在图层上面的,我们首先要了解图层。

认识CALayer

大家平时使用最多的就是UIView,我们创建一个视图时,其实就是创建了一个与视图相关的图层,视图负责管理着创建的图层,通过图层来显示相应内容。我们可以直接创建一个图层,然后将它添加到父图层上,这样就能看到我们创建的图层(代码如下)

 CALayer *layer = [CALayer layer];
layer.frame = CGRectMake(50, 200, 80, 80);
layer.backgroundColor = [UIColor greenColor].CGColor;
[self.view.layer addSublayer:layer];

效果图:

可能大家想知道 CALayer和UIView的区别在哪?也许最大的区别就是UIView可以和用户交互,CALayer不能交互。相信我们的目的并不是显示一个纯色的矩形块,其实CALayer也可以显示其他类型的内容。

CALayer属性

contents

查看头文件可以知道这个属性的类型为id,也就是说他可以显示任何类型的对象,但实践中,如果不是CGImage类型,将会显示空白,这点会让人产生疑惑,其实还有更奇怪的,你要赋给的类型不是CGImage,而是CGImageRef(指向CGImage结构的指针),UIImage有一个CGImage属性,返回CGImageRef,但是如果把这个值直接赋给contents的话会得到编译错误,因为CGImageRef是一个Core Foundation类型,我们需要用如下代码进行赋值:

layer.contents = (__bridge id)image.CGImage;

以上代码块的前提是使用ARC,不使用ARC的话,不需要__bridge

接下来我们来显示一张图片,代码如下:

// 图片是一个200x200的图,layer的frame故意设置成矩形,为了演示变形
UIImage *image = [UIImage imageNamed:@"image.png"];
CALayer *layer = [CALayer layer];
layer.frame = CGRectMake(50, 150, 80, 100);
layer.contents = (__bridge id)image.CGImage;
[self.view.layer addSublayer:layer];

效果图如下:

contentGravity

如上面代码所示,我们得到了一个变形了的图片,这当然不是我们想要的,如果想让图片正确显示,我们可以设置contentGravity,这个属性用法类似UIView中的contentMode属性,我们可以看一下它的枚举:

  • kCAGravityCenter
  • kCAGravityTop
  • kCAGravityBottom
  • kCAGravityLeft
  • kCAGravityRight
  • kCAGravityTopLeft
  • kCAGravityTopRight
  • kCAGravityBottomLeft
  • kCAGravityBottomRight
  • kCAGravityResize
  • kCAGravityResizeAspect
  • kCAGravityResizeAspectFill

是不是看着很熟悉啊!这对上面图片变形,我们可以设置这个属性值为kCAGravityResizeAspect,这样就能得到我们想要的正确的效果了

添加代码:

layer.contentsGravity = kCAGravityResizeAspect;

就能得到显示正确的图:

contentsScale

contentsScale属性定义了寄宿图的像素尺寸和视图大小的比例,默认情况下它是一个值为1.0的浮点数。

如果你设置了contentsGravity属性,contentsScale就不在寄宿图有影响,因为它已经被拉伸以适应图层的边界。

contentsScale属性其实属于支持高分辨率(又称Hi-DPI或Retina)屏幕机制的一部分,它用来判断在绘制图层的时候应该为寄宿图创建的空间大小,和需要显示的图片的拉伸度(假设并没有设置contentsGravity属性),UIView有一个类似功能但是非常少用到的contentScaleFactor属性。

maskToBounds

如果把contentsGravity设置成kCAGravityCenter,我们看到图片超出了视图边界

UIView有一个叫做clipsToBounds的属性可以用来决定是否显示超出边界的内容,CALayer对应的属性叫做masksToBounds,把它设置为YES,即可看到图片被剪切了,如下图:

custom drawing

contents赋CGImage的值不是唯一的设置寄宿图的方法。我们也可以直接用Core Graphics直接绘制寄宿图。能够通过继承UIView并实现-drawRect:方法来自定义绘制。

如果你不需要寄宿图,那就不要创建这个方法了,这会造成CPU资源和内存的浪费,这也是为什么苹果建议:如果没有自定义绘制的任务就不要在子类中写一个空的-drawRect:方法。

当视图出现在屏幕上时,-drawRect:方法会被调用,通常是开发者自己调用-setNeedsDisplay方法

CALayer有一个可选的delegate属性,实现了CALayerDelegate协议,当CALayer需要一个内容特定的信息时,就会从协议中请求。

当需要被重绘时,CALayer会请求它的代理给它一个寄宿图来显示。它通过调用下面这个方法做到的:

- (void)displayLayer:(CALayer *)layer

如果代理想直接设置contents属性的话,它就可以在这个方法中做,不然没有别的方法可以调用了

如果代理不实现-displayLayer:方法,CALayer就会转而尝试调用下面这个方法:

- (void)drawLayer:(CALayer *)layer inContext:(CGContextRef)ctx

在调用这个方法之前,CALayer创建了一个合适尺寸的空寄宿图(尺寸由boundscontentsScale决定)和一个Core Graphics的绘制上下文环境,为绘制寄宿图做准备,它作为ctx参数传入。

如下例子:

代码:


- (void)viewDidLoad {


[super viewDidLoad];

CALayer *blueLayer = [CALayer layer];

blueLayer.frame = CGRectMake(50.0f, 50.0f, 100.0f, 100.0f);

blueLayer.backgroundColor = [UIColor blueColor].CGColor;

blueLayer.delegate = self;

blueLayer.contentsScale = [UIScreen mainScreen].scale; //add layer to our view

[self.view.layer addSublayer:blueLayer];


[blueLayer display];


// Do any additional setup after loading the view, typically from a nib.

}


- (void)drawLayer:(CALayer *)layer inContext:(CGContextRef)ctx {


CGContextSetLineWidth(ctx, 5);


CGContextSetStrokeColorWithColor(ctx, [UIColor redColor].CGColor);


CGContextStrokeEllipseInRect(ctx, layer.bounds);


}


效果图:

iOS动画学习-CALayer的更多相关文章

  1. iOS动画学习-视觉效果

    CALayer不仅仅是iOS动画学习-CALayer中介绍的那些内容,他还有一些其他属性,比如shadowColor,borderWidth,borderColor等等,这些属性我们只需要简单点设置就 ...

  2. iOS动画学习

    学习一下动画,感谢以下大神的文章:    UIView:基础动画.关键帧动画.转场动画 Core Animation :基础动画,关键帧动画,动画组,转场动画,逐帧动画 CALayer :CALaye ...

  3. iOS 动画学习

    图层树.寄宿图以及图层几何学(一)图层的树状结构 技术交流新QQ群:414971585 巨妖有图层,洋葱也有图层,你有吗?我们都有图层 -- 史莱克 Core Animation其实是一个令人误解的命 ...

  4. ios 动画学习的套路 (二)

    有它们俩你就够了! 说明:下面有些概念我说的不怎么详细,网上实在是太多了,说了我觉得也意义不大了!但链接都给大家了,可以自己去看,重点梳理学习写动画的一个过程和一些好的博客! (一) 说说这两个三方库 ...

  5. iOS动画学习 -隐式动画

    事务 Core Animation基于一个假设,说屏幕上的任何东西都可以(或者可能)做动画.你并不需要在Core Animation中手动打开动画,但是你需要明确地关闭它,否则它会一直存在. 当你改变 ...

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

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

  7. iOS 动画笔记 (一)

    你也肯定喜欢炫酷的动画! 在APP中,动画就是一个点睛之笔!可以给用户增加一些独特的体验感,估计也有许多的和我一样的,看着那些觉得不错的动画,也就只能流口水的孩子,毕竟可能不知道从哪里下手去写!动画学 ...

  8. iOS 动画Animation - 5:UIBezier

    首先说明:这是一系列文章,參考本专题下其它的文章有助于你对本文的理解. 在之前的bolg中大家会发现总是会出现UIBezier,可是我也没有做过多介绍,今天就集中介绍一下UIBezier.首先.UIB ...

  9. iOS核心动画学习整理

    最近利用业余时间终于把iOS核心动画高级技巧(https://zsisme.gitbooks.io/ios-/content/chapter1/the-layer-tree.html)看完,对应其中一 ...

随机推荐

  1. Swift 入门之简单语法(五)

    面向对象 目标 构造函数 构造函数的基本概念 构造函数的执行顺序 KVC 在构造函数中的使用及原理 便利构造函数 析构函数 区分 重载 和 重写 懒加载 只读属性(计算型属性) 设置模型数据(didS ...

  2. 机器学习:Python实现聚类算法(一)之AP算法

    1.算法简介 AP(Affinity Propagation)通常被翻译为近邻传播算法或者亲和力传播算法,是在2007年的Science杂志上提出的一种新的聚类算法.AP算法的基本思想是将全部数据点都 ...

  3. 几个常用的linux快捷键和shell知识

    1)   !$    !$是一个特殊的环境变量,它代表了上一个命令的最后一个字符串.如:你可能会这样:     $mkdir mydir     $mv mydir yourdir     $cd y ...

  4. git reset、git checkout和git revert的区别

    这三个git命令都是用来撤销代码仓库中的某些更改,而前两个命令不仅可以作用于commit层面,还可以作用于file层面Reset在commit层面,reset通过移除当前分支的一些节点来实现版本回滚; ...

  5. Django初探——工程创建以及models数据库相关配置

    Python的WEB框架有Django.Tornado.Flask 等多种,Django相较与其他WEB框架其优势为:大而全,框架本身集成了ORM.模型绑定.模板引擎.缓存.Session等诸多功能. ...

  6. 【Android】又一个Gank客户端来啦

    介绍 Gank平台的移动端又来了,非常感谢Gank平台开放接口,让我们这些小白有机会练手.学习. 本项目在架构方面有稍微花点心思,虽然还是最简单的MVC模式,但基本参考MVP的思想,Activity只 ...

  7. javascript常见面试题

    闭包相关面试题:1. var a=0,b=0; function A(a){ A=function(b){console.log(a+b++);}; console.log(a); } A(1); A ...

  8. java计算某个日期是什么节气(24节气)

    package com.test; import java.util.Calendar; import java.util.Date; /** * Created by json */ public ...

  9. python socketserver监听多端口多进程

    多进程监听多端口 # 多线程socket # 程序监听两个端口,端口逻辑相同其中一个端口放在子进程下 # 每次请求会在产生一个进程处理请求 import SocketServer from multi ...

  10. 浅谈this那些事

    一直以来,对this的讨论都是热门话题.有人说掌握了this就掌握了JavaScript的80%,说法有点夸张,但可见this的重要性.本人至今也是记录了很多关于this的零碎笔记,今天就来个小结. ...