mirror of
https://github.com/facebook/react-native.git
synced 2025-11-01 09:14:26 +00:00
0937f13cc3
Reviewed By: sammy-SC Differential Revision: D24990452 fbshipit-source-id: ce4d5ec9e3cf237c6edbd17368f2dcf3aecbec2b
113 lines
4.3 KiB
Plaintext
113 lines
4.3 KiB
Plaintext
/*
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
#import "RCTImageManager.h"
|
|
|
|
#import <react/renderer/debug/SystraceSection.h>
|
|
#import <react/utils/ManagedObjectWrapper.h>
|
|
#import <react/utils/SharedFunction.h>
|
|
|
|
#import <React/RCTImageLoaderWithAttributionProtocol.h>
|
|
|
|
#import <react/renderer/imagemanager/ImageResponse.h>
|
|
#import <react/renderer/imagemanager/ImageResponseObserver.h>
|
|
|
|
#import "RCTImagePrimitivesConversions.h"
|
|
|
|
using namespace facebook::react;
|
|
|
|
@implementation RCTImageManager {
|
|
id<RCTImageLoaderWithAttributionProtocol> _imageLoader;
|
|
dispatch_queue_t _backgroundSerialQueue;
|
|
}
|
|
|
|
- (instancetype)initWithImageLoader:(id<RCTImageLoaderWithAttributionProtocol>)imageLoader
|
|
{
|
|
if (self = [super init]) {
|
|
_imageLoader = imageLoader;
|
|
_backgroundSerialQueue =
|
|
dispatch_queue_create("com.facebook.react-native.image-manager-queue", DISPATCH_QUEUE_SERIAL);
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
- (ImageRequest)requestImage:(ImageSource)imageSource surfaceId:(SurfaceId)surfaceId
|
|
{
|
|
SystraceSection s("RCTImageManager::requestImage");
|
|
|
|
NSURLRequest *request = NSURLRequestFromImageSource(imageSource);
|
|
std::shared_ptr<ImageTelemetry> telemetry;
|
|
if ([self->_imageLoader shouldEnablePerfLoggingForRequestUrl:request.URL]) {
|
|
telemetry = std::make_shared<ImageTelemetry>(surfaceId);
|
|
} else {
|
|
telemetry = nullptr;
|
|
}
|
|
|
|
auto imageRequest = ImageRequest(imageSource, telemetry);
|
|
auto weakObserverCoordinator =
|
|
(std::weak_ptr<const ImageResponseObserverCoordinator>)imageRequest.getSharedObserverCoordinator();
|
|
|
|
auto sharedCancelationFunction = SharedFunction<>();
|
|
imageRequest.setCancelationFunction(sharedCancelationFunction);
|
|
|
|
/*
|
|
* Even if an image is being loaded asynchronously on some other background thread, some other preparation
|
|
* work (such as creating an `NSURLRequest` object and some obscure logic inside `RCTImageLoader`) can take a couple
|
|
* of milliseconds, so we have to offload this to a separate thread. `ImageRequest` can be created as part of the
|
|
* layout process, so it must be highly performant.
|
|
*
|
|
* Technically, we don't need to dispatch this to *serial* queue. The interface of `RCTImageLoader` promises to be
|
|
* fully thread-safe. However, in reality, it crashes when we request images on concurrently on different threads. See
|
|
* T46024425 for more details.
|
|
*/
|
|
dispatch_async(_backgroundSerialQueue, ^{
|
|
auto completionBlock = ^(NSError *error, UIImage *image, id metadata) {
|
|
auto observerCoordinator = weakObserverCoordinator.lock();
|
|
if (!observerCoordinator) {
|
|
return;
|
|
}
|
|
|
|
if (image && !error) {
|
|
auto wrappedMetadata = metadata ? wrapManagedObject(metadata) : nullptr;
|
|
observerCoordinator->nativeImageResponseComplete(ImageResponse(wrapManagedObject(image), wrappedMetadata));
|
|
} else {
|
|
observerCoordinator->nativeImageResponseFailed();
|
|
}
|
|
};
|
|
|
|
auto progressBlock = ^(int64_t progress, int64_t total) {
|
|
auto observerCoordinator = weakObserverCoordinator.lock();
|
|
if (!observerCoordinator) {
|
|
return;
|
|
}
|
|
|
|
observerCoordinator->nativeImageResponseProgress(progress / (float)total);
|
|
};
|
|
|
|
RCTImageURLLoaderRequest *loaderRequest =
|
|
[self->_imageLoader loadImageWithURLRequest:request
|
|
size:CGSizeMake(imageSource.size.width, imageSource.size.height)
|
|
scale:imageSource.scale
|
|
clipped:NO
|
|
resizeMode:RCTResizeModeStretch
|
|
priority:RCTImageLoaderPriorityImmediate
|
|
attribution:{
|
|
.surfaceId = surfaceId,
|
|
}
|
|
progressBlock:progressBlock
|
|
partialLoadBlock:nil
|
|
completionBlock:completionBlock];
|
|
RCTImageLoaderCancellationBlock cancelationBlock = loaderRequest.cancellationBlock;
|
|
sharedCancelationFunction.assign([cancelationBlock]() { cancelationBlock(); });
|
|
});
|
|
|
|
return imageRequest;
|
|
}
|
|
|
|
@end
|