amis-rpc-design/node_modules/react-native/ReactCommon/react/renderer/runtimescheduler/RuntimeSchedulerBinding.cpp
2023-10-07 19:42:30 +08:00

205 lines
6.2 KiB
C++

/*
* 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.
*/
#include "RuntimeSchedulerBinding.h"
#include <ReactCommon/SchedulerPriority.h>
#include "RuntimeScheduler.h"
#include "SchedulerPriorityUtils.h"
#include "primitives.h"
#include <chrono>
#include <memory>
#include <utility>
namespace facebook::react {
std::shared_ptr<RuntimeSchedulerBinding>
RuntimeSchedulerBinding::createAndInstallIfNeeded(
jsi::Runtime &runtime,
std::shared_ptr<RuntimeScheduler> const &runtimeScheduler) {
auto runtimeSchedulerModuleName = "nativeRuntimeScheduler";
auto runtimeSchedulerValue =
runtime.global().getProperty(runtime, runtimeSchedulerModuleName);
if (runtimeSchedulerValue.isUndefined()) {
// The global namespace does not have an instance of the binding;
// we need to create, install and return it.
auto runtimeSchedulerBinding =
std::make_shared<RuntimeSchedulerBinding>(runtimeScheduler);
auto object =
jsi::Object::createFromHostObject(runtime, runtimeSchedulerBinding);
runtime.global().setProperty(
runtime, runtimeSchedulerModuleName, std::move(object));
return runtimeSchedulerBinding;
}
// The global namespace already has an instance of the binding;
// we need to return that.
auto runtimeSchedulerObject = runtimeSchedulerValue.asObject(runtime);
return runtimeSchedulerObject.getHostObject<RuntimeSchedulerBinding>(runtime);
}
std::shared_ptr<RuntimeSchedulerBinding> RuntimeSchedulerBinding::getBinding(
jsi::Runtime &runtime) {
auto runtimeSchedulerModuleName = "nativeRuntimeScheduler";
auto runtimeSchedulerValue =
runtime.global().getProperty(runtime, runtimeSchedulerModuleName);
if (runtimeSchedulerValue.isUndefined()) {
return nullptr;
}
auto runtimeSchedulerObject = runtimeSchedulerValue.asObject(runtime);
return runtimeSchedulerObject.getHostObject<RuntimeSchedulerBinding>(runtime);
}
RuntimeSchedulerBinding::RuntimeSchedulerBinding(
std::shared_ptr<RuntimeScheduler> runtimeScheduler)
: runtimeScheduler_(std::move(runtimeScheduler)) {}
bool RuntimeSchedulerBinding::getIsSynchronous() const {
return runtimeScheduler_->getIsSynchronous();
}
jsi::Value RuntimeSchedulerBinding::get(
jsi::Runtime &runtime,
jsi::PropNameID const &name) {
auto propertyName = name.utf8(runtime);
if (propertyName == "unstable_scheduleCallback") {
return jsi::Function::createFromHostFunction(
runtime,
name,
3,
[this](
jsi::Runtime &runtime,
jsi::Value const &,
jsi::Value const *arguments,
size_t count) noexcept -> jsi::Value {
SchedulerPriority priority = fromRawValue(arguments[0].getNumber());
auto callback = arguments[1].getObject(runtime).getFunction(runtime);
auto task =
runtimeScheduler_->scheduleTask(priority, std::move(callback));
return valueFromTask(runtime, task);
});
}
if (propertyName == "unstable_cancelCallback") {
return jsi::Function::createFromHostFunction(
runtime,
name,
1,
[this](
jsi::Runtime &runtime,
jsi::Value const &,
jsi::Value const *arguments,
size_t) noexcept -> jsi::Value {
runtimeScheduler_->cancelTask(*taskFromValue(runtime, arguments[0]));
return jsi::Value::undefined();
});
}
if (propertyName == "unstable_shouldYield") {
return jsi::Function::createFromHostFunction(
runtime,
name,
0,
[this](
jsi::Runtime &,
jsi::Value const &,
jsi::Value const *,
size_t) noexcept -> jsi::Value {
auto shouldYield = runtimeScheduler_->getShouldYield();
return {shouldYield};
});
}
if (propertyName == "unstable_requestPaint") {
return jsi::Function::createFromHostFunction(
runtime,
name,
0,
[](jsi::Runtime &,
jsi::Value const &,
jsi::Value const *,
size_t) noexcept -> jsi::Value {
// RequestPaint is left empty by design.
return jsi::Value::undefined();
});
}
if (propertyName == "unstable_now") {
return jsi::Function::createFromHostFunction(
runtime,
name,
0,
[this](
jsi::Runtime &,
jsi::Value const &,
jsi::Value const *,
size_t) noexcept -> jsi::Value {
auto now = runtimeScheduler_->now();
auto asDouble =
std::chrono::duration<double, std::milli>(now.time_since_epoch())
.count();
return {asDouble};
});
}
// TODO: remove this, as it's deprecated in the JS scheduler
if (propertyName == "unstable_getCurrentPriorityLevel") {
return jsi::Function::createFromHostFunction(
runtime,
name,
0,
[this](
jsi::Runtime &runtime,
jsi::Value const &,
jsi::Value const *,
size_t) noexcept -> jsi::Value {
auto currentPriorityLevel =
runtimeScheduler_->getCurrentPriorityLevel();
return jsi::Value(runtime, serialize(currentPriorityLevel));
});
}
if (propertyName == "unstable_ImmediatePriority") {
return jsi::Value(runtime, serialize(SchedulerPriority::ImmediatePriority));
}
if (propertyName == "unstable_UserBlockingPriority") {
return jsi::Value(
runtime, serialize(SchedulerPriority::UserBlockingPriority));
}
if (propertyName == "unstable_NormalPriority") {
return jsi::Value(runtime, serialize(SchedulerPriority::NormalPriority));
}
if (propertyName == "unstable_LowPriority") {
return jsi::Value(runtime, serialize(SchedulerPriority::LowPriority));
}
if (propertyName == "unstable_IdlePriority") {
return jsi::Value(runtime, serialize(SchedulerPriority::IdlePriority));
}
if (propertyName == "$$typeof") {
return jsi::Value::undefined();
}
#ifdef REACT_NATIVE_DEBUG
throw std::runtime_error("undefined property");
#else
return jsi::Value::undefined();
#endif
}
} // namespace facebook::react