mirror of
https://github.com/status-im/react-native.git
synced 2025-01-19 14:02:10 +00:00
1d6d1189f0
Summary: public The image loader was previously returning on the main thread, which could lead to poor performance due to various call sites doing further image processing (resizing, cropping, etc.) directly in the completion block. This diff modifies the loader to return on a background thread (the same one used to load the image), and updates the call sites to dispatch to the explicit thread they need. Reviewed By: javache Differential Revision: D2549774 fb-gh-sync-id: fed73b7c163fdf67ff65bae72ab1986327e75815
175 lines
6.8 KiB
Objective-C
175 lines
6.8 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 "RCTCameraRollManager.h"
|
|
|
|
#import <AssetsLibrary/AssetsLibrary.h>
|
|
#import <CoreLocation/CoreLocation.h>
|
|
#import <Foundation/Foundation.h>
|
|
#import <UIKit/UIKit.h>
|
|
|
|
#import "RCTAssetsLibraryImageLoader.h"
|
|
#import "RCTBridge.h"
|
|
#import "RCTImageLoader.h"
|
|
#import "RCTLog.h"
|
|
#import "RCTUtils.h"
|
|
|
|
@implementation RCTCameraRollManager
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
@synthesize bridge = _bridge;
|
|
|
|
RCT_EXPORT_METHOD(saveImageWithTag:(NSString *)imageTag
|
|
successCallback:(RCTResponseSenderBlock)successCallback
|
|
errorCallback:(RCTResponseErrorBlock)errorCallback)
|
|
{
|
|
[_bridge.imageLoader loadImageWithTag:imageTag callback:^(NSError *loadError, UIImage *loadedImage) {
|
|
if (loadError) {
|
|
errorCallback(loadError);
|
|
return;
|
|
}
|
|
// It's unclear if writeImageToSavedPhotosAlbum is thread-safe
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
[_bridge.assetsLibrary writeImageToSavedPhotosAlbum:loadedImage.CGImage metadata:nil completionBlock:^(NSURL *assetURL, NSError *saveError) {
|
|
if (saveError) {
|
|
RCTLogWarn(@"Error saving cropped image: %@", saveError);
|
|
errorCallback(saveError);
|
|
} else {
|
|
successCallback(@[assetURL.absoluteString]);
|
|
}
|
|
}];
|
|
});
|
|
}];
|
|
}
|
|
|
|
- (void)callCallback:(RCTResponseSenderBlock)callback withAssets:(NSArray *)assets hasNextPage:(BOOL)hasNextPage
|
|
{
|
|
if (!assets.count) {
|
|
callback(@[@{
|
|
@"edges": assets,
|
|
@"page_info": @{
|
|
@"has_next_page": @NO}
|
|
}]);
|
|
return;
|
|
}
|
|
callback(@[@{
|
|
@"edges": assets,
|
|
@"page_info": @{
|
|
@"start_cursor": assets[0][@"node"][@"image"][@"uri"],
|
|
@"end_cursor": assets[assets.count - 1][@"node"][@"image"][@"uri"],
|
|
@"has_next_page": @(hasNextPage)}
|
|
}]);
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(getPhotos:(NSDictionary *)params
|
|
callback:(RCTResponseSenderBlock)callback
|
|
errorCallback:(RCTResponseErrorBlock)errorCallback)
|
|
{
|
|
NSUInteger first = [params[@"first"] integerValue];
|
|
NSString *afterCursor = params[@"after"];
|
|
NSString *groupTypesStr = params[@"groupTypes"];
|
|
NSString *groupName = params[@"groupName"];
|
|
NSString *assetType = params[@"assetType"];
|
|
ALAssetsGroupType groupTypes;
|
|
|
|
if ([groupTypesStr isEqualToString:@"Album"]) {
|
|
groupTypes = ALAssetsGroupAlbum;
|
|
} else if ([groupTypesStr isEqualToString:@"All"]) {
|
|
groupTypes = ALAssetsGroupAll;
|
|
} else if ([groupTypesStr isEqualToString:@"Event"]) {
|
|
groupTypes = ALAssetsGroupEvent;
|
|
} else if ([groupTypesStr isEqualToString:@"Faces"]) {
|
|
groupTypes = ALAssetsGroupFaces;
|
|
} else if ([groupTypesStr isEqualToString:@"Library"]) {
|
|
groupTypes = ALAssetsGroupLibrary;
|
|
} else if ([groupTypesStr isEqualToString:@"PhotoStream"]) {
|
|
groupTypes = ALAssetsGroupPhotoStream;
|
|
} else {
|
|
groupTypes = ALAssetsGroupSavedPhotos;
|
|
}
|
|
|
|
BOOL __block foundAfter = NO;
|
|
BOOL __block hasNextPage = NO;
|
|
BOOL __block calledCallback = NO;
|
|
NSMutableArray *assets = [NSMutableArray new];
|
|
|
|
[_bridge.assetsLibrary enumerateGroupsWithTypes:groupTypes usingBlock:^(ALAssetsGroup *group, BOOL *stopGroups) {
|
|
if (group && (groupName == nil || [groupName isEqualToString:[group valueForProperty:ALAssetsGroupPropertyName]])) {
|
|
|
|
if (assetType == nil || [assetType isEqualToString:@"Photos"]) {
|
|
[group setAssetsFilter:ALAssetsFilter.allPhotos];
|
|
} else if ([assetType isEqualToString:@"Videos"]) {
|
|
[group setAssetsFilter:ALAssetsFilter.allVideos];
|
|
} else if ([assetType isEqualToString:@"All"]) {
|
|
[group setAssetsFilter:ALAssetsFilter.allAssets];
|
|
}
|
|
|
|
[group enumerateAssetsWithOptions:NSEnumerationReverse usingBlock:^(ALAsset *result, NSUInteger index, BOOL *stopAssets) {
|
|
if (result) {
|
|
NSString *uri = ((NSURL *)[result valueForProperty:ALAssetPropertyAssetURL]).absoluteString;
|
|
if (afterCursor && !foundAfter) {
|
|
if ([afterCursor isEqualToString:uri]) {
|
|
foundAfter = YES;
|
|
}
|
|
return; // Skip until we get to the first one
|
|
}
|
|
if (first == assets.count) {
|
|
*stopAssets = YES;
|
|
*stopGroups = YES;
|
|
hasNextPage = YES;
|
|
RCTAssert(calledCallback == NO, @"Called the callback before we finished processing the results.");
|
|
[self callCallback:callback withAssets:assets hasNextPage:hasNextPage];
|
|
calledCallback = YES;
|
|
return;
|
|
}
|
|
CGSize dimensions = [result defaultRepresentation].dimensions;
|
|
CLLocation *loc = [result valueForProperty:ALAssetPropertyLocation];
|
|
NSDate *date = [result valueForProperty:ALAssetPropertyDate];
|
|
[assets addObject:@{
|
|
@"node": @{
|
|
@"type": [result valueForProperty:ALAssetPropertyType],
|
|
@"group_name": [group valueForProperty:ALAssetsGroupPropertyName],
|
|
@"image": @{
|
|
@"uri": uri,
|
|
@"height": @(dimensions.height),
|
|
@"width": @(dimensions.width),
|
|
@"isStored": @YES,
|
|
},
|
|
@"timestamp": @(date.timeIntervalSince1970),
|
|
@"location": loc ?
|
|
@{
|
|
@"latitude": @(loc.coordinate.latitude),
|
|
@"longitude": @(loc.coordinate.longitude),
|
|
@"altitude": @(loc.altitude),
|
|
@"heading": @(loc.course),
|
|
@"speed": @(loc.speed),
|
|
} : @{},
|
|
}
|
|
}];
|
|
}
|
|
}];
|
|
} else {
|
|
// Sometimes the enumeration continues even if we set stop above, so we guard against calling the callback
|
|
// multiple times here.
|
|
if (!calledCallback) {
|
|
[self callCallback:callback withAssets:assets hasNextPage:hasNextPage];
|
|
calledCallback = YES;
|
|
}
|
|
}
|
|
} failureBlock:^(NSError *error) {
|
|
if (error.code != ALAssetsLibraryAccessUserDeniedError) {
|
|
RCTLogError(@"Failure while iterating through asset groups %@", error);
|
|
}
|
|
errorCallback(error);
|
|
}];
|
|
}
|
|
|
|
@end
|