Core Animation 第七章 隐式动画

本章开始将正式进入动画的部分,首先要介绍的是隐式动画。所谓隐式动画就是由系统自动完成的动画。

事务

Core Animation基于一个假设,说屏幕上的任何东西都可以(或者可能)做动画。 动画并不需要你在Core Animation中手动打开,相反需要明确地关闭,否则他会一 直存在。例如下面的例子,在改变CALayer背景色的时候它会自己从旧值平滑的过渡到新值。

class TransactionViewController: UIViewController {
    @IBOutlet weak var layerView: UIView!
    weak var colorLayer: CALayer!
    override func viewDidLoad() {
        super.viewDidLoad()
        let layer = CALayer()
        self.layerView.layer.addSublayer(layer)
        layer.frame = CGRect(x: 35, y: 20, width: 180, height: 180)
        layer.backgroundColor = UIColor.blue.cgColor;
        colorLayer = layer
    }
    
    @IBAction func changeBtnClick(_ sender: UIButton) {
        let red = CGFloat(arc4random() % 256) / 255.0
        let green = CGFloat(arc4random() % 256) / 255.0
        let blue = CGFloat(arc4random() % 256) / 255.0
        colorLayer.backgroundColor = UIColor(red: red, green: green, blue: blue, alpha: 1.0).cgColor
    }
}
默认颜色渐变

这个就是一个隐式动画,我们没有指定动画类型,只是改变了一个属性,然后由系统来自动完成动画效果。
那么这些又跟事务有什么关系呢?iOS中的事务可以理解为一系列动画的集合,任何用指定事 务去改变可以做动画的图层属性都不会立刻发生变化,而是当事务一旦提交的时候 开始用一个动画过渡到新值。

事务是通过 CATransaction 类来做管理。CATransaction 是在图层树改变的时候在一个没有活跃事务的线程中由CoreAnimation自动创建的,并且在run-loop重复的时候自动提交。CATransaction 没有属性或者实例方法,不需要使用者来单独创建。但是可以用 begin()commit() 分别来入栈或者出栈。

任何可以做动画的图层属性都会被添加到栈顶的事务,你可以通
setAnimationDuration(_:) 来设置或者通过 animationDuration() 来获取 当前动画的时间(默认值为0.25)

接下来我们来让上面例子中颜色的改变慢一点。

@IBAction func changeBtnClick(_ sender: UIButton) {
      //开始事务
        CATransaction.begin()
        defer {
            //提交事务
            CATransaction.commit()
        }
        CATransaction.setAnimationDuration(1.0)
        let red = CGFloat(arc4random() % 256) / 255.0
        let green = CGFloat(arc4random() % 256) / 255.0
        let blue = CGFloat(arc4random() % 256) / 255.0
        colorLayer.backgroundColor = UIColor(red: red, green: green, blue: blue, alpha: 1.0).cgColor
    }
使用事务来控制动画的时间

当然UIView中也有对应的方法 beginAnimations(_ : , context: ) , commitAnimations() 等方法。当然最常用的还是 UIView中基于闭包的动画方法:animate(withDuration: , animations:)

完成块

上面提到了 UIView 中常用的使用闭包来处理动画的方式,而且还有有一个 completion 的闭包来表示动画已经执行结束(完成块)。CATranscation 中也有对应的方法 setCompletionBlock() 作为动画完成后的回调。下面我们来让方块每次改变颜色后都旋转90°。

@IBAction func changeBtnClick(_ sender: UIButton) {
        //开始事务
        CATransaction.begin()
        defer {
            //提交事务
            CATransaction.commit()
        }
        CATransaction.setAnimationDuration(1.0)
        CATransaction.setCompletionBlock { 
            var transform = self.colorLayer.affineTransform()
            transform = transform.rotated(by: CGFloat(M_PI_2))
            self.colorLayer.setAffineTransform(transform)
        }
        let red = CGFloat(arc4random() % 256) / 255.0
        let green = CGFloat(arc4random() % 256) / 255.0
        let blue = CGFloat(arc4random() % 256) / 255.0
        colorLayer.backgroundColor = UIColor(red: red, green: green, blue: blue, alpha: 1.0).cgColor
    }
使用完成块旋转图层

图层行为

处于好奇,你可能会尝试打开事务之后直接对视图,也就是对UIView设置背景颜色。然后你就会发现视图的背景颜色是瞬间改变的,并没有渐变的过程。原因也很简单,隐式动画被UIView禁用掉了。
之前也提到过,UIView显示的内容实际都是它内部的关联图层上的内容,那么UIView是怎么禁用掉内部图层的隐式动画的呢?当我们改变图层属性的时候,它会调用 action(forKey:) 方法。关于 action(forKey:) 的调用我们可以在CALayer的头文件中找到说明:

  • 首先查看代理是否实现了 action(for layer: CALayer, forKey event: String) 如果实现了则调用并返回结果
  • 如果没有代理或者代理未实现该方法,那么图层接着会检查包含属性名称对应行为的 action 字典
  • 如果 action中没有包含的属性,那么会继续在图层的style字典中查找属性名
  • 如果 style 中也没有查找到的话,图层会直接调用 defaultActionForKey()方法,返回一个标准行为。

结果也就很明显了,UIView为继承了代理,并且如果当前View不在动画块内的话,action(forKey:) 方法就返回nil。

override func viewDidLoad() {
        super.viewDidLoad()
        print("Outside: \(layerView .action(for: self.layerView.layer, forKey: "backgroundColor"))")
        UIView.beginAnimations(nil, context: nil)
        print("Inside: \(layerView .action(for: self.layerView.layer, forKey: "backgroundColor"))")
        UIView.commitAnimations();
    }
视图开启事务与未开启事务 actionForKey 结果对比

当然我们也可以通过 CATransaction 来禁用动画,在 CATransaction.begin() 之后调用

CATransaction.setDisableActions(true)

铺垫了这么多,接下来我们尝试通过为colorLayer定义一个action字典来实现自定义图层行为,例如我们希望新的颜色不是渐变的,而是从左侧划入的。

class TransactionViewController: UIViewController {
    @IBOutlet weak var layerView: UIView!
    weak var colorLayer: CALayer!
    override func viewDidLoad() {
        super.viewDidLoad()
        let layer = CALayer()
        self.layerView.layer.addSublayer(layer)
        layer.frame = CGRect(x: 35, y: 20, width: 180, height: 180)
        layer.backgroundColor = UIColor.clear.cgColor;
        let transition = CATransition()
        transition.type = kCATransitionPush
        transition.subtype = kCATransitionFromLeft
        layer.actions = ["backgroundColor": transition]
        colorLayer = layer
    }
    
    @IBAction func changeBtnClick(_ sender: UIButton) {
        //开始事务
        CATransaction.begin()
        defer {
            //提交事务
            CATransaction.commit()
        }
        CATransaction.setAnimationDuration(1.0)
        CATransaction.setCompletionBlock { 
            var transform = self.colorLayer.affineTransform()
            transform = transform.rotated(by: CGFloat(M_PI_2))
            self.colorLayer.setAffineTransform(transform)
        }
        let red = CGFloat(arc4random() % 256) / 255.0
        let green = CGFloat(arc4random() % 256) / 255.0
        let blue = CGFloat(arc4random() % 256) / 255.0
        colorLayer.backgroundColor = UIColor(red: red, green: green, blue: blue, alpha: 1.0).cgColor
    }
}
自定义图层背景颜色行为

呈现(presentation)与模型(model)

这部分内容比较繁琐,我就只是引用原文的内容了,不过简单来讲就是我们看到的动画(presentation)与我们实际操作的图层(model)时两个图层。
CALayer的属性行为其实很不正常,因为改变一个图层的属性并没有立刻生效,而是通过一段时间渐变更新。这是怎么做到的呢?
当你改变一个图层的属性,属性值的确是立刻更新的(如果你读取它的数据,你会发现它的值在你设置它的那一刻就已经生效了),但是屏幕上并没有马上发生改变。这是因为你设置的属性并没有直接调整图层的外观,相反,他只是定义了图层动画结束之后将要变化的外观。
当设置 CALayer 的属性,实际上是在定义当前事务结束之后图层如何显示的模型。Core Animation 扮演了一个控制器的角色,并且负责根据图层行为和事务设置去不断更新视图的这些属性在屏幕上的状态。
我们讨论的就是一个典型的微型 MVC 模式。CALayer 是一个连接用户界面(就是 MVC 中的 view )虚构的类,但是在界面本身这个场景下,CALayer 的行为更像是存储了视图如何显示和动画的数据模型。实际上,在苹果自己的文档中,图层树通常都是值的图层树模型。
iOS中,屏幕每秒钟重绘60次。如果动画时长比60分之一秒要长, Core Animation 就需要在设置一次新值和新值生效之间,对屏幕上的图层进行重新组织。这意味着 CALayer 除了“真实”值(就是你设置的值)之外,必须要知道当前显示在屏幕上的属性值的记录。
每个图层属性的显示值都被存储在一个叫做呈现图层的独立图层当中,他可以通过 -presentationLayer 方法来访问。这个呈现图层实际上是模型图层的复制,但是它的属性值代表了在任何指定时刻当前外观效果?;痪浠八?,你可以通过呈现图层的值来获取当前屏幕上真正显示出来的值。
我们在第一章中提到除了图层树,另外还有呈现树。呈现树通过图层树中所有图层的呈现图层所形成。注意呈现图层仅仅当图层首次被提交(就是首次第一次在屏幕上显示)的时候创建,所以在那之前调用 -presentationLayer 将会返回nil。
你可能注意到有一个叫做 –modelLayer 的方法。在呈现图层上调用 –modelLayer 将会返回它正在呈现所依赖的 CALayer 。通常在一个图层上调用 -modelLayer 会返回 self(实际上我们已经创建的原始图层就是一种数据模型)。

大多数情况下,你不需要直接访问呈现图层,你可以通过和模型图层的交互,来让 Core Animation 更新显示。两种情况下呈现图层会变得很有用,一个是同步动画,一个是处理用户交互。

  • 如果你在实现一个基于定时器的动画,而不仅仅是基于事务的动画,这个时候准确地知道在某一时刻图层显示在什么位置就会对正确摆放图层很有用了。
  • 如果你想让你做动画的图层响应用户输入,你可以使用-hitTest:方法来判断指定图层是否被触摸,这时候对呈现图层而不是模型图层调用-hitTest:会显得更有意义,因为呈现图层代表了用户当前看到的图层位置,而不是当前动画结束之后的位置。

接下来的例子中,点击屏幕上的任意位置将会让图层平移到那里。点击图层本身可以随机改变它的颜色。我们通过对呈现图层调用 -hitTest: 来判断是否被点击。
如果修改代码让 -hitTest: 直接作用于 colorLayer 而不是呈现图层,你会发现当图层移动的时候它并不能正确显示。这时候你就需要点击图层将要移动到的位置而不是图层本身来响应点击(这就是为什么用呈现图层来响应交互的原因)。

class HitTestViewController: UIViewController {
    weak var colorLayer: CALayer!
    override func viewDidLoad() {
        super.viewDidLoad()
        let layer = CALayer()
        view.layer.addSublayer(layer)
        layer.frame = CGRect(x: 0, y: 0, width: 100, height: 100)
        layer.position = view.layer.position
        layer.backgroundColor = UIColor.red.cgColor
        colorLayer = layer
    }
    
    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        let point = touches.first?.location(in: view)
        if self.colorLayer.presentation()?.hitTest(point!) != nil {
            let red = CGFloat(arc4random() % 256) / 255.0
            let green = CGFloat(arc4random() % 256) / 255.0
            let blue = CGFloat(arc4random() % 256) / 255.0
            colorLayer.backgroundColor = UIColor(red: red, green: green, blue: blue, alpha: 1.0).cgColor
        } else {
            CATransaction.begin()
            CATransaction.setAnimationDuration(4.0)
            colorLayer.position = point!
            CATransaction.commit()
        }
    }
}

总结

本章主要介绍了 Core Animation 的隐式动画已经背后实现的原理。

往期回顾:

序章
第一章 - 图层树
第二章 - 寄宿图
第三章 - 图层几何
第四章 - 视觉效果
第五章 - 变换
第六章 专用图层(上)
第六章 专用图层(下)
项目中使用的代码

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

推荐阅读更多精彩内容