2015-09-02 08:25:10 -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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#import "RCTAssetsLibraryImageLoader.h"
|
|
|
|
|
|
|
|
#import <AssetsLibrary/AssetsLibrary.h>
|
|
|
|
#import <ImageIO/ImageIO.h>
|
2015-09-23 12:00:05 -07:00
|
|
|
#import <libkern/OSAtomic.h>
|
2015-09-02 08:25:10 -07:00
|
|
|
#import <UIKit/UIKit.h>
|
|
|
|
|
|
|
|
#import "RCTBridge.h"
|
|
|
|
#import "RCTConvert.h"
|
|
|
|
#import "RCTImageLoader.h"
|
|
|
|
#import "RCTImageUtils.h"
|
|
|
|
#import "RCTUtils.h"
|
|
|
|
|
|
|
|
static dispatch_queue_t RCTAssetsLibraryImageLoaderQueue(void);
|
|
|
|
|
|
|
|
@implementation RCTAssetsLibraryImageLoader
|
|
|
|
{
|
|
|
|
ALAssetsLibrary *_assetsLibrary;
|
|
|
|
}
|
|
|
|
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
|
|
|
|
@synthesize bridge = _bridge;
|
|
|
|
|
|
|
|
- (ALAssetsLibrary *)assetsLibrary
|
|
|
|
{
|
|
|
|
return _assetsLibrary ?: (_assetsLibrary = [ALAssetsLibrary new]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark - RCTImageLoader
|
|
|
|
|
|
|
|
- (BOOL)canLoadImageURL:(NSURL *)requestURL
|
|
|
|
{
|
2015-10-20 05:00:50 -07:00
|
|
|
return [requestURL.scheme caseInsensitiveCompare:@"assets-library"] == NSOrderedSame;
|
2015-09-02 08:25:10 -07:00
|
|
|
}
|
|
|
|
|
2015-10-20 05:00:50 -07:00
|
|
|
- (RCTImageLoaderCancellationBlock)loadImageForURL:(NSURL *)imageURL
|
|
|
|
size:(CGSize)size
|
|
|
|
scale:(CGFloat)scale
|
|
|
|
resizeMode:(UIViewContentMode)resizeMode
|
|
|
|
progressHandler:(RCTImageLoaderProgressBlock)progressHandler
|
|
|
|
completionHandler:(RCTImageLoaderCompletionBlock)completionHandler
|
2015-09-02 08:25:10 -07:00
|
|
|
{
|
2015-09-23 12:00:05 -07:00
|
|
|
__block volatile uint32_t cancelled = 0;
|
|
|
|
|
2015-09-02 08:25:10 -07:00
|
|
|
[[self assetsLibrary] assetForURL:imageURL resultBlock:^(ALAsset *asset) {
|
2015-09-23 12:00:05 -07:00
|
|
|
if (cancelled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-09-02 08:25:10 -07:00
|
|
|
if (asset) {
|
|
|
|
// ALAssetLibrary API is async and will be multi-threaded. Loading a few full
|
|
|
|
// resolution images at once will spike the memory up to store the image data,
|
|
|
|
// and might trigger memory warnings and/or OOM crashes.
|
|
|
|
// To improve this, process the loaded asset in a serial queue.
|
|
|
|
dispatch_async(RCTAssetsLibraryImageLoaderQueue(), ^{
|
2015-09-23 12:00:05 -07:00
|
|
|
if (cancelled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-09-02 08:25:10 -07:00
|
|
|
// Also make sure the image is released immediately after it's used so it
|
|
|
|
// doesn't spike the memory up during the process.
|
|
|
|
@autoreleasepool {
|
|
|
|
BOOL useMaximumSize = CGSizeEqualToSize(size, CGSizeZero);
|
|
|
|
ALAssetRepresentation *representation = [asset defaultRepresentation];
|
|
|
|
|
2015-10-20 05:00:50 -07:00
|
|
|
#if RCT_DEV
|
|
|
|
|
2015-09-25 02:26:25 -07:00
|
|
|
CGSize sizeBeingLoaded = size;
|
|
|
|
if (useMaximumSize) {
|
|
|
|
CGSize pointSize = representation.dimensions;
|
|
|
|
sizeBeingLoaded = CGSizeMake(pointSize.width * representation.scale, pointSize.height * representation.scale);
|
|
|
|
}
|
|
|
|
|
2015-09-28 16:30:58 -07:00
|
|
|
CGSize screenSize;
|
|
|
|
if ([[[UIDevice currentDevice] systemVersion] compare:@"8.0" options:NSNumericSearch] == NSOrderedDescending) {
|
2015-10-20 05:00:50 -07:00
|
|
|
screenSize = [UIScreen mainScreen].nativeBounds.size;
|
2015-09-28 16:30:58 -07:00
|
|
|
} else {
|
|
|
|
CGSize mainScreenSize = [UIScreen mainScreen].bounds.size;
|
|
|
|
CGFloat mainScreenScale = [[UIScreen mainScreen] scale];
|
|
|
|
screenSize = CGSizeMake(mainScreenSize.width * mainScreenScale, mainScreenSize.height * mainScreenScale);
|
|
|
|
}
|
2015-09-25 02:26:25 -07:00
|
|
|
CGFloat maximumPixelDimension = fmax(screenSize.width, screenSize.height);
|
|
|
|
|
|
|
|
if (sizeBeingLoaded.width > maximumPixelDimension || sizeBeingLoaded.height > maximumPixelDimension) {
|
2015-10-20 05:00:50 -07:00
|
|
|
RCTLogInfo(@"[PERF ASSETS] Loading %@ at size %@, which is larger than screen size %@",
|
|
|
|
representation.filename, NSStringFromCGSize(sizeBeingLoaded), NSStringFromCGSize(screenSize));
|
2015-09-25 02:26:25 -07:00
|
|
|
}
|
2015-10-20 05:00:50 -07:00
|
|
|
|
|
|
|
#endif
|
2015-09-25 02:26:25 -07:00
|
|
|
|
2015-11-10 05:03:07 -08:00
|
|
|
UIImage *image = nil;
|
2015-09-02 08:25:10 -07:00
|
|
|
NSError *error = nil;
|
|
|
|
if (useMaximumSize) {
|
2015-11-10 05:03:07 -08:00
|
|
|
|
2015-09-02 08:25:10 -07:00
|
|
|
image = [UIImage imageWithCGImage:representation.fullResolutionImage
|
|
|
|
scale:scale
|
|
|
|
orientation:(UIImageOrientation)representation.orientation];
|
|
|
|
} else {
|
2015-11-10 05:03:07 -08:00
|
|
|
|
|
|
|
NSUInteger length = (NSUInteger)representation.size;
|
|
|
|
uint8_t *buffer = (uint8_t *)malloc((size_t)length);
|
|
|
|
if ([representation getBytes:buffer
|
|
|
|
fromOffset:0
|
|
|
|
length:length
|
|
|
|
error:&error]) {
|
|
|
|
|
|
|
|
NSData *data = [[NSData alloc] initWithBytesNoCopy:buffer
|
|
|
|
length:length
|
|
|
|
freeWhenDone:YES];
|
|
|
|
|
|
|
|
image = RCTDecodeImageWithData(data, size, scale, resizeMode);
|
|
|
|
} else {
|
|
|
|
free(buffer);
|
|
|
|
}
|
2015-09-02 08:25:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
completionHandler(error, image);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
NSString *errorText = [NSString stringWithFormat:@"Failed to load asset at URL %@ with no error message.", imageURL];
|
2015-10-20 05:00:50 -07:00
|
|
|
completionHandler(RCTErrorWithMessage(errorText), nil);
|
2015-09-02 08:25:10 -07:00
|
|
|
}
|
|
|
|
} failureBlock:^(NSError *loadError) {
|
2015-09-23 12:00:05 -07:00
|
|
|
if (cancelled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-09-02 08:25:10 -07:00
|
|
|
NSString *errorText = [NSString stringWithFormat:@"Failed to load asset at URL %@.\niOS Error: %@", imageURL, loadError];
|
2015-10-20 05:00:50 -07:00
|
|
|
completionHandler(RCTErrorWithMessage(errorText), nil);
|
2015-09-02 08:25:10 -07:00
|
|
|
}];
|
|
|
|
|
2015-09-23 12:00:05 -07:00
|
|
|
return ^{
|
|
|
|
OSAtomicOr32Barrier(1, &cancelled);
|
|
|
|
};
|
2015-09-02 08:25:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation RCTBridge (RCTAssetsLibraryImageLoader)
|
|
|
|
|
|
|
|
- (ALAssetsLibrary *)assetsLibrary
|
|
|
|
{
|
2015-10-20 05:00:50 -07:00
|
|
|
return [self.modules[RCTBridgeModuleNameForClass([RCTAssetsLibraryImageLoader class])] assetsLibrary];
|
2015-09-02 08:25:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
static dispatch_queue_t RCTAssetsLibraryImageLoaderQueue(void)
|
|
|
|
{
|
|
|
|
static dispatch_queue_t queue;
|
|
|
|
static dispatch_once_t onceToken;
|
|
|
|
dispatch_once(&onceToken, ^{
|
|
|
|
queue = dispatch_queue_create("com.facebook.RCTAssetsLibraryImageLoader", DISPATCH_QUEUE_SERIAL);
|
|
|
|
});
|
|
|
|
|
|
|
|
return queue;
|
|
|
|
}
|