2015-03-23 20:28:42 +00:00
|
|
|
/**
|
|
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This source code is licensed under the BSD-style license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
|
|
*/
|
2015-03-06 00:36:41 +00:00
|
|
|
|
2015-09-02 15:25:10 +00:00
|
|
|
#import "RCTGIFImageDecoder.h"
|
2015-03-06 00:36:41 +00:00
|
|
|
|
2015-09-02 15:25:10 +00:00
|
|
|
#import <ImageIO/ImageIO.h>
|
|
|
|
#import <QuartzCore/QuartzCore.h>
|
2015-03-06 00:36:41 +00:00
|
|
|
|
2016-11-23 15:47:52 +00:00
|
|
|
#import <React/RCTUtils.h>
|
2015-09-02 15:25:10 +00:00
|
|
|
|
|
|
|
@implementation RCTGIFImageDecoder
|
|
|
|
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
|
|
|
|
- (BOOL)canDecodeImageData:(NSData *)imageData
|
2015-03-06 00:36:41 +00:00
|
|
|
{
|
2015-09-02 15:25:10 +00:00
|
|
|
char header[7] = {};
|
|
|
|
[imageData getBytes:header length:6];
|
|
|
|
|
2015-09-03 12:53:16 +00:00
|
|
|
return !strcmp(header, "GIF87a") || !strcmp(header, "GIF89a");
|
2015-09-02 15:25:10 +00:00
|
|
|
}
|
2015-03-06 00:36:41 +00:00
|
|
|
|
2015-09-04 11:35:44 +00:00
|
|
|
- (RCTImageLoaderCancellationBlock)decodeImageData:(NSData *)imageData
|
|
|
|
size:(CGSize)size
|
|
|
|
scale:(CGFloat)scale
|
2016-01-20 19:03:22 +00:00
|
|
|
resizeMode:(RCTResizeMode)resizeMode
|
2015-09-04 11:35:44 +00:00
|
|
|
completionHandler:(RCTImageLoaderCompletionBlock)completionHandler
|
2015-09-02 15:25:10 +00:00
|
|
|
{
|
|
|
|
CGImageSourceRef imageSource = CGImageSourceCreateWithData((CFDataRef)imageData, NULL);
|
2015-11-14 18:25:00 +00:00
|
|
|
NSDictionary<NSString *, id> *properties = (__bridge_transfer NSDictionary *)CGImageSourceCopyProperties(imageSource, NULL);
|
2015-03-06 00:36:41 +00:00
|
|
|
NSUInteger loopCount = [properties[(id)kCGImagePropertyGIFDictionary][(id)kCGImagePropertyGIFLoopCount] unsignedIntegerValue];
|
|
|
|
|
2015-09-04 11:35:44 +00:00
|
|
|
UIImage *image = nil;
|
2015-03-06 00:36:41 +00:00
|
|
|
size_t imageCount = CGImageSourceGetCount(imageSource);
|
2015-09-04 11:35:44 +00:00
|
|
|
if (imageCount > 1) {
|
|
|
|
|
|
|
|
NSTimeInterval duration = 0;
|
2015-11-03 22:45:46 +00:00
|
|
|
NSMutableArray<NSNumber *> *delays = [NSMutableArray arrayWithCapacity:imageCount];
|
|
|
|
NSMutableArray<id /* CGIMageRef */> *images = [NSMutableArray arrayWithCapacity:imageCount];
|
2015-09-04 11:35:44 +00:00
|
|
|
for (size_t i = 0; i < imageCount; i++) {
|
|
|
|
|
|
|
|
CGImageRef imageRef = CGImageSourceCreateImageAtIndex(imageSource, i, NULL);
|
|
|
|
if (!image) {
|
2015-10-08 18:32:11 +00:00
|
|
|
image = [UIImage imageWithCGImage:imageRef scale:scale orientation:UIImageOrientationUp];
|
2015-09-04 11:35:44 +00:00
|
|
|
}
|
|
|
|
|
2015-11-14 18:25:00 +00:00
|
|
|
NSDictionary<NSString *, id> *frameProperties = (__bridge_transfer NSDictionary *)CGImageSourceCopyPropertiesAtIndex(imageSource, i, NULL);
|
|
|
|
NSDictionary<NSString *, id> *frameGIFProperties = frameProperties[(id)kCGImagePropertyGIFDictionary];
|
2015-09-04 11:35:44 +00:00
|
|
|
|
|
|
|
const NSTimeInterval kDelayTimeIntervalDefault = 0.1;
|
|
|
|
NSNumber *delayTime = frameGIFProperties[(id)kCGImagePropertyGIFUnclampedDelayTime] ?: frameGIFProperties[(id)kCGImagePropertyGIFDelayTime];
|
|
|
|
if (delayTime == nil) {
|
|
|
|
if (i == 0) {
|
|
|
|
delayTime = @(kDelayTimeIntervalDefault);
|
|
|
|
} else {
|
|
|
|
delayTime = delays[i - 1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const NSTimeInterval kDelayTimeIntervalMinimum = 0.02;
|
|
|
|
if (delayTime.floatValue < (float)kDelayTimeIntervalMinimum - FLT_EPSILON) {
|
2015-03-06 00:36:41 +00:00
|
|
|
delayTime = @(kDelayTimeIntervalDefault);
|
|
|
|
}
|
2015-09-04 11:35:44 +00:00
|
|
|
|
|
|
|
duration += delayTime.doubleValue;
|
|
|
|
delays[i] = delayTime;
|
|
|
|
images[i] = (__bridge_transfer id)imageRef;
|
2015-03-06 00:36:41 +00:00
|
|
|
}
|
2015-09-04 11:35:44 +00:00
|
|
|
CFRelease(imageSource);
|
2015-03-06 00:36:41 +00:00
|
|
|
|
2015-11-03 22:45:46 +00:00
|
|
|
NSMutableArray<NSNumber *> *keyTimes = [NSMutableArray arrayWithCapacity:delays.count];
|
2015-09-04 11:35:44 +00:00
|
|
|
NSTimeInterval runningDuration = 0;
|
|
|
|
for (NSNumber *delayNumber in delays) {
|
|
|
|
[keyTimes addObject:@(runningDuration / duration)];
|
|
|
|
runningDuration += delayNumber.doubleValue;
|
2015-03-06 00:36:41 +00:00
|
|
|
}
|
|
|
|
|
2015-09-04 11:35:44 +00:00
|
|
|
[keyTimes addObject:@1.0];
|
2015-03-06 00:36:41 +00:00
|
|
|
|
2015-09-04 11:35:44 +00:00
|
|
|
// Create animation
|
|
|
|
CAKeyframeAnimation *animation = [CAKeyframeAnimation animationWithKeyPath:@"contents"];
|
|
|
|
animation.calculationMode = kCAAnimationDiscrete;
|
|
|
|
animation.repeatCount = loopCount == 0 ? HUGE_VALF : loopCount;
|
|
|
|
animation.keyTimes = keyTimes;
|
|
|
|
animation.values = images;
|
|
|
|
animation.duration = duration;
|
2016-06-24 10:18:23 +00:00
|
|
|
animation.removedOnCompletion = NO;
|
2015-09-04 11:35:44 +00:00
|
|
|
image.reactKeyframeAnimation = animation;
|
2015-03-06 00:36:41 +00:00
|
|
|
|
2015-09-04 11:35:44 +00:00
|
|
|
} else {
|
2015-03-06 00:36:41 +00:00
|
|
|
|
2015-09-04 11:35:44 +00:00
|
|
|
// Don't bother creating an animation
|
|
|
|
CGImageRef imageRef = CGImageSourceCreateImageAtIndex(imageSource, 0, NULL);
|
|
|
|
if (imageRef) {
|
2015-10-08 18:32:11 +00:00
|
|
|
image = [UIImage imageWithCGImage:imageRef scale:scale orientation:UIImageOrientationUp];
|
2015-09-04 11:35:44 +00:00
|
|
|
CFRelease(imageRef);
|
|
|
|
}
|
|
|
|
CFRelease(imageSource);
|
|
|
|
}
|
2015-03-06 00:36:41 +00:00
|
|
|
|
2015-09-04 11:35:44 +00:00
|
|
|
completionHandler(nil, image);
|
|
|
|
return ^{};
|
2015-03-06 00:36:41 +00:00
|
|
|
}
|
|
|
|
|
2015-09-02 15:25:10 +00:00
|
|
|
@end
|