mirror of
https://github.com/status-im/react-native.git
synced 2025-01-25 08:48:58 +00:00
e1577df1fd
Summary: To make React Native play nicely with our internal build infrastructure we need to properly namespace all of our header includes. Where previously you could do `#import "RCTBridge.h"`, you must now write this as `#import <React/RCTBridge.h>`. If your xcode project still has a custom header include path, both variants will likely continue to work, but for new projects, we're defaulting the header include path to `$(BUILT_PRODUCTS_DIR)/usr/local/include`, where the React and CSSLayout targets will copy a subset of headers too. To make Xcode copy headers phase work properly, you may need to add React as an explicit dependency to your app's scheme and disable "parallelize build". Reviewed By: mmmulani Differential Revision: D4213120 fbshipit-source-id: 84a32a4b250c27699e6795f43584f13d594a9a82
238 lines
7.0 KiB
Objective-C
238 lines
7.0 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 "RCTImageStoreManager.h"
|
|
|
|
#import <libkern/OSAtomic.h>
|
|
|
|
#import <ImageIO/ImageIO.h>
|
|
#import <MobileCoreServices/UTType.h>
|
|
|
|
#import <React/RCTAssert.h>
|
|
#import <React/RCTLog.h>
|
|
#import <React/RCTUtils.h>
|
|
|
|
#import "RCTImageUtils.h"
|
|
|
|
static NSString *const RCTImageStoreURLScheme = @"rct-image-store";
|
|
|
|
@implementation RCTImageStoreManager
|
|
{
|
|
NSMutableDictionary<NSString *, NSData *> *_store;
|
|
NSUInteger _id;
|
|
}
|
|
|
|
@synthesize methodQueue = _methodQueue;
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
- (void)removeImageForTag:(NSString *)imageTag withBlock:(void (^)())block
|
|
{
|
|
dispatch_async(_methodQueue, ^{
|
|
[self removeImageForTag:imageTag];
|
|
if (block) {
|
|
block();
|
|
}
|
|
});
|
|
}
|
|
|
|
- (NSString *)_storeImageData:(NSData *)imageData
|
|
{
|
|
RCTAssertThread(_methodQueue, @"Must be called on RCTImageStoreManager thread");
|
|
|
|
if (!_store) {
|
|
_store = [NSMutableDictionary new];
|
|
_id = 0;
|
|
}
|
|
|
|
NSString *imageTag = [NSString stringWithFormat:@"%@://%tu", RCTImageStoreURLScheme, _id++];
|
|
_store[imageTag] = imageData;
|
|
return imageTag;
|
|
}
|
|
|
|
- (void)storeImageData:(NSData *)imageData withBlock:(void (^)(NSString *imageTag))block
|
|
{
|
|
RCTAssertParam(block);
|
|
dispatch_async(_methodQueue, ^{
|
|
block([self _storeImageData:imageData]);
|
|
});
|
|
}
|
|
|
|
- (void)getImageDataForTag:(NSString *)imageTag withBlock:(void (^)(NSData *imageData))block
|
|
{
|
|
RCTAssertParam(block);
|
|
dispatch_async(_methodQueue, ^{
|
|
block(self->_store[imageTag]);
|
|
});
|
|
}
|
|
|
|
- (void)storeImage:(UIImage *)image withBlock:(void (^)(NSString *imageTag))block
|
|
{
|
|
RCTAssertParam(block);
|
|
dispatch_async(_methodQueue, ^{
|
|
NSString *imageTag = [self _storeImageData:RCTGetImageData(image.CGImage, 0.75)];
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
block(imageTag);
|
|
});
|
|
});
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(removeImageForTag:(NSString *)imageTag)
|
|
{
|
|
[_store removeObjectForKey:imageTag];
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(hasImageForTag:(NSString *)imageTag
|
|
callback:(RCTResponseSenderBlock)callback)
|
|
{
|
|
callback(@[@(_store[imageTag] != nil)]);
|
|
}
|
|
|
|
// TODO (#5906496): Name could be more explicit - something like getBase64EncodedDataForTag:?
|
|
RCT_EXPORT_METHOD(getBase64ForTag:(NSString *)imageTag
|
|
successCallback:(RCTResponseSenderBlock)successCallback
|
|
errorCallback:(RCTResponseErrorBlock)errorCallback)
|
|
{
|
|
NSData *imageData = _store[imageTag];
|
|
if (!imageData) {
|
|
errorCallback(RCTErrorWithMessage([NSString stringWithFormat:@"Invalid imageTag: %@", imageTag]));
|
|
return;
|
|
}
|
|
// Dispatching to a background thread to perform base64 encoding
|
|
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
|
|
successCallback(@[[imageData base64EncodedStringWithOptions:0]]);
|
|
});
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(addImageFromBase64:(NSString *)base64String
|
|
successCallback:(RCTResponseSenderBlock)successCallback
|
|
errorCallback:(RCTResponseErrorBlock)errorCallback)
|
|
|
|
{
|
|
// Dispatching to a background thread to perform base64 decoding
|
|
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
|
|
NSData *imageData = [[NSData alloc] initWithBase64EncodedString:base64String options:0];
|
|
if (imageData) {
|
|
dispatch_async(self->_methodQueue, ^{
|
|
successCallback(@[[self _storeImageData:imageData]]);
|
|
});
|
|
} else {
|
|
errorCallback(RCTErrorWithMessage(@"Failed to add image from base64String"));
|
|
}
|
|
});
|
|
}
|
|
|
|
#pragma mark - RCTURLRequestHandler
|
|
|
|
- (BOOL)canHandleRequest:(NSURLRequest *)request
|
|
{
|
|
return [request.URL.scheme caseInsensitiveCompare:RCTImageStoreURLScheme] == NSOrderedSame;
|
|
}
|
|
|
|
- (id)sendRequest:(NSURLRequest *)request withDelegate:(id<RCTURLRequestDelegate>)delegate
|
|
{
|
|
__block volatile uint32_t cancelled = 0;
|
|
void (^cancellationBlock)(void) = ^{
|
|
OSAtomicOr32Barrier(1, &cancelled);
|
|
};
|
|
|
|
// Dispatch async to give caller time to cancel the request
|
|
dispatch_async(_methodQueue, ^{
|
|
if (cancelled) {
|
|
return;
|
|
}
|
|
|
|
NSString *imageTag = request.URL.absoluteString;
|
|
NSData *imageData = self->_store[imageTag];
|
|
if (!imageData) {
|
|
NSError *error = RCTErrorWithMessage([NSString stringWithFormat:@"Invalid imageTag: %@", imageTag]);
|
|
[delegate URLRequest:cancellationBlock didCompleteWithError:error];
|
|
return;
|
|
}
|
|
|
|
CGImageSourceRef sourceRef = CGImageSourceCreateWithData((__bridge CFDataRef)imageData, NULL);
|
|
if (!sourceRef) {
|
|
NSError *error = RCTErrorWithMessage([NSString stringWithFormat:@"Unable to decode data for imageTag: %@", imageTag]);
|
|
[delegate URLRequest:cancellationBlock didCompleteWithError:error];
|
|
return;
|
|
}
|
|
CFStringRef UTI = CGImageSourceGetType(sourceRef);
|
|
CFRelease(sourceRef);
|
|
|
|
NSString *MIMEType = (__bridge_transfer NSString *)UTTypeCopyPreferredTagWithClass(UTI, kUTTagClassMIMEType);
|
|
NSURLResponse *response = [[NSURLResponse alloc] initWithURL:request.URL
|
|
MIMEType:MIMEType
|
|
expectedContentLength:imageData.length
|
|
textEncodingName:nil];
|
|
|
|
[delegate URLRequest:cancellationBlock didReceiveResponse:response];
|
|
[delegate URLRequest:cancellationBlock didReceiveData:imageData];
|
|
[delegate URLRequest:cancellationBlock didCompleteWithError:nil];
|
|
|
|
});
|
|
|
|
return cancellationBlock;
|
|
}
|
|
|
|
- (void)cancelRequest:(id)requestToken
|
|
{
|
|
if (requestToken) {
|
|
((void (^)(void))requestToken)();
|
|
}
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation RCTImageStoreManager (Deprecated)
|
|
|
|
- (NSString *)storeImage:(UIImage *)image
|
|
{
|
|
RCTAssertMainQueue();
|
|
RCTLogWarn(@"RCTImageStoreManager.storeImage() is deprecated and has poor performance. Use an alternative method instead.");
|
|
__block NSString *imageTag;
|
|
dispatch_sync(_methodQueue, ^{
|
|
imageTag = [self _storeImageData:RCTGetImageData(image.CGImage, 0.75)];
|
|
});
|
|
return imageTag;
|
|
}
|
|
|
|
- (UIImage *)imageForTag:(NSString *)imageTag
|
|
{
|
|
RCTAssertMainQueue();
|
|
RCTLogWarn(@"RCTImageStoreManager.imageForTag() is deprecated and has poor performance. Use an alternative method instead.");
|
|
__block NSData *imageData;
|
|
dispatch_sync(_methodQueue, ^{
|
|
imageData = self->_store[imageTag];
|
|
});
|
|
return [UIImage imageWithData:imageData];
|
|
}
|
|
|
|
- (void)getImageForTag:(NSString *)imageTag withBlock:(void (^)(UIImage *image))block
|
|
{
|
|
RCTAssertParam(block);
|
|
dispatch_async(_methodQueue, ^{
|
|
NSData *imageData = self->_store[imageTag];
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
// imageWithData: is not thread-safe, so we can't do this on methodQueue
|
|
block([UIImage imageWithData:imageData]);
|
|
});
|
|
});
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation RCTBridge (RCTImageStoreManager)
|
|
|
|
- (RCTImageStoreManager *)imageStoreManager
|
|
{
|
|
return [self moduleForClass:[RCTImageStoreManager class]];
|
|
}
|
|
|
|
@end
|