mirror of
https://github.com/facebook/react-native.git
synced 2025-11-01 09:14:26 +00:00
b241bc2d0b
Summary: We don't need it anymore because we don't use folly::Future. Reviewed By: JoshuaGross Differential Revision: D15344961 fbshipit-source-id: a368e600d69e6879f6a98b4a997de85c71186e92
159 lines
5.3 KiB
Plaintext
159 lines
5.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 "RCTImageComponentView.h"
|
|
|
|
#import <React/RCTImageResponseObserverProxy.h>
|
|
#import <react/components/image/ImageComponentDescriptor.h>
|
|
#import <react/components/image/ImageEventEmitter.h>
|
|
#import <react/components/image/ImageLocalData.h>
|
|
#import <react/components/image/ImageProps.h>
|
|
#import <react/imagemanager/ImageRequest.h>
|
|
#import <react/imagemanager/RCTImagePrimitivesConversions.h>
|
|
|
|
#import "RCTConversions.h"
|
|
|
|
@implementation RCTImageComponentView {
|
|
UIImageView *_imageView;
|
|
SharedImageLocalData _imageLocalData;
|
|
const ImageResponseObserverCoordinator *_coordinator;
|
|
std::unique_ptr<RCTImageResponseObserverProxy> _imageResponseObserverProxy;
|
|
}
|
|
|
|
- (instancetype)initWithFrame:(CGRect)frame
|
|
{
|
|
if (self = [super initWithFrame:frame]) {
|
|
static const auto defaultProps = std::make_shared<const ImageProps>();
|
|
_props = defaultProps;
|
|
|
|
_imageView = [[UIImageView alloc] initWithFrame:self.bounds];
|
|
_imageView.clipsToBounds = YES;
|
|
|
|
_imageView.contentMode = (UIViewContentMode)RCTResizeModeFromImageResizeMode(defaultProps->resizeMode);
|
|
|
|
_imageResponseObserverProxy = std::make_unique<RCTImageResponseObserverProxy>((__bridge void *)self);
|
|
|
|
self.contentView = _imageView;
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
#pragma mark - RCTComponentViewProtocol
|
|
|
|
+ (ComponentDescriptorProvider)componentDescriptorProvider
|
|
{
|
|
return concreteComponentDescriptorProvider<ImageComponentDescriptor>();
|
|
}
|
|
|
|
- (void)updateProps:(SharedProps)props oldProps:(SharedProps)oldProps
|
|
{
|
|
const auto &oldImageProps = *std::static_pointer_cast<const ImageProps>(oldProps ?: _props);
|
|
const auto &newImageProps = *std::static_pointer_cast<const ImageProps>(props);
|
|
|
|
[super updateProps:props oldProps:oldProps];
|
|
|
|
// `resizeMode`
|
|
if (oldImageProps.resizeMode != newImageProps.resizeMode) {
|
|
if (newImageProps.resizeMode == ImageResizeMode::Repeat) {
|
|
// Repeat resize mode is handled by the UIImage. Use scale to fill
|
|
// so the repeated image fills the UIImageView.
|
|
_imageView.contentMode = UIViewContentModeScaleToFill;
|
|
} else {
|
|
_imageView.contentMode = (UIViewContentMode)RCTResizeModeFromImageResizeMode(newImageProps.resizeMode);
|
|
}
|
|
}
|
|
|
|
// `tintColor`
|
|
if (oldImageProps.tintColor != newImageProps.tintColor) {
|
|
_imageView.tintColor = [UIColor colorWithCGColor:newImageProps.tintColor.get()];
|
|
}
|
|
}
|
|
|
|
- (void)updateLocalData:(SharedLocalData)localData oldLocalData:(SharedLocalData)oldLocalData
|
|
{
|
|
SharedImageLocalData previousData = _imageLocalData;
|
|
_imageLocalData = std::static_pointer_cast<const ImageLocalData>(localData);
|
|
assert(_imageLocalData);
|
|
bool havePreviousData = previousData != nullptr;
|
|
|
|
if (!havePreviousData || _imageLocalData->getImageSource() != previousData->getImageSource()) {
|
|
self.coordinator = &_imageLocalData->getImageRequest().getObserverCoordinator();
|
|
|
|
// Loading actually starts a little before this, but this is the first time we know
|
|
// the image is loading and can fire an event from this component
|
|
std::static_pointer_cast<const ImageEventEmitter>(_eventEmitter)->onLoadStart();
|
|
}
|
|
}
|
|
|
|
- (void)setCoordinator:(const ImageResponseObserverCoordinator *)coordinator
|
|
{
|
|
if (_coordinator) {
|
|
_coordinator->removeObserver(_imageResponseObserverProxy.get());
|
|
}
|
|
_coordinator = coordinator;
|
|
if (_coordinator != nullptr) {
|
|
_coordinator->addObserver(_imageResponseObserverProxy.get());
|
|
}
|
|
}
|
|
|
|
- (void)prepareForRecycle
|
|
{
|
|
[super prepareForRecycle];
|
|
self.coordinator = nullptr;
|
|
_imageView.image = nil;
|
|
_imageLocalData.reset();
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
self.coordinator = nullptr;
|
|
_imageResponseObserverProxy.reset();
|
|
}
|
|
|
|
#pragma mark - RCTImageResponseDelegate
|
|
|
|
- (void)didReceiveImage:(UIImage *)image fromObserver:(void *)observer
|
|
{
|
|
std::static_pointer_cast<const ImageEventEmitter>(_eventEmitter)->onLoad();
|
|
|
|
const auto &imageProps = *std::static_pointer_cast<const ImageProps>(_props);
|
|
|
|
if (imageProps.tintColor) {
|
|
image = [image imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate];
|
|
}
|
|
|
|
if (imageProps.resizeMode == ImageResizeMode::Repeat) {
|
|
image = [image resizableImageWithCapInsets:RCTUIEdgeInsetsFromEdgeInsets(imageProps.capInsets)
|
|
resizingMode:UIImageResizingModeTile];
|
|
} else if (imageProps.capInsets != EdgeInsets()) {
|
|
// Applying capInsets of 0 will switch the "resizingMode" of the image to "tile" which is undesired.
|
|
image = [image resizableImageWithCapInsets:RCTUIEdgeInsetsFromEdgeInsets(imageProps.capInsets)
|
|
resizingMode:UIImageResizingModeStretch];
|
|
}
|
|
|
|
self->_imageView.image = image;
|
|
|
|
// Apply trilinear filtering to smooth out mis-sized images.
|
|
self->_imageView.layer.minificationFilter = kCAFilterTrilinear;
|
|
self->_imageView.layer.magnificationFilter = kCAFilterTrilinear;
|
|
|
|
std::static_pointer_cast<const ImageEventEmitter>(self->_eventEmitter)->onLoadEnd();
|
|
}
|
|
|
|
- (void)didReceiveProgress:(float)progress fromObserver:(void *)observer
|
|
{
|
|
std::static_pointer_cast<const ImageEventEmitter>(_eventEmitter)->onProgress(progress);
|
|
}
|
|
|
|
- (void)didReceiveFailureFromObserver:(void *)observer
|
|
{
|
|
std::static_pointer_cast<const ImageEventEmitter>(_eventEmitter)->onError();
|
|
}
|
|
|
|
@end
|