mirror of
https://github.com/wnagrodzki/AnimationRestoration.git
synced 2025-04-05 12:02:12 +02:00
142 lines
4.3 KiB
Swift
142 lines
4.3 KiB
Swift
|
//
|
||
|
// AnimationPreservingView.swift
|
||
|
// AnimationPreservingView
|
||
|
//
|
||
|
// Created by Wojciech Nagrodzki on 08/03/2017.
|
||
|
// Copyright © 2017 Wojciech Nagrodzki. All rights reserved.
|
||
|
//
|
||
|
|
||
|
import UIKit
|
||
|
|
||
|
|
||
|
/// The `AnimationPreservingView` class keeps it's layer tree animations safe from being removed.
|
||
|
/// There are two cases when `CAAnimation` can be removed from `CALayer` automatically:
|
||
|
/// - when application goes to background
|
||
|
/// - when view backed by the layer is removed from window
|
||
|
class AnimationPreservingView: UIView {
|
||
|
|
||
|
override init(frame: CGRect) {
|
||
|
super.init(frame: frame)
|
||
|
registerForNotifications()
|
||
|
}
|
||
|
|
||
|
required init?(coder aDecoder: NSCoder) {
|
||
|
super.init(coder: aDecoder)
|
||
|
registerForNotifications()
|
||
|
}
|
||
|
|
||
|
override func willMove(toWindow newWindow: UIWindow?) {
|
||
|
super.willMove(toWindow: newWindow)
|
||
|
if newWindow == nil { layer.storeAnimations() }
|
||
|
}
|
||
|
|
||
|
override func didMoveToWindow() {
|
||
|
super.didMoveToWindow()
|
||
|
if window != nil { layer.restoreAnimations() }
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
extension AnimationPreservingView {
|
||
|
|
||
|
fileprivate func registerForNotifications() {
|
||
|
|
||
|
NotificationCenter.default.addObserver(self,
|
||
|
selector: #selector(AnimationPreservingView.applicationWillResignActive),
|
||
|
name: .UIApplicationWillResignActive,
|
||
|
object: nil)
|
||
|
NotificationCenter.default.addObserver(self,
|
||
|
selector: #selector(AnimationPreservingView.applicationDidBecomeActive),
|
||
|
name: .UIApplicationDidBecomeActive,
|
||
|
object: nil)
|
||
|
}
|
||
|
|
||
|
@objc private func applicationWillResignActive() {
|
||
|
|
||
|
guard window != nil else { return }
|
||
|
layer.storeAnimations()
|
||
|
}
|
||
|
|
||
|
@objc private func applicationDidBecomeActive() {
|
||
|
|
||
|
guard window != nil else { return }
|
||
|
layer.restoreAnimations()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
extension CALayer {
|
||
|
|
||
|
private static let association = ObjectAssociation<NSDictionary>()
|
||
|
|
||
|
private var animationsStorage: [String: CAAnimation] {
|
||
|
|
||
|
get { return CALayer.association[self] as? [String : CAAnimation] ?? [:] }
|
||
|
set { CALayer.association[self] = newValue as NSDictionary }
|
||
|
}
|
||
|
|
||
|
/// Returns a dictionary of copies of animations currently attached to the layer along with their's keys.
|
||
|
private var animationsForKeys: [String: CAAnimation] {
|
||
|
|
||
|
guard let keys = animationKeys() else { return [:] }
|
||
|
return keys.reduce([:], {
|
||
|
var result = $0
|
||
|
let key = $1
|
||
|
result[key] = (animation(forKey: key)!.copy() as! CAAnimation)
|
||
|
return result
|
||
|
})
|
||
|
}
|
||
|
|
||
|
/// Pauses the layer tree and stores it's animations.
|
||
|
func storeAnimations() {
|
||
|
|
||
|
pause()
|
||
|
depositAnimations()
|
||
|
}
|
||
|
|
||
|
/// Resumes the layer tree and restores it's animations.
|
||
|
func restoreAnimations() {
|
||
|
|
||
|
withdrawAnimations()
|
||
|
resume()
|
||
|
}
|
||
|
|
||
|
private func depositAnimations() {
|
||
|
|
||
|
animationsStorage = animationsForKeys
|
||
|
sublayers?.forEach { $0.depositAnimations() }
|
||
|
}
|
||
|
|
||
|
private func withdrawAnimations() {
|
||
|
|
||
|
sublayers?.forEach { $0.withdrawAnimations() }
|
||
|
animationsStorage.forEach { add($0.value, forKey: $0.key) }
|
||
|
animationsStorage = [:]
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
extension CALayer {
|
||
|
|
||
|
/// Pauses animations in layer tree.
|
||
|
/// - note: [Technical Q&A QA1673](https://developer.apple.com/library/ios/qa/qa1673/_index.html#//apple_ref/doc/uid/DTS40010053)
|
||
|
fileprivate func pause() {
|
||
|
|
||
|
let pausedTime = convertTime(CACurrentMediaTime(), from: nil)
|
||
|
speed = 0.0;
|
||
|
timeOffset = pausedTime;
|
||
|
}
|
||
|
|
||
|
/// Resumes animations in layer tree.
|
||
|
/// - note: [Technical Q&A QA1673](https://developer.apple.com/library/ios/qa/qa1673/_index.html#//apple_ref/doc/uid/DTS40010053)
|
||
|
fileprivate func resume() {
|
||
|
|
||
|
let pausedTime = timeOffset;
|
||
|
speed = 1.0;
|
||
|
timeOffset = 0.0;
|
||
|
beginTime = 0.0;
|
||
|
let timeSincePause = convertTime(CACurrentMediaTime(), from: nil) - pausedTime;
|
||
|
beginTime = timeSincePause;
|
||
|
}
|
||
|
}
|