yeet
This commit is contained in:
19
node_modules/react-native/React/Fabric/Mounting/ComponentViews/Image/RCTImageComponentView.h
generated
vendored
Normal file
19
node_modules/react-native/React/Fabric/Mounting/ComponentViews/Image/RCTImageComponentView.h
generated
vendored
Normal file
@ -0,0 +1,19 @@
|
||||
/*
|
||||
* 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 <React/RCTViewComponentView.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
/**
|
||||
* UIView class for root <Image> component.
|
||||
*/
|
||||
@interface RCTImageComponentView : RCTViewComponentView
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
204
node_modules/react-native/React/Fabric/Mounting/ComponentViews/Image/RCTImageComponentView.mm
generated
vendored
Normal file
204
node_modules/react-native/React/Fabric/Mounting/ComponentViews/Image/RCTImageComponentView.mm
generated
vendored
Normal file
@ -0,0 +1,204 @@
|
||||
/*
|
||||
* 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/RCTImageResponseDelegate.h>
|
||||
#import <React/RCTImageResponseObserverProxy.h>
|
||||
#import <react/components/image/ImageComponentDescriptor.h>
|
||||
#import <react/components/image/ImageEventEmitter.h>
|
||||
#import <react/components/image/ImageProps.h>
|
||||
#import <react/imagemanager/ImageInstrumentation.h>
|
||||
#import <react/imagemanager/ImageRequest.h>
|
||||
#import <react/imagemanager/RCTImageInstrumentationProxy.h>
|
||||
#import <react/imagemanager/RCTImagePrimitivesConversions.h>
|
||||
|
||||
#import "RCTConversions.h"
|
||||
#import "RCTFabricComponentsPlugins.h"
|
||||
|
||||
using namespace facebook::react;
|
||||
|
||||
@interface RCTImageComponentView () <RCTImageResponseDelegate>
|
||||
@end
|
||||
|
||||
@implementation RCTImageComponentView {
|
||||
UIImageView *_imageView;
|
||||
ImageShadowNode::ConcreteState::Shared _state;
|
||||
ImageResponseObserverCoordinator const *_coordinator;
|
||||
RCTImageResponseObserverProxy _imageResponseObserverProxy;
|
||||
}
|
||||
|
||||
- (instancetype)initWithFrame:(CGRect)frame
|
||||
{
|
||||
if (self = [super initWithFrame:frame]) {
|
||||
static auto const defaultProps = std::make_shared<ImageProps const>();
|
||||
_props = defaultProps;
|
||||
|
||||
_imageView = [[UIImageView alloc] initWithFrame:self.bounds];
|
||||
_imageView.clipsToBounds = YES;
|
||||
_imageView.contentMode = (UIViewContentMode)RCTResizeModeFromImageResizeMode(defaultProps->resizeMode);
|
||||
|
||||
_imageResponseObserverProxy = RCTImageResponseObserverProxy(self);
|
||||
|
||||
self.contentView = _imageView;
|
||||
}
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
#pragma mark - RCTComponentViewProtocol
|
||||
|
||||
+ (ComponentDescriptorProvider)componentDescriptorProvider
|
||||
{
|
||||
return concreteComponentDescriptorProvider<ImageComponentDescriptor>();
|
||||
}
|
||||
|
||||
- (void)updateProps:(Props::Shared const &)props oldProps:(Props::Shared const &)oldProps
|
||||
{
|
||||
auto const &oldImageProps = *std::static_pointer_cast<ImageProps const>(_props);
|
||||
auto const &newImageProps = *std::static_pointer_cast<ImageProps const>(props);
|
||||
|
||||
// `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()];
|
||||
}
|
||||
|
||||
[super updateProps:props oldProps:oldProps];
|
||||
}
|
||||
|
||||
- (void)updateState:(State::Shared const &)state oldState:(State::Shared const &)oldState
|
||||
{
|
||||
_state = std::static_pointer_cast<ImageShadowNode::ConcreteState const>(state);
|
||||
auto _oldState = std::static_pointer_cast<ImageShadowNode::ConcreteState const>(oldState);
|
||||
auto data = _state->getData();
|
||||
|
||||
// This call (setting `coordinator`) must be unconditional (at the same block as setting `State`)
|
||||
// because the setter stores a raw pointer to object that `State` owns.
|
||||
self.coordinator = &data.getImageRequest().getObserverCoordinator();
|
||||
|
||||
bool havePreviousData = _oldState && _oldState->getData().getImageSource() != ImageSource{};
|
||||
|
||||
if (!havePreviousData || data.getImageSource() != _oldState->getData().getImageSource()) {
|
||||
// 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<ImageEventEmitter const>(_eventEmitter)->onLoadStart();
|
||||
|
||||
// TODO (T58941612): Tracking for visibility should be done directly on this class.
|
||||
// For now, we consolidate instrumentation logic in the image loader, so that pre-Fabric gets the same treatment.
|
||||
auto instrumentation = std::static_pointer_cast<RCTImageInstrumentationProxy const>(
|
||||
data.getImageRequest().getSharedImageInstrumentation());
|
||||
if (instrumentation) {
|
||||
instrumentation->trackNativeImageView(self);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
- (void)setCoordinator:(ImageResponseObserverCoordinator const *)coordinator
|
||||
{
|
||||
if (_coordinator) {
|
||||
_coordinator->removeObserver(_imageResponseObserverProxy);
|
||||
}
|
||||
_coordinator = coordinator;
|
||||
if (_coordinator != nullptr) {
|
||||
_coordinator->addObserver(_imageResponseObserverProxy);
|
||||
}
|
||||
}
|
||||
|
||||
- (void)prepareForRecycle
|
||||
{
|
||||
[super prepareForRecycle];
|
||||
self.coordinator = nullptr;
|
||||
_imageView.image = nil;
|
||||
_state.reset();
|
||||
}
|
||||
|
||||
- (void)dealloc
|
||||
{
|
||||
self.coordinator = nullptr;
|
||||
}
|
||||
|
||||
#pragma mark - RCTImageResponseDelegate
|
||||
|
||||
- (void)didReceiveImage:(UIImage *)image fromObserver:(void const *)observer
|
||||
{
|
||||
if (!_eventEmitter || !_state) {
|
||||
// Notifications are delivered asynchronously and might arrive after the view is already recycled.
|
||||
// In the future, we should incorporate an `EventEmitter` into a separate object owned by `ImageRequest` or `State`.
|
||||
// See for more info: T46311063.
|
||||
return;
|
||||
}
|
||||
|
||||
std::static_pointer_cast<ImageEventEmitter const>(_eventEmitter)->onLoad();
|
||||
std::static_pointer_cast<ImageEventEmitter const>(_eventEmitter)->onLoadEnd();
|
||||
|
||||
const auto &imageProps = *std::static_pointer_cast<ImageProps const>(_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;
|
||||
|
||||
auto data = _state->getData();
|
||||
auto instrumentation = std::static_pointer_cast<RCTImageInstrumentationProxy const>(
|
||||
data.getImageRequest().getSharedImageInstrumentation());
|
||||
if (instrumentation) {
|
||||
instrumentation->didSetImage();
|
||||
}
|
||||
}
|
||||
|
||||
- (void)didReceiveProgress:(float)progress fromObserver:(void const *)observer
|
||||
{
|
||||
if (!_eventEmitter) {
|
||||
return;
|
||||
}
|
||||
|
||||
std::static_pointer_cast<ImageEventEmitter const>(_eventEmitter)->onProgress(progress);
|
||||
}
|
||||
|
||||
- (void)didReceiveFailureFromObserver:(void const *)observer
|
||||
{
|
||||
_imageView.image = nil;
|
||||
|
||||
if (!_eventEmitter) {
|
||||
return;
|
||||
}
|
||||
|
||||
std::static_pointer_cast<ImageEventEmitter const>(_eventEmitter)->onError();
|
||||
std::static_pointer_cast<ImageEventEmitter const>(_eventEmitter)->onLoadEnd();
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
Class<RCTComponentViewProtocol> RCTImageCls(void)
|
||||
{
|
||||
return RCTImageComponentView.class;
|
||||
}
|
Reference in New Issue
Block a user