mirror of
https://github.com/status-im/react-native.git
synced 2025-01-15 12:05:06 +00:00
0686b0147c
- [React Native] Fix RCTText crashes | Alex Akers - Ensure that NSLocationWhenInUseUsageDescription is set, throw error if not | Alex Kotliarskyi - [ReactNative] fix exception handler method name | Spencer Ahrens - [ReactNative] Re-configure horizontal swipe animations | Eric Vicenti - [ReactNative] <Text>: apply the fontWeight correctly if fontFamily style is also present | Kevin Gozali - [MAdMan] Dimensions.get('window') considered harmful | Philipp von Weitershausen - Navigator: Changed transitioner background color to 'transparent' | Eric Vicenti - [react-native] Listen on all IPv6 interfaces | Ben Alpert - [react-packager] Don't depend on error.stack being available | Amjad Masad - [ReactNative] fixup AnimationExperimental a bit | Spencer Ahrens - [react-packager] Implement new style asset packaging (with dimensions) | Amjad Masad - [React Native] RCT_EXPORT lvl.2 | Alex Akers - [react_native] Implement TextInput end editing | Andrei Coman - [react_native] Make TextInput focus, blur, dismiss and show keyboard work | Andrei Coman - Added non-class-scanning-based approach fror registering js methods | Nick Lockwood - [ReactNative] Update package.json | Christopher Chedeau - [ReactNative] Do flow check when running packager | Spencer Ahrens - [ReactNative] Fix typo/bug in Navigator._completeTransition | Eric Vicenti - [ReactNative] Fix Navigator exception when touching during transition | Eric Vicenti - [ReactNative] Remove bridge retaining cycles | Tadeu Zagallo - [ReactNative] Fix and re-add WebView executor | Tadeu Zagallo
158 lines
6.3 KiB
Objective-C
158 lines
6.3 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 "RCTImageLoader.h"
|
|
#import "RCTLog.h"
|
|
|
|
@implementation RCTCameraRollManager
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
RCT_EXPORT_METHOD(saveImageWithTag:(NSString *)imageTag
|
|
successCallback:(RCTResponseSenderBlock)successCallback
|
|
errorCallback:(RCTResponseSenderBlock)errorCallback)
|
|
{
|
|
[RCTImageLoader loadImageWithTag:imageTag callback:^(NSError *loadError, UIImage *loadedImage) {
|
|
if (loadError) {
|
|
errorCallback(@[[loadError localizedDescription]]);
|
|
return;
|
|
}
|
|
[[RCTImageLoader assetsLibrary] writeImageToSavedPhotosAlbum:[loadedImage CGImage] metadata:nil completionBlock:^(NSURL *assetURL, NSError *saveError) {
|
|
if (saveError) {
|
|
NSString *errorMessage = [NSString stringWithFormat:@"Error saving cropped image: %@", saveError];
|
|
RCTLogWarn(@"%@", errorMessage);
|
|
errorCallback(@[errorMessage]);
|
|
return;
|
|
}
|
|
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:(RCTResponseSenderBlock)errorCallback)
|
|
{
|
|
NSUInteger first = [params[@"first"] integerValue];
|
|
NSString *afterCursor = params[@"after"];
|
|
NSString *groupTypesStr = params[@"groupTypes"];
|
|
NSString *groupName = params[@"groupName"];
|
|
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 alloc] init];
|
|
|
|
[[RCTImageLoader assetsLibrary] enumerateGroupsWithTypes:groupTypes usingBlock:^(ALAssetsGroup *group, BOOL *stopGroups) {
|
|
if (group && (groupName == nil || [groupName isEqualToString:[group valueForProperty:ALAssetsGroupPropertyName]])) {
|
|
[group setAssetsFilter:ALAssetsFilter.allPhotos];
|
|
[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.description]);
|
|
}];
|
|
}
|
|
|
|
@end
|