2015-03-23 22:07:33 +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-01-30 01:10:49 +00:00
|
|
|
|
|
|
|
#import "RCTImageDownloader.h"
|
2015-02-19 01:48:13 +00:00
|
|
|
|
2015-09-02 15:25:10 +00:00
|
|
|
#import "RCTImageLoader.h"
|
2015-07-14 11:06:17 +00:00
|
|
|
#import "RCTImageUtils.h"
|
2015-05-22 14:17:08 +00:00
|
|
|
#import "RCTLog.h"
|
2015-07-27 20:46:59 +00:00
|
|
|
#import "RCTNetworking.h"
|
2015-01-30 01:10:49 +00:00
|
|
|
#import "RCTUtils.h"
|
|
|
|
|
|
|
|
@implementation RCTImageDownloader
|
2015-02-19 01:48:13 +00:00
|
|
|
{
|
2015-06-29 12:15:25 +00:00
|
|
|
NSURLCache *_cache;
|
2015-03-31 03:12:32 +00:00
|
|
|
dispatch_queue_t _processingQueue;
|
2015-02-19 01:48:13 +00:00
|
|
|
}
|
2015-01-30 01:10:49 +00:00
|
|
|
|
2015-07-27 20:46:59 +00:00
|
|
|
@synthesize bridge = _bridge;
|
|
|
|
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
|
2015-02-19 01:48:13 +00:00
|
|
|
- (instancetype)init
|
|
|
|
{
|
|
|
|
if ((self = [super init])) {
|
2015-06-29 12:15:25 +00:00
|
|
|
_cache = [[NSURLCache alloc] initWithMemoryCapacity:5 * 1024 * 1024 diskCapacity:200 * 1024 * 1024 diskPath:@"React/RCTImageDownloader"];
|
2015-03-31 03:12:32 +00:00
|
|
|
_processingQueue = dispatch_queue_create("com.facebook.React.DownloadProcessingQueue", DISPATCH_QUEUE_SERIAL);
|
2015-02-19 01:48:13 +00:00
|
|
|
}
|
2015-06-29 12:15:25 +00:00
|
|
|
return self;
|
2015-02-19 01:48:13 +00:00
|
|
|
}
|
|
|
|
|
2015-09-02 15:25:10 +00:00
|
|
|
- (BOOL)canLoadImageURL:(NSURL *)requestURL
|
|
|
|
{
|
2015-10-12 11:14:21 +00:00
|
|
|
static NSSet *schemes = nil;
|
|
|
|
static dispatch_once_t onceToken;
|
|
|
|
dispatch_once(&onceToken, ^{
|
2015-10-13 09:42:16 +00:00
|
|
|
schemes = [[NSSet alloc] initWithObjects:@"http", @"https", @"file", @"data", nil];
|
2015-10-12 11:14:21 +00:00
|
|
|
});
|
|
|
|
return [schemes containsObject:requestURL.scheme.lowercaseString];
|
2015-09-02 15:25:10 +00:00
|
|
|
}
|
|
|
|
|
2015-07-27 20:46:59 +00:00
|
|
|
/**
|
|
|
|
* Downloads a block of raw data and returns it. Note that the callback block
|
|
|
|
* will not be executed on the same thread you called the method from, nor on
|
|
|
|
* the main thread. Returns a token that can be used to cancel the download.
|
|
|
|
*/
|
|
|
|
- (RCTImageLoaderCancellationBlock)downloadDataForURL:(NSURL *)url
|
2015-09-02 15:25:10 +00:00
|
|
|
progressHandler:(RCTImageLoaderProgressBlock)progressBlock
|
2015-09-04 11:35:44 +00:00
|
|
|
completionHandler:(void (^)(NSError *error, NSData *data))completionBlock
|
2015-02-19 01:48:13 +00:00
|
|
|
{
|
2015-07-27 20:46:59 +00:00
|
|
|
if (![_bridge respondsToSelector:NSSelectorFromString(@"networking")]) {
|
|
|
|
RCTLogError(@"You need to import the RCTNetworking library in order to download remote images.");
|
|
|
|
return ^{};
|
|
|
|
}
|
2015-02-19 01:48:13 +00:00
|
|
|
|
2015-07-27 20:46:59 +00:00
|
|
|
__weak RCTImageDownloader *weakSelf = self;
|
|
|
|
RCTURLRequestCompletionBlock runBlocks = ^(NSURLResponse *response, NSData *data, NSError *error) {
|
2015-07-09 16:48:22 +00:00
|
|
|
|
2015-07-27 20:46:59 +00:00
|
|
|
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-02-19 01:51:14 +00:00
|
|
|
|
2015-03-31 03:12:32 +00:00
|
|
|
dispatch_async(_processingQueue, ^{
|
2015-07-27 20:46:59 +00:00
|
|
|
completionBlock(error, data);
|
2015-03-31 03:12:32 +00:00
|
|
|
});
|
2015-02-19 01:48:13 +00:00
|
|
|
};
|
|
|
|
|
2015-07-27 20:46:59 +00:00
|
|
|
NSURLRequest *request = [NSURLRequest requestWithURL:url];
|
|
|
|
{
|
|
|
|
NSCachedURLResponse *cachedResponse = [_cache cachedResponseForRequest:request];
|
|
|
|
if (cachedResponse) {
|
|
|
|
runBlocks(cachedResponse.response, cachedResponse.data, nil);
|
|
|
|
return ^{};
|
2015-02-19 01:51:14 +00:00
|
|
|
}
|
2015-07-27 20:46:59 +00:00
|
|
|
}
|
2015-02-19 01:48:13 +00:00
|
|
|
|
2015-07-27 20:46:59 +00:00
|
|
|
RCTDownloadTask *task = [_bridge.networking downloadTaskWithRequest:request completionBlock:^(NSURLResponse *response, NSData *data, NSError *error) {
|
2015-10-13 15:12:54 +00:00
|
|
|
if (response && !error && [response.URL.scheme hasPrefix:@"http"]) {
|
2015-07-27 20:46:59 +00:00
|
|
|
RCTImageDownloader *strongSelf = weakSelf;
|
|
|
|
NSCachedURLResponse *cachedResponse = [[NSCachedURLResponse alloc] initWithResponse:response data:data userInfo:nil storagePolicy:NSURLCacheStorageAllowed];
|
|
|
|
[strongSelf->_cache storeCachedResponse:cachedResponse forRequest:request];
|
|
|
|
}
|
|
|
|
runBlocks(response, data, error);
|
2015-02-04 00:15:20 +00:00
|
|
|
}];
|
2015-07-27 20:46:59 +00:00
|
|
|
if (progressBlock) {
|
|
|
|
task.downloadProgressBlock = progressBlock;
|
|
|
|
}
|
|
|
|
return ^{ [task cancel]; };
|
2015-02-04 00:15:20 +00:00
|
|
|
}
|
|
|
|
|
2015-09-02 15:25:10 +00:00
|
|
|
- (RCTImageLoaderCancellationBlock)loadImageForURL:(NSURL *)imageURL
|
|
|
|
size:(CGSize)size
|
|
|
|
scale:(CGFloat)scale
|
|
|
|
resizeMode:(UIViewContentMode)resizeMode
|
|
|
|
progressHandler:(RCTImageLoaderProgressBlock)progressHandler
|
|
|
|
completionHandler:(RCTImageLoaderCompletionBlock)completionHandler
|
2015-06-29 12:15:25 +00:00
|
|
|
{
|
2015-10-13 15:12:54 +00:00
|
|
|
if ([imageURL.scheme.lowercaseString hasPrefix:@"http"] ||
|
|
|
|
[imageURL.scheme caseInsensitiveCompare:@"file"] == NSOrderedSame) {
|
2015-09-02 23:28:04 +00:00
|
|
|
__block RCTImageLoaderCancellationBlock decodeCancel = nil;
|
|
|
|
|
2015-10-13 15:12:54 +00:00
|
|
|
// Add missing png extension
|
|
|
|
if (imageURL.fileURL && imageURL.pathExtension.length == 0) {
|
|
|
|
imageURL = [NSURL fileURLWithPath:[imageURL.path stringByAppendingPathExtension:@"png"]];
|
|
|
|
}
|
|
|
|
|
2015-09-02 23:28:04 +00:00
|
|
|
__weak RCTImageDownloader *weakSelf = self;
|
|
|
|
RCTImageLoaderCancellationBlock downloadCancel = [self downloadDataForURL:imageURL progressHandler:progressHandler completionHandler:^(NSError *error, NSData *imageData) {
|
|
|
|
if (error) {
|
|
|
|
completionHandler(error, nil);
|
|
|
|
} else {
|
|
|
|
decodeCancel = [weakSelf.bridge.imageLoader decodeImageData:imageData size:size scale:scale resizeMode:resizeMode completionBlock:completionHandler];
|
|
|
|
}
|
|
|
|
}];
|
2015-07-27 20:46:59 +00:00
|
|
|
|
2015-09-02 23:28:04 +00:00
|
|
|
return ^{
|
|
|
|
downloadCancel();
|
2015-06-29 12:15:25 +00:00
|
|
|
|
2015-09-02 23:28:04 +00:00
|
|
|
if (decodeCancel) {
|
|
|
|
decodeCancel();
|
|
|
|
}
|
|
|
|
};
|
2015-09-03 12:53:16 +00:00
|
|
|
} else if ([imageURL.scheme caseInsensitiveCompare:@"data"] == NSOrderedSame) {
|
|
|
|
__block BOOL cancelled = NO;
|
|
|
|
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
|
|
|
|
if (cancelled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Normally -dataWithContentsOfURL: would be bad but this is a data URL.
|
|
|
|
NSData *data = [NSData dataWithContentsOfURL:imageURL];
|
|
|
|
|
2015-10-08 18:32:11 +00:00
|
|
|
UIImage *image = [UIImage imageWithData:data scale:scale];
|
2015-09-03 12:53:16 +00:00
|
|
|
if (image) {
|
|
|
|
if (progressHandler) {
|
|
|
|
progressHandler(1, 1);
|
|
|
|
}
|
|
|
|
if (completionHandler) {
|
|
|
|
completionHandler(nil, image);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (completionHandler) {
|
|
|
|
NSString *message = [NSString stringWithFormat:@"Invalid image data for URL: %@", imageURL];
|
|
|
|
completionHandler(RCTErrorWithMessage(message), nil);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return ^{
|
|
|
|
cancelled = YES;
|
|
|
|
};
|
2015-09-02 23:28:04 +00:00
|
|
|
} else {
|
|
|
|
RCTLogError(@"Unexpected image schema %@", imageURL.scheme);
|
|
|
|
return ^{};
|
|
|
|
}
|
2015-06-29 12:15:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
2015-07-27 20:46:59 +00:00
|
|
|
|
|
|
|
@implementation RCTBridge (RCTImageDownloader)
|
|
|
|
|
|
|
|
- (RCTImageDownloader *)imageDownloader
|
|
|
|
{
|
|
|
|
return self.modules[RCTBridgeModuleNameForClass([RCTImageDownloader class])];
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|