4#include <qohosplugincore.h>
7#include <QtCore/private/qohoslogger_p.h>
13#include <napi/native_api.h>
26constexpr const char *forceLoadJsModulesEnvVariableName =
"IO__QT__OHOS__FORCE_LOAD_JS_MODULES";
28constexpr const char *napiLoadableModules[] = {
29 "@ohos.abilityAccessCtrl",
30 "@ohos.accessibility",
31 "@ohos.app.ability.AbilityConstant",
32 "@ohos.app.ability.ConfigurationConstant",
33 "@ohos.app.ability.childProcessManager",
34 "@ohos.app.ability.contextConstant",
35 "@ohos.app.ability.wantConstant",
36 "@ohos.arkui.uiExtension",
37 "@ohos.bluetooth.access",
38 "@ohos.bluetooth.connection",
39 "@ohos.bluetooth.socket",
40 "@ohos.bundle.bundleManager",
41 "@ohos.data.uniformTypeDescriptor",
47 "@ohos.geoLocationManager",
48 "@ohos.graphics.text",
52 "@ohos.multimedia.image",
53 "@ohos.multimedia.media",
54 "@ohos.multimodalInput.inputDevice",
55 "@ohos.multimodalInput.pointer",
56 "@ohos.notificationManager",
67 DummyQAbilityPeer() =
default;
69 std::string instanceId()
override;
70 QNapi::Object uiContext()
override;
71 QNapi::Object qAbility()
override;
72 QNapi::Object launchWant()
override;
73 QObjectThreadSafeRef qWindowRef()
override;
74 QOhosOptional<QNapi::Promise> qWindowDestroyPromise()
override;
75 std::shared_ptr<
std::atomic_bool> destroyAllowedFlag()
override;
76 bool isTerminating()
override;
78 void setQWindow(Napi::Env env, QObjectThreadSafeRef)
override;
80 void *tryCastWithTypeIdObject(
const void *matchTypeIdObject)
override;
83std::string DummyQAbilityPeer::instanceId()
88QNapi::Object DummyQAbilityPeer::uiContext()
90 return QNapi::Object();
93QNapi::Object DummyQAbilityPeer::qAbility()
95 return QNapi::Object();
98QNapi::Object DummyQAbilityPeer::launchWant()
100 return QNapi::Object();
103QObjectThreadSafeRef DummyQAbilityPeer::qWindowRef()
108QOhosOptional<QNapi::Promise> DummyQAbilityPeer::qWindowDestroyPromise()
113std::shared_ptr<
std::atomic_bool> DummyQAbilityPeer::destroyAllowedFlag()
115 return std::make_shared<
std::atomic_bool>(
false);
118bool DummyQAbilityPeer::isTerminating()
123void DummyQAbilityPeer::setQWindow(Napi::Env, QObjectThreadSafeRef)
127void *DummyQAbilityPeer::tryCastWithTypeIdObject(
const void *)
132template<
typename Task>
133class PreQueuingTasksExecutor
136 void setUnderlyingExecutor(
137 std::function<
void(Task)> executor,
138 std::function<
void(Task)> optSyncFlushExecutor =
nullptr);
139 void invokeTask(Task task);
142 std::recursive_mutex m_executorMutex;
143 std::function<
void(Task)> m_optUnderlyingExecutor;
144 std::deque<Task> m_pendingTasks;
147template<
typename Task>
148void PreQueuingTasksExecutor<Task>::setUnderlyingExecutor(
149 std::function<
void(Task)> executor,
std::function<
void(Task)> optSyncFlushExecutor)
151 std::lock_guard<
std::recursive_mutex> executorLock(m_executorMutex);
152 auto &flushExecutor = optSyncFlushExecutor ? optSyncFlushExecutor : executor;
153 while (!m_pendingTasks.empty()) {
154 auto task =
std::move(m_pendingTasks.front());
155 m_pendingTasks.pop_front();
156 flushExecutor(
std::move(task));
158 m_optUnderlyingExecutor =
std::move(executor);
161template<
typename Task>
162void PreQueuingTasksExecutor<Task>::invokeTask(Task task)
164 std::lock_guard<
std::recursive_mutex> executorLock(m_executorMutex);
165 if (m_optUnderlyingExecutor)
166 m_optUnderlyingExecutor(
std::move(task));
168 m_pendingTasks.push_back(std::move(task));
171using PreQueuingJsTasksExecutor = PreQueuingTasksExecutor<
std::function<
void(
JsState &)>>;
176 std::mutex pendingTasksMutex;
177 bool pendingTasksRunnerActive =
false;
178 std::vector<std::function<
void(JsState &)>> pendingTasks;
179 Napi::ThreadSafeFunction executorThreadSafeFunc;
181 auto ctx =
std::make_shared<Context>();
183 ctx->executorThreadSafeFunc = Napi::ThreadSafeFunction::New(
185 QNapi::Function::New(
187 [ctx = ctx.get()](
const CallbackInfo &cbInfo) {
189 std::vector<std::function<
void(JsState &)>> pendingTasks;
191 std::lock_guard<std::mutex> pendingTasksLock(ctx->pendingTasksMutex);
192 pendingTasks = std::exchange(ctx->pendingTasks, {});
193 if (pendingTasks.empty()) {
194 ctx->pendingTasksRunnerActive =
false;
198 for (
auto &task : pendingTasks)
199 task(cbInfo.jsState());
202 "QCoreOhosTasksExecutorFunc"),
203 "QCoreOhosTasksExecutor", 0, 1);
205 return [ctx](
std::function<
void(
JsState &)> task) {
206 std::lock_guard<
std::mutex> pendingTasksLock(ctx->pendingTasksMutex);
207 ctx->pendingTasks.push_back(
std::move(task));
208 if (!ctx->pendingTasksRunnerActive) {
209 ctx->executorThreadSafeFunc.NonBlockingCall();
210 ctx->pendingTasksRunnerActive =
true;
215QNapi::Symbol getJsWindowsTrackerIsClosingPropSymbol(
JsState &jsState)
221 return jsState.getJsSymbolForType<SymbolTag>();
224QNapi::Object loadJsModuleViaNapiOrFail(napi_env env,
const std::string &moduleName)
226 qOhosPrintfDebug(
"%s: loading napi module '%s' via napi_load_module()", Q_FUNC_INFO, moduleName.c_str());
228 napi_value result =
nullptr;
229 auto status = napi_load_module(env, moduleName.c_str(), &result);
230 if (status != napi_ok) {
231 qOhosReportFatalErrorAndAbort(
232 "%s: napi_load_module failed for module '%s' (status=%d)",
233 Q_FUNC_INFO, moduleName.c_str(),
static_cast<
int>(status));
236 return QNapi::Object(env, result);
239QNapi::Object loadJsModuleViaEtsFactoryOrFail(
240 const QNapi::Function &etsModuleFactory,
const std::string &moduleName)
242 qOhosPrintfDebug(
"%s: loading napi module '%s' via ets factory", Q_FUNC_INFO, moduleName.c_str());
244 auto result = etsModuleFactory.Call({});
245 if (!result.IsObject()) {
246 qOhosReportFatalErrorAndAbort(
247 "%s: ets factory for module '%s' did not return an object",
248 Q_FUNC_INFO, moduleName.c_str());
251 return QNapi::checkedCast<QNapi::Object>(result);
255std::function<T(
JsState &)> makeMemoizingJsValueSupplier(
std::function<T(
JsState &)> baseJsSupplier)
257 auto memoizedValue = moveToSharedPtr(QNapi::Reference<T>());
258 return [baseJsSupplier =
std::move(baseJsSupplier), memoizedValue](
JsState &jsState)
mutable {
259 if (memoizedValue->IsEmpty()) {
260 *memoizedValue = QNapi::Reference<T>::makePersistentFrom(baseJsSupplier(jsState));
261 baseJsSupplier =
nullptr;
263 return memoizedValue->Value();
267class JsStateImpl :
public JsState
270 JsStateImpl() =
default;
273 napi_env env, std::map<std::string, QNapi::Reference<QNapi::Function>> &&etsModulesFactories,
274 std::shared_ptr<AppFunctions> appFunctions, QtRunMode qtRunMode);
278 void addQAbilityPeerInJsThread(
std::shared_ptr<
QAbilityPeer> qAbilityPeer);
279 void removeMatchingQAbilityPeerInJsThread(QNapi::Object qAbility);
281 void dispatchNewWantInJsThread(QNapi::Object want, QNapi::Object launchParam);
283 void invokeTask(
std::function<
void(
JsState &)> &&task);
285 napi_env env()
override;
286 QNapi::Object getModule(
const std::string &moduleName)
override;
288 QNapi::Object appLaunchWant()
override;
289 QOhosOptional<QNapi::Object> optAppLaunchParam()
override;
291 QNapi::Object defaultWindowStageOrEmpty()
override;
292 QNapi::Object defaultUiContextOrEmpty()
override;
295 std::shared_ptr<
QAbilityPeer> tryGetQAbilityPeerByInstanceId(
const std::string &instanceId)
override;
296 std::shared_ptr<
QAbilityPeer> tryGetQAbilityPeerByInstance(QNapi::Object qAbility)
override;
297 std::shared_ptr<
QAbilityPeer> tryGetQAbilityPeerByQWindow(QObjectThreadSafeRef qwindow)
override;
299 void visitEachQAbilityPeer(
const std::function<
void(
std::shared_ptr<
QAbilityPeer>)> &visitor)
override;
301 void startNewQAbilityInstance(
302 std::shared_ptr<
QAbilityPeer> baseQAbilityPeer, QObjectThreadSafeRef qwindow,
303 QNapi::Object optStartOptions,
306 void startAppProcess(
307 const std::string &processId, QNapi::Object requestWant,
308 QNapi::Object optStartOptions)
override;
310 void startAppProcess(
311 const std::string &processId, QNapi::Object requestWant,
312 QNapi::Object optStartOptions,
std::function<
void(
JsState &)> continueFunc)
override;
314 void addNewWantConsumer(QOhosConsumer<JsState &, QNapi::Object, QNapi::Object> wantConsumer)
override;
316 void startNoUiChildProcess(
const std::string &libraryName,
const std::vector<std::string> &args)
override;
321 template<
typename PeerMatchFunc>
322 std::shared_ptr<
QAbilityPeer> tryFindMatchingQAbilityPeer(PeerMatchFunc &&matchFunc);
324 void *getAttachedObjectWithLazyCreate(
325 const std::type_info &objectTypeInfo, QOhosSupplier<std::shared_ptr<
void>> objectFactory)
override;
327 std::tuple<QNapi::Object, std::string> extractModuleFromEvalExpr(
const std::string &expr)
override;
328 QNapi::Number mapOhosEnumToJs(
int enumValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
override;
329 std::optional<
int> tryMapOhosEnumFromJs(QNapi::Number enumJsValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
override;
330 int mapOhosEnumFromJs(QNapi::Number enumJsValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
override;
331 QNapi::Symbol getJsSymbolForType(
const std::type_info &typeInfo)
override;
333 const std::vector<std::pair<
int,
double>> &getOhosEnumEnumerators(
334 const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)());
335 std::vector<std::pair<
int,
double>> resolveOhosEnumEnumerators(
const OhosEnumInfo &enumInfo);
337 void forceLoadAllJsModulesIfRequested();
339 pthread_t m_jsThread = {};
340 napi_env m_env =
nullptr;
341 QNapi::Reference<QNapi::Object> m_appLaunchWant;
342 QNapi::Reference<QNapi::Object> m_optAppLaunchParam;
344 std::map<std::string, std::shared_ptr<QAbilityPeer>> m_qAbilityPeers;
345 std::map<std::string, std::function<QNapi::Object(JsState &)>> m_jsModulesFactories;
347 PreQueuingJsTasksExecutor m_tasksExecutor;
348 std::vector<QOhosConsumer<JsState &, QNapi::Object, QNapi::Object>> m_newWantConsumers;
349 std::map<std::type_index, std::vector<std::pair<
int,
double>>> m_ohosEnumsEnumerators;
350 std::map<std::type_index, QNapi::Reference<QNapi::Symbol>> m_jsSymbolsRefs;
352 std::map<std::type_index, std::shared_ptr<
void>> m_attachedObjects;
355bool JsStateImpl::isJsThread()
357 return pthread_self() == m_jsThread;
360void JsStateImpl::initInJsThread(
361 napi_env env, std::map<std::string, QNapi::Reference<QNapi::Function>> &&etsModulesFactories,
362 std::shared_ptr<AppFunctions> appFunctions, QtRunMode qtRunMode)
364 m_jsThread = pthread_self();
366 m_defaultQAbilityPeer = std::make_shared<DummyQAbilityPeer>();
367 m_appFunctions = appFunctions;
368 m_qtRunMode = qtRunMode;
370 for (
const char *moduleName : napiLoadableModules) {
371 m_jsModulesFactories.emplace(
373 makeMemoizingJsValueSupplier<QNapi::Object>(
374 [moduleName](JsState &jsState) {
375 return loadJsModuleViaNapiOrFail(jsState.env(), moduleName);
378 for (
auto &etsModulesFactoryEntry : etsModulesFactories) {
379 const std::string &moduleName = etsModulesFactoryEntry.first;
380 auto etsModuleFactory = moveToSharedPtr(std::move(etsModulesFactoryEntry.second));
381 m_jsModulesFactories.emplace(
383 makeMemoizingJsValueSupplier<QNapi::Object>(
384 [moduleName, etsModuleFactory](JsState &) {
385 return loadJsModuleViaEtsFactoryOrFail(etsModuleFactory->Value(), moduleName);
388 m_jsModulesFactories.emplace(
390 [](JsState &jsState) -> QNapi::Object {
391 return Napi::Env(jsState.env()).Global();
394 m_tasksExecutor.setUnderlyingExecutor(makeJsThreadTasksExecutor(
this));
397void JsStateImpl::addQAbilityPeerInJsThread(
std::shared_ptr<
QAbilityPeer> qAbilityPeer)
399 forceLoadAllJsModulesIfRequested();
401 if (m_appLaunchWant.IsEmpty())
402 m_appLaunchWant = Napi::Persistent(qAbilityPeer->launchWant());
405 if (m_optAppLaunchParam.IsEmpty() && optQUiAbilityPeer) {
406 m_optAppLaunchParam = QNapi::Reference<>::makePersistentFrom(optQUiAbilityPeer->launchParam());
408 "JsStateImpl: appLaunchParam set from Ability with instanceId='%s'",
409 qAbilityPeer->instanceId().c_str());
412 if (m_qAbilityPeers.empty())
413 m_defaultQAbilityPeer = qAbilityPeer;
415 m_qAbilityPeers.emplace(qAbilityPeer->instanceId(), qAbilityPeer);
418 "JsStateImpl: added QAbilityPeer, instanceId: %s, qwindow: %s",
419 qAbilityPeer->instanceId().c_str(), qAbilityPeer->qWindowRef().refName().c_str());
422void JsStateImpl::removeMatchingQAbilityPeerInJsThread(QNapi::Object qAbility)
424 Napi::HandleScope funcScope{qAbility.Env()};
426 auto foundPeerIter =
std::find_if(
427 m_qAbilityPeers.begin(), m_qAbilityPeers.end(),
428 [&](
const auto &entry) {
429 return entry.second->qAbility() == qAbility;
432 if (foundPeerIter != m_qAbilityPeers.end()) {
433 auto removedPeer = foundPeerIter->second;
434 m_qAbilityPeers.erase(foundPeerIter);
435 if (m_defaultQAbilityPeer == removedPeer) {
436 m_defaultQAbilityPeer =
437 !m_qAbilityPeers.empty()
438 ? m_qAbilityPeers.begin()->second
439 : std::make_shared<DummyQAbilityPeer>();
442 "JsStateImpl: removed QAbilityPeer, instanceId: %s, qwindow: %s",
443 removedPeer->instanceId().c_str(), removedPeer->qWindowRef().refName().c_str());
447void JsStateImpl::dispatchNewWantInJsThread(QNapi::Object want, QNapi::Object launchParam)
449 const auto consumersCount = m_newWantConsumers.size();
450 for (std::size_t i = 0; i < consumersCount; ++i)
451 m_newWantConsumers[i](*
this, want, launchParam);
454void JsStateImpl::invokeTask(
std::function<
void(
JsState &)> &&task)
456 m_tasksExecutor.invokeTask(std::move(task));
459napi_env JsStateImpl::env()
464void JsStateImpl::forceLoadAllJsModulesIfRequested()
466 if (qEnvironmentVariableIntValue(forceLoadJsModulesEnvVariableName) == 0)
469 qOhosPrintfDebug(
"%s: forcibly loading all JS modules", Q_FUNC_INFO);
471 for (
const auto &moduleFactoryEntry : m_jsModulesFactories)
472 getModule(moduleFactoryEntry.first);
475QNapi::Object JsStateImpl::getModule(
const std::string &moduleName)
477 using namespace std::string_literals;
479 auto moduleFactoryIter = m_jsModulesFactories.find(moduleName);
480 if (moduleFactoryIter == m_jsModulesFactories.end())
481 throw QNapi::makeLoggedException(m_env,
"JS module not found: "s + moduleName);
483 return moduleFactoryIter->second(*
this);
486QNapi::Object JsStateImpl::appLaunchWant()
488 return m_appLaunchWant.Value();
491QOhosOptional<QNapi::Object> JsStateImpl::optAppLaunchParam()
493 return !m_optAppLaunchParam.IsEmpty()
494 ? makeQOhosOptional(m_optAppLaunchParam.Value())
495 : makeEmptyQOhosOptional();
498QNapi::Object JsStateImpl::defaultWindowStageOrEmpty()
502 return QNapi::Object();
504 return qUiAbilityPeer->windowStage();
507QNapi::Object JsStateImpl::defaultUiContextOrEmpty()
509 return m_defaultQAbilityPeer->uiContext();
514 return m_defaultQAbilityPeer;
517std::shared_ptr<
QAbilityPeer> JsStateImpl::tryGetQAbilityPeerByInstanceId(
const std::string &instanceId)
519 auto foundIter = m_qAbilityPeers.find(instanceId);
520 return foundIter != m_qAbilityPeers.end() ? foundIter->second :
nullptr;
523std::shared_ptr<
QAbilityPeer> JsStateImpl::tryGetQAbilityPeerByInstance(QNapi::Object qAbility)
525 Napi::HandleScope findPeerScope(qAbility.Env());
526 return tryFindMatchingQAbilityPeer(
527 [&](
const auto &peer) {
528 return peer->qAbility() == qAbility;
532std::shared_ptr<
QAbilityPeer> JsStateImpl::tryGetQAbilityPeerByQWindow(QObjectThreadSafeRef qwindow)
534 return tryFindMatchingQAbilityPeer(
535 [&](
const auto &peer) {
536 return peer->qWindowRef() == qwindow;
540void JsStateImpl::visitEachQAbilityPeer(
const std::function<
void(
std::shared_ptr<
QAbilityPeer>)> &visitor)
542 for (
const auto &qAbilityPeerEntry : m_qAbilityPeers)
543 visitor(qAbilityPeerEntry.second);
546void JsStateImpl::startNewQAbilityInstance(
547 std::shared_ptr<
QAbilityPeer> baseQAbilityPeer, QObjectThreadSafeRef qwindow,
548 QNapi::Object optStartOptions,
551 auto baseQAbility = baseQAbilityPeer->qAbility();
552 if (!baseQAbility.IsEmpty()) {
553 m_appFunctions->startQAbilityInstance(
554 baseQAbility, qwindow, optStartOptions, std::move(startupNotifyFunc));
558void JsStateImpl::startAppProcess(
559 const std::string &processId, QNapi::Object requestWant, QNapi::Object optStartOptions)
561 auto baseQAbility = defaultQAbilityPeer()->qAbility();
562 if (!baseQAbility.IsEmpty())
563 m_appFunctions->startAppProcess(baseQAbility, processId, requestWant, optStartOptions);
566void JsStateImpl::startAppProcess(
567 const std::string &processId, QNapi::Object requestWant,
568 QNapi::Object optStartOptions,
std::function<
void(
JsState &)> continueFunc)
570 auto baseQAbility = defaultQAbilityPeer()->qAbility();
571 if (!baseQAbility.IsEmpty()) {
572 m_appFunctions->startAppProcess(
573 baseQAbility, processId, requestWant, optStartOptions, std::move(continueFunc));
575 invokeTask(
std::move(continueFunc));
579void JsStateImpl::addNewWantConsumer(QOhosConsumer<JsState &, QNapi::Object, QNapi::Object> wantConsumer)
581 m_newWantConsumers.push_back(std::move(wantConsumer));
584void JsStateImpl::startNoUiChildProcess(
const std::string &libraryName,
const std::vector<std::string> &args)
586 m_appFunctions->startNoUiChildProcess(*
this, libraryName, args);
594template<
typename PeerMatchFunc>
595std::shared_ptr<
QAbilityPeer> JsStateImpl::tryFindMatchingQAbilityPeer(PeerMatchFunc &&matchFunc)
597 auto foundIter =
std::find_if(
598 m_qAbilityPeers.begin(), m_qAbilityPeers.end(),
599 [&](
const auto &entry) {
600 return matchFunc(entry.second);
602 return foundIter != m_qAbilityPeers.end() ? foundIter->second :
nullptr;
605void *JsStateImpl::getAttachedObjectWithLazyCreate(
606 const std::type_info &objectTypeInfo, QOhosSupplier<std::shared_ptr<
void>> objectFactory)
608 auto objectIter = m_attachedObjects.find(objectTypeInfo);
609 if (objectIter == m_attachedObjects.end())
610 std::tie(objectIter, std::ignore) = m_attachedObjects.emplace(objectTypeInfo, objectFactory());
612 return objectIter->second.get();
615std::tuple<QNapi::Object, std::string> JsStateImpl::extractModuleFromEvalExpr(
const std::string &expr)
617 using namespace std::string_literals;
619 std::size_t longestModulePrefixSize = 0;
620 auto considerAsLongestPrefix = [&](
const std::string &name) {
621 if (name.size() > longestModulePrefixSize
622 && expr.compare(0, name.size(), name) == 0
623 && (expr.size() == name.size() || expr[name.size()] ==
'.')) {
624 longestModulePrefixSize = name.size();
627 for (
const auto &moduleFactoryEntry : m_jsModulesFactories)
628 considerAsLongestPrefix(moduleFactoryEntry.first);
630 if (longestModulePrefixSize == 0) {
631 throw QNapi::makeLoggedException(
632 env(),
"global expression doesn't start with known module path: '"s + expr +
"'");
635 return std::make_tuple(
636 getModule(expr.substr(0, longestModulePrefixSize)),
637 longestModulePrefixSize < expr.size()
638 ? expr.substr(longestModulePrefixSize + 1)
642QNapi::Number JsStateImpl::mapOhosEnumToJs(
643 int enumValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
645 using namespace std::string_literals;
647 const auto &enumeratorsValues = getOhosEnumEnumerators(enumTypeInfo, ohosEnumInfoFactory);
649 auto valueIter =
std::find_if(
650 enumeratorsValues.begin(), enumeratorsValues.end(),
651 [&](
const auto &valuePair) {
652 return valuePair.first == enumValue;
655 if (valueIter == enumeratorsValues.end()) {
656 throw QNapi::makeLoggedException(
658 "Illegal C++ value of enumerator for enum '"s + enumTypeInfo.name() +
"': "
659 + std::to_string(enumValue));
662 return QNapi::Number::New(env(), valueIter->second);
665std::optional<
int> JsStateImpl::tryMapOhosEnumFromJs(
666 QNapi::Number enumJsValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
668 double enumDoubleJsValue = enumJsValue.DoubleValue();
670 const auto &enumeratorsValues = getOhosEnumEnumerators(enumTypeInfo, ohosEnumInfoFactory);
672 auto valueIter =
std::find_if(
673 enumeratorsValues.begin(), enumeratorsValues.end(),
674 [&](
const auto &valuePair) {
675 return valuePair.second == enumDoubleJsValue;
678 return valueIter != enumeratorsValues.end()
679 ?
std::optional(valueIter->first)
683int JsStateImpl::mapOhosEnumFromJs(
684 QNapi::Number enumJsValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
686 using namespace std::string_literals;
688 auto optEnumValue = tryMapOhosEnumFromJs(enumJsValue, enumTypeInfo, ohosEnumInfoFactory);
689 if (!optEnumValue.has_value()) {
690 throw QNapi::makeLoggedException(
692 "Illegal Napi value of enumerator for enum '"s + enumTypeInfo.name() +
"': "
693 + std::to_string(enumJsValue.DoubleValue()));
695 return optEnumValue.value();
698QNapi::Symbol JsStateImpl::getJsSymbolForType(
const std::type_info &typeInfo)
700 using namespace std::string_literals;
702 auto jsSymbolRefIter = m_jsSymbolsRefs.find(typeInfo);
703 if (jsSymbolRefIter == m_jsSymbolsRefs.end()) {
704 std::tie(jsSymbolRefIter, std::ignore) = m_jsSymbolsRefs.emplace(
706 QNapi::Reference<>::makePersistentFrom(
707 QNapi::Symbol::New(env(),
"_io_qt_"s + typeInfo.name())));
709 return jsSymbolRefIter->second.Value();
712const std::vector<std::pair<
int,
double>> &JsStateImpl::getOhosEnumEnumerators(
713 const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
715 auto enumeratorsIter = m_ohosEnumsEnumerators.find(enumTypeInfo);
716 if (enumeratorsIter == m_ohosEnumsEnumerators.end()) {
717 std::tie(enumeratorsIter, std::ignore) = m_ohosEnumsEnumerators.emplace(
718 enumTypeInfo, resolveOhosEnumEnumerators(ohosEnumInfoFactory()));
721 return enumeratorsIter->second;
724std::vector<std::pair<
int,
double>> JsStateImpl::resolveOhosEnumEnumerators(
const OhosEnumInfo &enumInfo)
726 using namespace std::string_literals;
728 QNapi::Object enumObj;
730 enumObj = eval<QNapi::Object>(enumInfo.fullTypeName);
731 }
catch (
const Napi::Error &e) {
732 throw QNapi::makeLoggedException(
733 env(),
"JS enum '"s + enumInfo.fullTypeName +
"' not found: "s + e.what());
736 std::vector<std::pair<
int,
double>> enumeratorsValues;
738 for (
const auto &enumNamePair : enumInfo.enumeratorsNames) {
739 const auto *enumeratorName = enumNamePair.second;
741 QNapi::Value jsEnumeratorValue = QNapi::getPropOrUndefined(enumObj, enumeratorName);
743 if (!jsEnumeratorValue.IsNumber()) {
744 throw QNapi::makeLoggedException(
745 env(),
"JS enum '"s + enumInfo.fullTypeName +
"' doesn't contain: "s + enumeratorName);
748 enumeratorsValues.emplace_back(
749 enumNamePair.first, QNapi::checkedCast<QNapi::Number>(jsEnumeratorValue).DoubleValue());
752 return enumeratorsValues;
755JsStateImpl &getJsStateImpl()
757 static JsStateImpl jsStateImpl;
763 static auto qtJsBlockingCallsGateway = QOhosMtBlockingCallsGateway<JsState>::makeInstance(
764 invokeInQtThread, invokeInJsThread);
765 return *qtJsBlockingCallsGateway;
768class JsThreadOpsImpl :
public ::QOhosJsThreadOps
771 QOhosJsState &jsState()
override;
773 void invoke(std::function<
void(QOhosJsState &)> task)
override;
774 void invokeAndWaitForContinue(
775 std::function<
void(QOhosJsState &, std::function<
void()>)> &&task)
override;
776 void runAndWait(
const std::function<
void(QOhosJsState &)> &task)
override;
779QOhosJsState &JsThreadOpsImpl::jsState()
781 return getJsStateImpl();
784void JsThreadOpsImpl::invoke(std::function<
void(QOhosJsState &)> task)
789void JsThreadOpsImpl::invokeAndWaitForContinue(
790 std::function<
void(QOhosJsState &, std::function<
void()>)> &&task)
795void JsThreadOpsImpl::runAndWait(
const std::function<
void(QOhosJsState &)> &task)
816 return matchTypeIdObject == &typeIdObject ?
this :
nullptr;
826 qAbilityPeer->tryCastWithTypeIdObject(&typeIdObject));
828 return qUiAbilityPeer !=
nullptr
829 ? qUiAbilityPeer->shared_from_this()
843 return getJsStateImpl();
848 qOhosPrintfDebug(
"Tagging JS Window as closing (from: %s)", logContext);
849 Napi::HandleScope funcScope{jsWindow.Env()};
851 getJsWindowsTrackerIsClosingPropSymbol(getJsStateImpl()),
852 QNapi::Boolean::New(jsWindow.Env(),
true));
857 Napi::HandleScope funcScope{jsWindow.Env()};
858 auto isClosingPropSymbol = getJsWindowsTrackerIsClosingPropSymbol(getJsStateImpl());
859 auto isClosingValue = jsWindow.Has(isClosingPropSymbol)
860 ? jsWindow.Get(isClosingPropSymbol)
862 return isClosingValue.IsBoolean() && QNapi::checkedCast<QNapi::Boolean>(isClosingValue).Value();
869 static JsThreadOpsImpl jsThreadOpsImpl;
870 QOhosJsThreadOps::registerInstance(&jsThreadOpsImpl);
872 getJsStateImpl().initInJsThread(env,
std::move(jsModulesFactories), appFunctions, qtRunMode);
877 getJsStateImpl().addQAbilityPeerInJsThread(qAbilityPeer);
882 getJsStateImpl().removeMatchingQAbilityPeerInJsThread(qAbility);
887 getJsStateImpl().dispatchNewWantInJsThread(want, launchParam);
892 getJsStateImpl().invokeTask(
std::move(task));
897 if (getQtState().isQtThread()) {
898 getQtJsBlockingCallsGateway().runInSlaveThreadAndWaitForContinue(
std::move(task));
900 auto taskReadyPromise =
std::make_shared<
std::promise<
void>>();
901 auto taskReadyFuture = taskReadyPromise->get_future();
903 auto continueFunc = [taskReadyPromise]() {
904 taskReadyPromise->set_value();
908 [task = std::move(task), continueFunc = std::move(continueFunc)](
JsState &jsState)
mutable {
909 task(jsState,
std::move(continueFunc));
911 taskReadyFuture.wait();
917 if (getJsStateImpl().isJsThread()) {
918 task(getJsStateImpl());
920 invokeInJsThreadAndWaitForContinue(
921 [&](
auto &jsState,
auto continueFunc) {
929 std::function<
void(
std::function<
void()>)> &&task,
930 std::chrono::nanoseconds timeout)
932 if (getJsStateImpl().isJsThread()) {
933 auto result = getQtJsBlockingCallsGateway().tryInvokeInMasterThreadAndTryWaitForContinue(
std::move(task), timeout);
934 return result == QOhosMtBlockingCallsGateway<JsState>::MasterThreadTaskResult::Finished;
936 auto taskReadyPromise =
std::make_shared<
std::promise<
void>>();
937 auto taskReadyFuture = taskReadyPromise->get_future();
939 auto continueFunc = [taskReadyPromise]() {
940 taskReadyPromise->set_value();
944 [task = std::move(task), continueFunc = std::move(continueFunc)]()
mutable {
945 task(
std::move(continueFunc));
947 auto waitResult = taskReadyFuture.wait_for(timeout);
949 return waitResult ==
std::future_status::ready;
JsState & jsState() const
virtual ~QAbilityEngine()
void * tryCastWithTypeIdObject(const void *matchTypeIdObject) final
static std::shared_ptr< QUiAbilityPeer > tryCastFromQAbilityPeerOrNull(std::shared_ptr< QAbilityPeer > qAbilityPeer)
~QUiAbilityPeer() override
void invokeInJsThread(std::function< void(JsState &)> task)
void dispatchNewWant(QNapi::Object want, QNapi::Object launchParam)
void removeMatchingJsQAbilityPeer(QNapi::Object qAbility)
void initJsThreadState(napi_env env, std::map< std::string, QNapi::Reference< QNapi::Function > > &&jsModulesFactories, std::shared_ptr< AppFunctions > appFunctions, QtRunMode qtRunMode)
Q_REQUIRED_RESULT bool tryInvokeInQtThreadAndTryWaitForContinue(std::function< void(std::function< void()>)> &&task, std::chrono::nanoseconds timeout)
void runInJsThreadAndWait(const std::function< void(JsState &)> &task)
void invokeInJsThreadAndWaitForContinue(std::function< void(JsState &, std::function< void()>)> &&task)
void addJsQAbilityPeer(std::shared_ptr< QAbilityPeer > qAbilityPeer)