/* * 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 "RCTScheduler.h" #import #import #import #import #import #include #import #import "RCTConversions.h" using namespace facebook::react; class SchedulerDelegateProxy : public SchedulerDelegate { public: SchedulerDelegateProxy(void *scheduler) : scheduler_(scheduler) {} void schedulerDidFinishTransaction(MountingCoordinator::Shared mountingCoordinator) override { RCTScheduler *scheduler = (__bridge RCTScheduler *)scheduler_; [scheduler.delegate schedulerDidFinishTransaction:std::move(mountingCoordinator)]; } void schedulerDidRequestPreliminaryViewAllocation(SurfaceId surfaceId, const ShadowNode &shadowNode) override { // Does nothing. // This delegate method is not currently used on iOS. } void schedulerDidDispatchCommand( const ShadowView &shadowView, const std::string &commandName, const folly::dynamic &args) override { RCTScheduler *scheduler = (__bridge RCTScheduler *)scheduler_; [scheduler.delegate schedulerDidDispatchCommand:shadowView commandName:commandName args:args]; } void schedulerDidSetIsJSResponder(ShadowView const &shadowView, bool isJSResponder, bool blockNativeResponder) override { RCTScheduler *scheduler = (__bridge RCTScheduler *)scheduler_; [scheduler.delegate schedulerDidSetIsJSResponder:isJSResponder blockNativeResponder:blockNativeResponder forShadowView:shadowView]; } void schedulerDidSendAccessibilityEvent(const ShadowView &shadowView, std::string const &eventType) override { RCTScheduler *scheduler = (__bridge RCTScheduler *)scheduler_; [scheduler.delegate schedulerDidSendAccessibilityEvent:shadowView eventType:eventType]; } private: void *scheduler_; }; class LayoutAnimationDelegateProxy : public LayoutAnimationStatusDelegate, public RunLoopObserver::Delegate { public: LayoutAnimationDelegateProxy(void *scheduler) : scheduler_(scheduler) {} virtual ~LayoutAnimationDelegateProxy() {} void onAnimationStarted() override { RCTScheduler *scheduler = (__bridge RCTScheduler *)scheduler_; [scheduler onAnimationStarted]; } /** * Called when the LayoutAnimation engine completes all pending animations. */ void onAllAnimationsComplete() override { RCTScheduler *scheduler = (__bridge RCTScheduler *)scheduler_; [scheduler onAllAnimationsComplete]; } void activityDidChange(RunLoopObserver::Delegate const *delegate, RunLoopObserver::Activity activity) const noexcept override { RCTScheduler *scheduler = (__bridge RCTScheduler *)scheduler_; [scheduler animationTick]; } private: void *scheduler_; }; @implementation RCTScheduler { std::unique_ptr _scheduler; std::shared_ptr _animationDriver; std::shared_ptr _delegateProxy; std::shared_ptr _layoutAnimationDelegateProxy; RunLoopObserver::Unique _uiRunLoopObserver; } - (instancetype)initWithToolbox:(SchedulerToolbox)toolbox { if (self = [super init]) { auto reactNativeConfig = toolbox.contextContainer->at>("ReactNativeConfig"); _delegateProxy = std::make_shared((__bridge void *)self); if (reactNativeConfig->getBool("react_fabric:enabled_layout_animations_ios")) { _layoutAnimationDelegateProxy = std::make_shared((__bridge void *)self); _animationDriver = std::make_shared( toolbox.runtimeExecutor, toolbox.contextContainer, _layoutAnimationDelegateProxy.get()); _uiRunLoopObserver = toolbox.mainRunLoopObserverFactory(RunLoopObserver::Activity::BeforeWaiting, _layoutAnimationDelegateProxy); _uiRunLoopObserver->setDelegate(_layoutAnimationDelegateProxy.get()); } _scheduler = std::make_unique( toolbox, (_animationDriver ? _animationDriver.get() : nullptr), _delegateProxy.get()); } return self; } - (void)animationTick { _scheduler->animationTick(); } - (void)dealloc { if (_animationDriver) { _animationDriver->setLayoutAnimationStatusDelegate(nullptr); } _scheduler->setDelegate(nullptr); } - (void)registerSurface:(facebook::react::SurfaceHandler const &)surfaceHandler { _scheduler->registerSurface(surfaceHandler); } - (void)unregisterSurface:(facebook::react::SurfaceHandler const &)surfaceHandler { _scheduler->unregisterSurface(surfaceHandler); } - (ComponentDescriptor const *)findComponentDescriptorByHandle_DO_NOT_USE_THIS_IS_BROKEN:(ComponentHandle)handle { return _scheduler->findComponentDescriptorByHandle_DO_NOT_USE_THIS_IS_BROKEN(handle); } - (void)setupAnimationDriver:(facebook::react::SurfaceHandler const &)surfaceHandler { surfaceHandler.getMountingCoordinator()->setMountingOverrideDelegate(_animationDriver); } - (void)onAnimationStarted { if (_uiRunLoopObserver) { _uiRunLoopObserver->enable(); } } - (void)onAllAnimationsComplete { if (_uiRunLoopObserver) { _uiRunLoopObserver->disable(); } } - (void)addEventListener:(std::shared_ptr const &)listener { return _scheduler->addEventListener(listener); } - (void)removeEventListener:(std::shared_ptr const &)listener { return _scheduler->removeEventListener(listener); } - (std::shared_ptr const)uiManager { return _scheduler->getUIManager(); } @end