2015-01-23 01:02:08 +01:00
|
|
|
import UIKit
|
2015-01-22 11:54:27 +01:00
|
|
|
|
2015-01-23 01:02:08 +01:00
|
|
|
/// A subclass of `UIImageView` that can be animated using an image name string or raw data.
|
2015-06-04 19:24:34 -04:00
|
|
|
public class AnimatableImageView: UIImageView {
|
2015-01-23 01:02:08 +01:00
|
|
|
/// An `Animator` instance that holds the frames of a specific image in memory.
|
2015-06-09 14:28:11 -07:00
|
|
|
var animator: Animator?
|
2015-06-04 19:24:34 -04:00
|
|
|
/// A display link that keeps calling the `updateFrame` method on every screen refresh.
|
2015-06-09 14:28:11 -07:00
|
|
|
lazy var displayLink: CADisplayLink = CADisplayLink(target: self, selector: Selector("updateFrame"))
|
2015-01-22 11:54:27 +01:00
|
|
|
|
2015-06-04 19:36:47 -04:00
|
|
|
/// The size of the frame cache.
|
|
|
|
public var framePreloadCount = 50
|
2015-06-04 15:38:52 -04:00
|
|
|
|
2015-01-23 01:02:08 +01:00
|
|
|
/// A computed property that returns whether the image view is animating.
|
2015-01-22 11:54:27 +01:00
|
|
|
public var isAnimatingGIF: Bool {
|
2015-06-04 19:24:34 -04:00
|
|
|
return !displayLink.paused
|
2015-01-22 11:54:27 +01:00
|
|
|
}
|
|
|
|
|
2015-01-23 01:02:08 +01:00
|
|
|
/// Prepares the frames using a GIF image file name, without starting the animation.
|
|
|
|
/// The file name should include the `.gif` extension.
|
|
|
|
///
|
2015-06-09 14:28:11 -07:00
|
|
|
/// - parameter imageName: The name of the GIF file. The method looks for the file in the app bundle.
|
2015-01-22 11:54:27 +01:00
|
|
|
public func prepareForAnimation(imageNamed imageName: String) {
|
2015-06-09 14:28:11 -07:00
|
|
|
let imagePath = NSBundle.mainBundle().bundleURL.URLByAppendingPathComponent(imageName)
|
|
|
|
prepareForAnimation <^> NSData(contentsOfURL: imagePath)
|
2015-01-22 11:54:27 +01:00
|
|
|
}
|
|
|
|
|
2015-01-23 01:02:08 +01:00
|
|
|
/// Prepares the frames using raw GIF image data, without starting the animation.
|
|
|
|
///
|
2015-06-09 14:28:11 -07:00
|
|
|
/// - parameter data: GIF image data.
|
2015-01-22 11:54:27 +01:00
|
|
|
public func prepareForAnimation(imageData data: NSData) {
|
|
|
|
image = UIImage(data: data)
|
2015-06-04 19:36:47 -04:00
|
|
|
animator = Animator(data: data, size: frame.size, contentMode: contentMode, framePreloadCount: framePreloadCount)
|
|
|
|
animator?.prepareFrames()
|
2015-06-04 19:24:34 -04:00
|
|
|
attachDisplayLink()
|
2015-01-22 11:54:27 +01:00
|
|
|
}
|
|
|
|
|
2015-01-23 01:02:08 +01:00
|
|
|
/// Prepares the frames using a GIF image file name and starts animating the image view.
|
|
|
|
///
|
2015-06-09 14:28:11 -07:00
|
|
|
/// - parameter imageName: The name of the GIF file. The method looks for the file in the app bundle.
|
2015-01-22 11:54:27 +01:00
|
|
|
public func animateWithImage(named imageName: String) {
|
|
|
|
prepareForAnimation(imageNamed: imageName)
|
|
|
|
startAnimatingGIF()
|
|
|
|
}
|
|
|
|
|
2015-01-23 01:02:08 +01:00
|
|
|
/// Prepares the frames using raw GIF image data and starts animating the image view.
|
|
|
|
///
|
2015-06-09 14:28:11 -07:00
|
|
|
/// - parameter data: GIF image data.
|
|
|
|
public func animateWithImageData(data: NSData) {
|
2015-01-22 11:54:27 +01:00
|
|
|
prepareForAnimation(imageData: data)
|
|
|
|
startAnimatingGIF()
|
|
|
|
}
|
|
|
|
|
2015-06-09 14:28:11 -07:00
|
|
|
/// Updates the `image` property of the image view if necessary. This method should not be called manually.
|
|
|
|
override public func displayLayer(layer: CALayer) {
|
2015-02-10 20:07:54 +01:00
|
|
|
image = animator?.currentFrame
|
2015-01-22 11:54:27 +01:00
|
|
|
}
|
|
|
|
|
2015-01-23 01:02:08 +01:00
|
|
|
/// Starts the image view animation.
|
2015-01-22 11:54:27 +01:00
|
|
|
public func startAnimatingGIF() {
|
2015-06-04 19:24:34 -04:00
|
|
|
if animator?.isAnimatable ?? false {
|
|
|
|
displayLink.paused = false
|
|
|
|
}
|
2015-01-22 11:54:27 +01:00
|
|
|
}
|
|
|
|
|
2015-01-23 01:02:08 +01:00
|
|
|
/// Stops the image view animation.
|
2015-01-22 11:54:27 +01:00
|
|
|
public func stopAnimatingGIF() {
|
2015-06-04 19:24:34 -04:00
|
|
|
displayLink.paused = true
|
2015-01-22 11:54:27 +01:00
|
|
|
}
|
2015-06-04 15:38:52 -04:00
|
|
|
|
2015-06-09 14:28:11 -07:00
|
|
|
/// Update the current frame with the displayLink duration
|
|
|
|
func updateFrame() {
|
|
|
|
if animator?.updateCurrentFrame(displayLink.duration) ?? false {
|
|
|
|
layer.setNeedsDisplay()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-04 19:36:47 -04:00
|
|
|
/// Invalidate the displayLink so it releases this object.
|
2015-06-09 14:28:11 -07:00
|
|
|
deinit {
|
2015-06-04 19:36:47 -04:00
|
|
|
displayLink.invalidate()
|
2015-06-04 15:38:52 -04:00
|
|
|
}
|
2015-06-04 19:24:34 -04:00
|
|
|
|
2015-06-04 19:36:47 -04:00
|
|
|
/// Attaches the display link.
|
2015-06-09 14:28:11 -07:00
|
|
|
func attachDisplayLink() {
|
2015-06-04 19:36:47 -04:00
|
|
|
displayLink.addToRunLoop(.mainRunLoop(), forMode: NSRunLoopCommonModes)
|
2015-06-04 19:24:34 -04:00
|
|
|
}
|
2015-01-22 11:54:27 +01:00
|
|
|
}
|
|
|
|
|