iOS 核心动画

自己做的笔记,过段时间却又忘记了,为了尽量避免这种情况,同时治服拖延种种,打算开始多写写技术向的博客。


UIKit Animation or Core Animation?

在 iOS 中,动画可以分为两个类别:UIKit,和底层的 Core Animation。

通过最基本的 UIView.animate(withDuration:) 方法建立如下的 UIKit animation,能胜任复杂程度不高的动画:

UIView.animate(withDuration: 0.8) {
      self.orangeBlock.frame = CGRect(x: 38, y: 70, width: 300, height: 60)
}![image.png](https://upload-images.jianshu.io/upload_images/8048781-4b3a1db18a8c7915.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

但是当要处理不单一的属性时,往往会出现一些不合预期的效果,这是因为,用 UIKit 接口创建的动画是基于视图(view)的,它并不能直接对图层(layer)属性作出更改。

为了制作出能符合预期的动画效果,我们就需要直接对图层进行控制,这就是 Core Animation 了。

基于 Core Animation,我们可以创建两步的 CABasicAnimation,或者用 CAKeyframAnimation 绘制路径等。

Layer v.s. View

图层不是视图的替代,而是后者的底层支持。在能满足需求的情况下,对视图而不是图层进行操作——即使用 UIKit 动画——是被建议的做法。

在 iOS 开发中,通常是每一个视图有一个对应的图层(称为 layer-backed view),图层对象会保证两者保持同步。然而在一些特殊情况下,这种一对一的关系不是必须的(比如多个图层支持一个单独 UIImageView 来节省因单张图片重复出现的内存占用)。

image

基本过渡动画 - CABasicAnimation

想要对哪一组值进行设置,知道对应的 keyPath 就好了。

let radiusAnimation = CABasicAnimation(keyPath: "cornerRadius")
radiusAnimation.fromValue = 50
radiusAnimation.toValue = 30
radiusAnimation.duration = 3

purpleBlock.layer.add(radiusAnimation, forKey: "cornerRadius")
// 为模型图层设置新的半径
purpleBlock.layer.cornerRadius = 30

需要注意的一点是,add(_: CAAnimation, forKey:) 设置的动画针对的是模型图层(model layer)而不是显示图层(presentation layer),当动画显示完,它就被移除了,显示图层属性将会恢复成初始的模型图层属性。所以,为了保持对象视图变化后的状态,要记得手动更新。

Core Animation 控制的图层中包含有两套平行的继承树:模型图层树(model layer tree)和显示图层树(presentation layer tree)。前者反映的是图层状态,后者是图层在动画中动态的值。如果要实时跟踪图层的属性变化,要检测的是显示图层的值。

关键帧动画 - CAKeyframeAnimation

现在,试想我们要实现的动画超过了两步(而这是非常常见的情况),那么以 fromValuetoValue 简单两组值就显然过于局限了。这时候我们用 CAKeyframeAnimation 就能任意地定义和设置帧。

let shakeAnimation = CAKeyframeAnimation(keyPath: "position.x")
shakeAnimation.values = [0, 10, -10, 10, 0]
shakeAnimation.keyTimes = [0, 0.2, 0.6, 0.8, 1]
shakeAnimation.duration = 0.4
shakeAnimation.isAdditive = true

blueBlock.layer.add(shakeAnimation, forKey: "position.x")

values 数组代表的是对象的位置,keyTimes 数组代表划分的时间段。

除了使用状态数组的方式,设定路径(CGPath)也是可以的:

let boundingRect = CGRect(x: -150, y: -150, width: 300, height: 300)
let orbitAnimation = CAKeyframeAnimation(keyPath: "position")
orbitAnimation.path = CGPath(ellipseIn: boundingRect, transform: nil)
orbitAnimation.duration = 4
orbitAnimation.isAdditive = true
orbitAnimation.repeatCount = Float.infinity
orbitAnimation.calculationMode = kCAAnimationPaced
orbitAnimation.rotationMode = kCAAnimationRotateAuto
​
greyBlock.layer.add(orbitAnimation, forKey: "position")

CAAnimation 的可重用性

一个动画实例由 keyPath 指定其所定义的图层属性类型,因此对于该属性(如 cornerRadius)不必再重复创建新实例,也即是能被多个对象所共用的。

为了提高其可重用性,我们还可以用 byValue 来替代 CABasicAnimation 中的 toValue,前者设定的是变化量。

let radiusAnimation = CABasicAnimation(keyPath: "cornerRadius")
radiusAnimation.fromValue = 50
// radiusAnimation.toValue = 30
// equals:
radiusAnimation.byValue = -20

如果同时创建了多个动画,可以使用 CAAnimationGroup 打包成一个动画组:

let aniGroup = CAAnimationGroup()
aniGroup.animations = [shakeAnimation, radiusAnimation]
aniGroup.duration = 3
​
magentaBlock.layer.add(aniGroup, forKey: "whatsoever")

Timing Functions 使动画流畅

为了让动画看起来更自然,添加一个 CAMediaTimingFunction 对象实例。可以直接对动画实例设置:

// 现有的「淡入淡出」效果
let timingFunc1 = CAMediaTimingFunction(name: kCAMediaTimingFunctionEaseInEaseOut)
ANI_INSTANCE.timingFunction = timingFunc

也可以为 CATransaction 设置:

// 自定义函数,通过 controlPoints 设置贝塞尔曲线
let timingFunc2 = CAMediaTimingFunction(controlPoints: 0.65, -0.55, 0.27, 1.55)
CATransaction.setAnimationTimingFunction(timingFunc)

CATransaction 事务机制

当我们要进行多组图层操作时,一个很好的习惯是把它们显式(explicitly)合并到一个事务中。

CATransaction Definition: A mechanism for grouping multiple layer-tree operations into atomic updates to the render tree.

事实上,即使我们不主动创建 CATransaction 事务,系统也会默认给我们创建一个隐式(implicit)事务。显式声明还有个好处是,可以为事务内部的代码统一设置一些默认值(如 duration)。嵌套的 CATransaction 是允许的。

CATransaction.begin()
CATransaction.setAnimationDuration(3)

let positionAnimation = CABasicAnimation(keyPath: "position")
positionAnimation.fromValue = [38, 484]
positionAnimation.toValue = [112, 484]
greenBlockLayer.add(positionAnimation, forKey: "position")
greenBlockLayer.position = CGPoint(x: 112, y: 484)

CATransaction.begin()
CATransaction.setAnimationDuration(1)
​
UIView.animate(withDuration: b_Interval) {
 self.greenBlock.alpha = 0.5
}
​
CATransaction.commit()
CATransaction.commit()

ref:

  1. Design Patterns on iOS using Swift - Part 1/2 - raywenderlich
  2. Design Patterns on iOS using Swift – Part 2/2 - raywenderlich
  3. 设计模式 - 菜鸟教程
  4. KVO - Swifter
  5. Key-Value Observing Programming Guide - Apple
  6. Is key-value observation (KVO) available in Swift? - Stack Overflow
  7. Exploring KVO alternatives with Swift - Scott Logic
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容