172 lines
5.6 KiB
Plaintext
172 lines
5.6 KiB
Plaintext
/*
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
#import "UIView+ComponentViewProtocol.h"
|
|
|
|
#import <React/RCTAssert.h>
|
|
#import <React/RCTLog.h>
|
|
#import <React/RCTUtils.h>
|
|
|
|
#import "RCTConversions.h"
|
|
|
|
using namespace facebook::react;
|
|
|
|
@implementation UIView (ComponentViewProtocol)
|
|
|
|
+ (ComponentDescriptorProvider)componentDescriptorProvider
|
|
{
|
|
RCTAssert(NO, @"`-[RCTComponentViewProtocol componentDescriptorProvider]` must be implemented in a concrete class.");
|
|
return {};
|
|
}
|
|
|
|
+ (std::vector<facebook::react::ComponentDescriptorProvider>)supplementalComponentDescriptorProviders
|
|
{
|
|
return {};
|
|
}
|
|
|
|
- (void)mountChildComponentView:(UIView<RCTComponentViewProtocol> *)childComponentView index:(NSInteger)index
|
|
{
|
|
RCTAssert(
|
|
childComponentView.superview == nil,
|
|
@"Attempt to mount already mounted component view. (parent: %@, child: %@, index: %@, existing parent: %@)",
|
|
self,
|
|
childComponentView,
|
|
@(index),
|
|
@([childComponentView.superview tag]));
|
|
[self insertSubview:childComponentView atIndex:index];
|
|
}
|
|
|
|
- (void)unmountChildComponentView:(UIView<RCTComponentViewProtocol> *)childComponentView index:(NSInteger)index
|
|
{
|
|
RCTAssert(
|
|
childComponentView.superview == self,
|
|
@"Attempt to unmount a view which is mounted inside different view. (parent: %@, child: %@, index: %@)",
|
|
self,
|
|
childComponentView,
|
|
@(index));
|
|
RCTAssert(
|
|
(self.subviews.count > index) && [self.subviews objectAtIndex:index] == childComponentView,
|
|
@"Attempt to unmount a view which has a different index. (parent: %@, child: %@, index: %@, actual index: %@, tag at index: %@)",
|
|
self,
|
|
childComponentView,
|
|
@(index),
|
|
@([self.subviews indexOfObject:childComponentView]),
|
|
@([[self.subviews objectAtIndex:index] tag]));
|
|
|
|
[childComponentView removeFromSuperview];
|
|
}
|
|
|
|
- (void)updateProps:(const Props::Shared &)props oldProps:(const Props::Shared &)oldProps
|
|
{
|
|
// Default implementation does nothing.
|
|
}
|
|
|
|
- (void)updateEventEmitter:(const EventEmitter::Shared &)eventEmitter
|
|
{
|
|
// Default implementation does nothing.
|
|
}
|
|
|
|
- (void)updateState:(const facebook::react::State::Shared &)state
|
|
oldState:(const facebook::react::State::Shared &)oldState
|
|
{
|
|
// Default implementation does nothing.
|
|
}
|
|
|
|
- (void)handleCommand:(NSString *)commandName args:(NSArray *)args
|
|
{
|
|
// Default implementation does nothing.
|
|
}
|
|
|
|
- (void)updateLayoutMetrics:(const LayoutMetrics &)layoutMetrics
|
|
oldLayoutMetrics:(const LayoutMetrics &)oldLayoutMetrics
|
|
{
|
|
bool forceUpdate = oldLayoutMetrics == EmptyLayoutMetrics;
|
|
|
|
if (forceUpdate || (layoutMetrics.frame != oldLayoutMetrics.frame)) {
|
|
CGRect frame = RCTCGRectFromRect(layoutMetrics.frame);
|
|
|
|
if (!std::isfinite(frame.origin.x) || !std::isfinite(frame.origin.y) || !std::isfinite(frame.size.width) ||
|
|
!std::isfinite(frame.size.height)) {
|
|
// CALayer will crash if we pass NaN or Inf values.
|
|
// It's unclear how to detect this case on cross-platform manner holistically, so we have to do it on the mounting
|
|
// layer as well. NaN/Inf is a kinda valid result of some math operations. Even if we can (and should) detect (and
|
|
// report early) incorrect (NaN and Inf) values which come from JavaScript side, we sometimes cannot backtrace the
|
|
// sources of a calculation that produced an incorrect/useless result.
|
|
RCTLogWarn(
|
|
@"-[UIView(ComponentViewProtocol) updateLayoutMetrics:oldLayoutMetrics:]: Received invalid layout metrics (%@) for a view (%@).",
|
|
NSStringFromCGRect(frame),
|
|
self);
|
|
} else {
|
|
// Note: Changing `frame` when `layer.transform` is not the `identity transform` is undefined behavior.
|
|
// Therefore, we must use `center` and `bounds`.
|
|
self.center = CGPoint{CGRectGetMidX(frame), CGRectGetMidY(frame)};
|
|
self.bounds = CGRect{CGPointZero, frame.size};
|
|
}
|
|
}
|
|
|
|
if (forceUpdate || (layoutMetrics.layoutDirection != oldLayoutMetrics.layoutDirection)) {
|
|
self.semanticContentAttribute = layoutMetrics.layoutDirection == LayoutDirection::RightToLeft
|
|
? UISemanticContentAttributeForceRightToLeft
|
|
: UISemanticContentAttributeForceLeftToRight;
|
|
}
|
|
|
|
if (forceUpdate || (layoutMetrics.displayType != oldLayoutMetrics.displayType)) {
|
|
self.hidden = layoutMetrics.displayType == DisplayType::None;
|
|
}
|
|
}
|
|
|
|
- (void)finalizeUpdates:(RNComponentViewUpdateMask)updateMask
|
|
{
|
|
// Default implementation does nothing.
|
|
}
|
|
|
|
- (void)prepareForRecycle
|
|
{
|
|
// Default implementation does nothing.
|
|
}
|
|
|
|
- (facebook::react::Props::Shared)props
|
|
{
|
|
RCTAssert(NO, @"props access should be implemented by RCTViewComponentView.");
|
|
return nullptr;
|
|
}
|
|
|
|
- (BOOL)isJSResponder
|
|
{
|
|
// Default implementation always returns `NO`.
|
|
return NO;
|
|
}
|
|
|
|
- (void)setIsJSResponder:(BOOL)isJSResponder
|
|
{
|
|
// Default implementation does nothing.
|
|
}
|
|
|
|
- (void)setPropKeysManagedByAnimated_DO_NOT_USE_THIS_IS_BROKEN:(nullable NSSet<NSString *> *)propKeys
|
|
{
|
|
// Default implementation does nothing.
|
|
}
|
|
|
|
- (nullable NSSet<NSString *> *)propKeysManagedByAnimated_DO_NOT_USE_THIS_IS_BROKEN
|
|
{
|
|
return nil;
|
|
}
|
|
|
|
- (void)updateClippedSubviewsWithClipRect:(CGRect)clipRect relativeToView:(UIView *)clipView
|
|
{
|
|
clipRect = [clipView convertRect:clipRect toView:self];
|
|
|
|
// Normal views don't support unmounting, so all
|
|
// this does is forward message to our subviews,
|
|
// in case any of those do support it
|
|
for (UIView *subview in self.subviews) {
|
|
[subview updateClippedSubviewsWithClipRect:clipRect relativeToView:self];
|
|
}
|
|
}
|
|
|
|
@end
|