2015-03-23 15:07:33 -07: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-01-29 17:10:49 -08:00
|
|
|
|
|
|
|
#import "RCTImageDownloader.h"
|
2015-02-18 17:48:13 -08:00
|
|
|
|
2015-07-09 15:48:22 -01:00
|
|
|
#import "RCTDownloadTaskWrapper.h"
|
2015-07-14 04:06:17 -07:00
|
|
|
#import "RCTImageUtils.h"
|
2015-05-22 07:17:08 -07:00
|
|
|
#import "RCTLog.h"
|
2015-01-29 17:10:49 -08:00
|
|
|
#import "RCTUtils.h"
|
|
|
|
|
2015-07-14 13:15:24 -07:00
|
|
|
typedef void (^RCTCachedDataDownloadBlock)(BOOL cached, NSURLResponse *response,
|
|
|
|
NSData *data, NSError *error);
|
2015-02-18 17:51:14 -08:00
|
|
|
|
2015-06-29 05:15:25 -07:00
|
|
|
CGSize RCTTargetSizeForClipRect(CGRect);
|
|
|
|
CGRect RCTClipRect(CGSize, CGFloat, CGSize, CGFloat, UIViewContentMode);
|
|
|
|
|
2015-01-29 17:10:49 -08:00
|
|
|
@implementation RCTImageDownloader
|
2015-02-18 17:48:13 -08:00
|
|
|
{
|
2015-06-29 05:15:25 -07:00
|
|
|
NSURLCache *_cache;
|
2015-03-30 20:12:32 -07:00
|
|
|
dispatch_queue_t _processingQueue;
|
2015-02-18 17:51:14 -08:00
|
|
|
NSMutableDictionary *_pendingBlocks;
|
2015-07-10 09:11:31 -01:00
|
|
|
RCTDownloadTaskWrapper *_downloadTaskWrapper;
|
2015-02-18 17:48:13 -08:00
|
|
|
}
|
2015-01-29 17:10:49 -08:00
|
|
|
|
2015-06-29 05:15:25 -07:00
|
|
|
+ (RCTImageDownloader *)sharedInstance
|
2015-01-29 17:10:49 -08:00
|
|
|
{
|
|
|
|
static RCTImageDownloader *sharedInstance;
|
|
|
|
static dispatch_once_t onceToken;
|
|
|
|
dispatch_once(&onceToken, ^{
|
2015-06-29 05:15:25 -07:00
|
|
|
sharedInstance = [[RCTImageDownloader alloc] init];
|
2015-01-29 17:10:49 -08:00
|
|
|
});
|
|
|
|
return sharedInstance;
|
|
|
|
}
|
|
|
|
|
2015-02-18 17:48:13 -08:00
|
|
|
- (instancetype)init
|
|
|
|
{
|
|
|
|
if ((self = [super init])) {
|
2015-06-29 05:15:25 -07:00
|
|
|
_cache = [[NSURLCache alloc] initWithMemoryCapacity:5 * 1024 * 1024 diskCapacity:200 * 1024 * 1024 diskPath:@"React/RCTImageDownloader"];
|
2015-03-30 20:12:32 -07:00
|
|
|
_processingQueue = dispatch_queue_create("com.facebook.React.DownloadProcessingQueue", DISPATCH_QUEUE_SERIAL);
|
|
|
|
_pendingBlocks = [[NSMutableDictionary alloc] init];
|
2015-07-10 09:11:31 -01:00
|
|
|
|
|
|
|
NSURLSessionConfiguration *config = [NSURLSessionConfiguration defaultSessionConfiguration];
|
|
|
|
_downloadTaskWrapper = [[RCTDownloadTaskWrapper alloc] initWithSessionConfiguration:config delegateQueue:nil];
|
2015-02-18 17:48:13 -08:00
|
|
|
}
|
|
|
|
|
2015-06-29 05:15:25 -07:00
|
|
|
return self;
|
2015-02-18 17:48:13 -08:00
|
|
|
}
|
|
|
|
|
2015-07-10 09:11:31 -01:00
|
|
|
- (RCTImageDownloadCancellationBlock)_downloadDataForURL:(NSURL *)url progressBlock:progressBlock block:(RCTCachedDataDownloadBlock)block
|
2015-02-18 17:48:13 -08:00
|
|
|
{
|
2015-07-10 09:11:31 -01:00
|
|
|
NSString *const cacheKey = url.absoluteString;
|
2015-02-18 17:48:13 -08:00
|
|
|
|
|
|
|
__block BOOL cancelled = NO;
|
2015-07-09 15:48:22 -01:00
|
|
|
__block NSURLSessionDownloadTask *task = nil;
|
|
|
|
|
2015-07-10 09:11:31 -01:00
|
|
|
RCTImageDownloadCancellationBlock cancel = ^{
|
2015-02-18 17:48:13 -08:00
|
|
|
cancelled = YES;
|
2015-02-18 17:51:14 -08:00
|
|
|
|
2015-03-30 20:12:32 -07:00
|
|
|
dispatch_async(_processingQueue, ^{
|
|
|
|
NSMutableArray *pendingBlocks = self->_pendingBlocks[cacheKey];
|
|
|
|
[pendingBlocks removeObject:block];
|
|
|
|
});
|
2015-02-18 17:51:14 -08:00
|
|
|
|
2015-02-18 17:48:13 -08:00
|
|
|
if (task) {
|
|
|
|
[task cancel];
|
|
|
|
task = nil;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-03-30 20:12:32 -07:00
|
|
|
dispatch_async(_processingQueue, ^{
|
|
|
|
NSMutableArray *pendingBlocks = _pendingBlocks[cacheKey];
|
|
|
|
if (pendingBlocks) {
|
|
|
|
[pendingBlocks addObject:block];
|
2015-02-18 17:51:14 -08:00
|
|
|
} else {
|
2015-03-30 20:12:32 -07:00
|
|
|
_pendingBlocks[cacheKey] = [NSMutableArray arrayWithObject:block];
|
|
|
|
|
|
|
|
__weak RCTImageDownloader *weakSelf = self;
|
2015-07-14 13:15:24 -07:00
|
|
|
RCTCachedDataDownloadBlock runBlocks = ^(BOOL cached, NSURLResponse *response, NSData *data, NSError *error) {
|
|
|
|
|
|
|
|
if (!error && [response isKindOfClass:[NSHTTPURLResponse class]]) {
|
|
|
|
NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse *)response;
|
|
|
|
if (httpResponse.statusCode != 200) {
|
|
|
|
data = nil;
|
|
|
|
error = [[NSError alloc] initWithDomain:NSURLErrorDomain
|
|
|
|
code:httpResponse.statusCode
|
|
|
|
userInfo:nil];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-07 18:26:09 -07:00
|
|
|
dispatch_async(_processingQueue, ^{
|
|
|
|
RCTImageDownloader *strongSelf = weakSelf;
|
|
|
|
NSArray *blocks = strongSelf->_pendingBlocks[cacheKey];
|
|
|
|
[strongSelf->_pendingBlocks removeObjectForKey:cacheKey];
|
2015-06-09 12:25:24 -07:00
|
|
|
for (RCTCachedDataDownloadBlock downloadBlock in blocks) {
|
2015-07-14 13:15:24 -07:00
|
|
|
downloadBlock(cached, response, data, error);
|
2015-04-07 18:26:09 -07:00
|
|
|
}
|
|
|
|
});
|
2015-03-30 20:12:32 -07:00
|
|
|
};
|
|
|
|
|
2015-07-09 15:48:22 -01:00
|
|
|
NSURLRequest *request = [NSURLRequest requestWithURL:url];
|
2015-07-10 09:11:31 -01:00
|
|
|
task = [_downloadTaskWrapper downloadData:url progressBlock:progressBlock completionBlock:^(NSURLResponse *response, NSData *data, NSError *error) {
|
2015-07-14 13:15:24 -07:00
|
|
|
|
2015-06-29 05:15:25 -07:00
|
|
|
if (!cancelled) {
|
2015-07-14 13:15:24 -07:00
|
|
|
runBlocks(NO, response, data, error);
|
2015-06-29 05:15:25 -07:00
|
|
|
}
|
2015-02-18 17:51:14 -08:00
|
|
|
|
2015-07-14 13:56:55 -07:00
|
|
|
if (response && !error) {
|
2015-07-03 02:15:29 -07:00
|
|
|
RCTImageDownloader *strongSelf = weakSelf;
|
|
|
|
NSCachedURLResponse *cachedResponse = [[NSCachedURLResponse alloc] initWithResponse:response data:data userInfo:nil storagePolicy:NSURLCacheStorageAllowed];
|
2015-07-09 15:48:22 -01:00
|
|
|
[strongSelf->_cache storeCachedResponse:cachedResponse forRequest:request];
|
2015-07-03 02:15:29 -07:00
|
|
|
}
|
2015-06-29 05:15:25 -07:00
|
|
|
task = nil;
|
|
|
|
}];
|
|
|
|
|
2015-07-09 15:48:22 -01:00
|
|
|
NSCachedURLResponse *cachedResponse = [_cache cachedResponseForRequest:request];
|
2015-07-13 08:42:32 -07:00
|
|
|
if (cancelled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cachedResponse) {
|
2015-07-14 13:15:24 -07:00
|
|
|
runBlocks(YES, cachedResponse.response, cachedResponse.data, nil);
|
2015-07-13 08:42:32 -07:00
|
|
|
} else {
|
|
|
|
[task resume];
|
|
|
|
}
|
2015-02-18 17:51:14 -08:00
|
|
|
}
|
2015-03-30 20:12:32 -07:00
|
|
|
});
|
2015-02-18 17:48:13 -08:00
|
|
|
|
|
|
|
return [cancel copy];
|
|
|
|
}
|
|
|
|
|
2015-07-10 09:11:31 -01:00
|
|
|
- (RCTImageDownloadCancellationBlock)downloadDataForURL:(NSURL *)url progressBlock:(RCTDataProgressBlock)progressBlock block:(RCTDataDownloadBlock)block
|
2015-02-03 16:15:20 -08:00
|
|
|
{
|
2015-07-14 13:15:24 -07:00
|
|
|
return [self _downloadDataForURL:url progressBlock:progressBlock block:^(BOOL cached, NSURLResponse *response, NSData *data, NSError *error) {
|
2015-03-30 20:12:32 -07:00
|
|
|
block(data, error);
|
2015-02-03 16:15:20 -08:00
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2015-07-10 09:11:31 -01:00
|
|
|
- (RCTImageDownloadCancellationBlock)downloadImageForURL:(NSURL *)url
|
|
|
|
size:(CGSize)size
|
|
|
|
scale:(CGFloat)scale
|
|
|
|
resizeMode:(UIViewContentMode)resizeMode
|
2015-07-13 10:30:34 -07:00
|
|
|
tintColor:(UIColor *)tintColor
|
2015-07-10 09:11:31 -01:00
|
|
|
backgroundColor:(UIColor *)backgroundColor
|
|
|
|
progressBlock:(RCTDataProgressBlock)progressBlock
|
|
|
|
block:(RCTImageDownloadBlock)block
|
2015-06-29 05:15:25 -07:00
|
|
|
{
|
2015-07-09 15:48:22 -01:00
|
|
|
return [self downloadDataForURL:url progressBlock:progressBlock block:^(NSData *data, NSError *error) {
|
2015-06-29 05:15:25 -07:00
|
|
|
if (!data || error) {
|
|
|
|
block(nil, error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CGSizeEqualToSize(size, CGSizeZero)) {
|
|
|
|
// Target size wasn't available yet, so abort image drawing
|
|
|
|
block(nil, nil);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
UIImage *image = [UIImage imageWithData:data scale:scale];
|
|
|
|
if (image) {
|
|
|
|
|
|
|
|
// Get scale and size
|
|
|
|
CGFloat destScale = scale ?: RCTScreenScale();
|
|
|
|
CGRect imageRect = RCTClipRect(image.size, image.scale, size, destScale, resizeMode);
|
|
|
|
CGSize destSize = RCTTargetSizeForClipRect(imageRect);
|
|
|
|
|
|
|
|
// Opacity optimizations
|
|
|
|
UIColor *blendColor = nil;
|
|
|
|
BOOL opaque = !RCTImageHasAlpha(image.CGImage);
|
|
|
|
if (!opaque && backgroundColor) {
|
|
|
|
CGFloat alpha;
|
|
|
|
[backgroundColor getRed:NULL green:NULL blue:NULL alpha:&alpha];
|
|
|
|
if (alpha > 0.999) { // no benefit to blending if background is translucent
|
|
|
|
opaque = YES;
|
|
|
|
blendColor = backgroundColor;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Decompress image at required size
|
|
|
|
UIGraphicsBeginImageContextWithOptions(destSize, opaque, destScale);
|
|
|
|
if (blendColor) {
|
|
|
|
[blendColor setFill];
|
|
|
|
UIRectFill((CGRect){CGPointZero, destSize});
|
|
|
|
}
|
2015-07-13 10:30:34 -07:00
|
|
|
if (tintColor) {
|
|
|
|
image = [image imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate];
|
|
|
|
[tintColor setFill];
|
|
|
|
}
|
2015-06-29 05:15:25 -07:00
|
|
|
[image drawInRect:imageRect];
|
|
|
|
image = UIGraphicsGetImageFromCurrentImageContext();
|
|
|
|
UIGraphicsEndImageContext();
|
|
|
|
}
|
|
|
|
|
|
|
|
block(image, nil);
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2015-07-10 09:11:31 -01:00
|
|
|
- (void)cancelDownload:(RCTImageDownloadCancellationBlock)downloadToken
|
2015-06-29 05:15:25 -07:00
|
|
|
{
|
|
|
|
if (downloadToken) {
|
2015-07-10 09:11:31 -01:00
|
|
|
downloadToken();
|
2015-06-29 05:15:25 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|