167 lines
4.6 KiB
Objective-C
167 lines
4.6 KiB
Objective-C
/**
|
|
* 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.
|
|
*/
|
|
|
|
#import "RCTImageDownloader.h"
|
|
|
|
#import "RCTCache.h"
|
|
#import "RCTUtils.h"
|
|
|
|
typedef void (^RCTCachedDataDownloadBlock)(BOOL cached, NSData *data, NSError *error);
|
|
|
|
@implementation RCTImageDownloader
|
|
{
|
|
RCTCache *_cache;
|
|
dispatch_queue_t _processingQueue;
|
|
NSMutableDictionary *_pendingBlocks;
|
|
}
|
|
|
|
+ (instancetype)sharedInstance
|
|
{
|
|
static RCTImageDownloader *sharedInstance;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
sharedInstance = [[self alloc] init];
|
|
});
|
|
return sharedInstance;
|
|
}
|
|
|
|
- (instancetype)init
|
|
{
|
|
if ((self = [super init])) {
|
|
_cache = [[RCTCache alloc] initWithName:@"RCTImageDownloader"];
|
|
_processingQueue = dispatch_queue_create("com.facebook.React.DownloadProcessingQueue", DISPATCH_QUEUE_SERIAL);
|
|
_pendingBlocks = [[NSMutableDictionary alloc] init];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
static NSString *RCTCacheKeyForURL(NSURL *url)
|
|
{
|
|
return url.absoluteString;
|
|
}
|
|
|
|
- (id)_downloadDataForURL:(NSURL *)url block:(RCTCachedDataDownloadBlock)block
|
|
{
|
|
NSString *cacheKey = RCTCacheKeyForURL(url);
|
|
|
|
__block BOOL cancelled = NO;
|
|
__block NSURLSessionDataTask *task = nil;
|
|
|
|
dispatch_block_t cancel = ^{
|
|
|
|
cancelled = YES;
|
|
|
|
dispatch_async(_processingQueue, ^{
|
|
NSMutableArray *pendingBlocks = self->_pendingBlocks[cacheKey];
|
|
[pendingBlocks removeObject:block];
|
|
});
|
|
|
|
if (task) {
|
|
[task cancel];
|
|
task = nil;
|
|
}
|
|
};
|
|
|
|
dispatch_async(_processingQueue, ^{
|
|
NSMutableArray *pendingBlocks = _pendingBlocks[cacheKey];
|
|
if (pendingBlocks) {
|
|
[pendingBlocks addObject:block];
|
|
} else {
|
|
_pendingBlocks[cacheKey] = [NSMutableArray arrayWithObject:block];
|
|
|
|
__weak RCTImageDownloader *weakSelf = self;
|
|
RCTCachedDataDownloadBlock runBlocks = ^(BOOL cached, NSData *data, NSError *error) {
|
|
dispatch_async(_processingQueue, ^{
|
|
RCTImageDownloader *strongSelf = weakSelf;
|
|
NSArray *blocks = strongSelf->_pendingBlocks[cacheKey];
|
|
[strongSelf->_pendingBlocks removeObjectForKey:cacheKey];
|
|
for (RCTCachedDataDownloadBlock block in blocks) {
|
|
block(cached, data, error);
|
|
}
|
|
});
|
|
};
|
|
|
|
if ([_cache hasDataForKey:cacheKey]) {
|
|
[_cache fetchDataForKey:cacheKey completionHandler:^(NSData *data) {
|
|
if (!cancelled) {
|
|
runBlocks(YES, data, nil);
|
|
}
|
|
}];
|
|
} else {
|
|
task = [[NSURLSession sharedSession] dataTaskWithURL:url completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
|
|
if (!cancelled) {
|
|
runBlocks(NO, data, error);
|
|
}
|
|
}];
|
|
|
|
[task resume];
|
|
}
|
|
}
|
|
});
|
|
|
|
return [cancel copy];
|
|
}
|
|
|
|
- (id)downloadDataForURL:(NSURL *)url block:(RCTDataDownloadBlock)block
|
|
{
|
|
NSString *cacheKey = RCTCacheKeyForURL(url);
|
|
__weak RCTImageDownloader *weakSelf = self;
|
|
return [self _downloadDataForURL:url block:^(BOOL cached, NSData *data, NSError *error) {
|
|
if (!cached) {
|
|
RCTImageDownloader *strongSelf = weakSelf;
|
|
[strongSelf->_cache setData:data forKey:cacheKey];
|
|
}
|
|
block(data, error);
|
|
}];
|
|
}
|
|
|
|
- (id)downloadImageForURL:(NSURL *)url size:(CGSize)size
|
|
scale:(CGFloat)scale block:(RCTImageDownloadBlock)block
|
|
{
|
|
return [self downloadDataForURL:url block:^(NSData *data, NSError *error) {
|
|
|
|
UIImage *image = [UIImage imageWithData:data scale:scale];
|
|
if (image) {
|
|
|
|
// Resize (TODO: should we take aspect ratio into account?)
|
|
CGSize imageSize = size;
|
|
if (CGSizeEqualToSize(imageSize, CGSizeZero)) {
|
|
imageSize = image.size;
|
|
} else {
|
|
imageSize = (CGSize){
|
|
MIN(size.width, image.size.width),
|
|
MIN(size.height, image.size.height)
|
|
};
|
|
}
|
|
|
|
// Rescale image if required size is smaller
|
|
CGFloat imageScale = scale;
|
|
if (imageScale == 0 || imageScale < image.scale) {
|
|
imageScale = image.scale;
|
|
}
|
|
|
|
// Decompress image at required size
|
|
UIGraphicsBeginImageContextWithOptions(imageSize, NO, imageScale);
|
|
[image drawInRect:(CGRect){{0, 0}, imageSize}];
|
|
image = UIGraphicsGetImageFromCurrentImageContext();
|
|
UIGraphicsEndImageContext();
|
|
}
|
|
block(image, nil);
|
|
}];
|
|
}
|
|
|
|
- (void)cancelDownload:(id)downloadToken
|
|
{
|
|
if (downloadToken) {
|
|
((dispatch_block_t)downloadToken)();
|
|
}
|
|
}
|
|
|
|
@end
|