Gifu/Source/ImageSourceHelpers.swift

78 lines
2.8 KiB
Swift
Raw Normal View History

2014-09-06 16:12:09 +02:00
import ImageIO
import MobileCoreServices
2015-01-23 01:02:08 +01:00
import UIKit
2014-09-06 16:12:09 +02:00
2015-01-23 01:02:08 +01:00
typealias GIFProperties = [String : Double]
let defaultDuration: Double = 0
2014-09-06 16:12:09 +02:00
2015-01-23 01:02:08 +01:00
/// Retruns the duration of a frame at a specific index using an image source (an `CGImageSource` instance).
///
/// - returns: A frame duration.
2014-09-06 16:12:09 +02:00
func CGImageSourceGIFFrameDuration(imageSource: CGImageSource, index: Int) -> NSTimeInterval {
2014-12-13 11:35:00 +01:00
if !imageSource.isAnimatedGIF { return 0.0 }
2014-09-06 16:12:09 +02:00
2016-04-24 12:19:13 +02:00
guard let properties = imageSource.GIFPropertiesAtIndex(index),
let duration = durationFromGIFProperties(properties),
let cappedDuration = capDuration(duration)
else { return defaultDuration }
2014-09-06 16:12:09 +02:00
2016-04-24 12:19:13 +02:00
return cappedDuration
2014-12-13 11:35:00 +01:00
}
2014-09-06 16:12:09 +02:00
2015-01-23 01:02:08 +01:00
/// Ensures that a duration is never smaller than a threshold value.
///
/// - returns: A capped frame duration.
func capDuration(duration: Double) -> Double? {
2014-12-13 11:35:00 +01:00
if duration < 0 { return .None }
2014-09-06 16:12:09 +02:00
let threshold = 0.02 - Double(FLT_EPSILON)
2014-12-13 11:35:00 +01:00
let cappedDuration = duration < threshold ? 0.1 : duration
return cappedDuration
}
2015-01-23 01:02:08 +01:00
/// Returns a frame duration from a `GIFProperties` dictionary.
///
/// - returns: A frame duration.
func durationFromGIFProperties(properties: GIFProperties) -> Double? {
2016-03-31 00:15:14 +02:00
guard let unclampedDelayTime = properties[String(kCGImagePropertyGIFUnclampedDelayTime)],
let delayTime = properties[String(kCGImagePropertyGIFDelayTime)]
else { return .None }
2014-12-13 11:35:00 +01:00
2016-03-31 00:15:14 +02:00
return duration(unclampedDelayTime, delayTime: delayTime)
2014-12-13 11:35:00 +01:00
}
2015-01-23 01:02:08 +01:00
/// Calculates frame duration based on both clamped and unclamped times.
///
/// - returns: A frame duration.
2016-03-31 00:15:14 +02:00
func duration(unclampedDelayTime: Double, delayTime: Double) -> Double {
2014-12-13 11:35:00 +01:00
let delayArray = [unclampedDelayTime, delayTime]
return delayArray.filter(isPositive).first ?? defaultDuration
}
2014-09-06 16:12:09 +02:00
2015-01-23 01:02:08 +01:00
/// Checks if a `Double` value is positive.
///
/// - returns: A boolean value that is `true` if the tested value is positive.
func isPositive(value: Double) -> Bool {
2014-12-13 11:35:00 +01:00
return value >= 0
}
2015-01-23 01:02:08 +01:00
/// An extension of `CGImageSourceRef` that add GIF introspection and easier property retrieval.
2014-12-13 11:35:00 +01:00
extension CGImageSourceRef {
2015-01-23 01:02:08 +01:00
/// Returns whether the image source contains an animated GIF.
///
/// - returns: A boolean value that is `true` if the image source contains animated GIF data.
2014-12-13 11:35:00 +01:00
var isAnimatedGIF: Bool {
let isTypeGIF = UTTypeConformsTo(CGImageSourceGetType(self) ?? "", kUTTypeGIF)
2014-12-13 11:35:00 +01:00
let imageCount = CGImageSourceGetCount(self)
return isTypeGIF != false && imageCount > 1
2014-09-06 16:12:09 +02:00
}
2015-01-23 01:02:08 +01:00
/// Returns the GIF properties at a specific index.
///
/// - parameter index: The index of the GIF properties to retrieve.
/// - returns: A dictionary containing the GIF properties at the passed in index.
2015-04-08 17:24:45 -04:00
func GIFPropertiesAtIndex(index: Int) -> GIFProperties? {
let imageProperties = CGImageSourceCopyPropertiesAtIndex(self, index, nil) as Dictionary?
return imageProperties?[String(kCGImagePropertyGIFDictionary)] as? GIFProperties
2014-12-13 11:35:00 +01:00
}
2014-09-06 16:12:09 +02:00
}