4#include <qohosplugincore.h>
7#include <QtCore/private/qohoslogger_p.h>
14#include <napi/native_api.h>
28constexpr const char *forceLoadJsModulesEnvVariableName =
"IO__QT__OHOS__FORCE_LOAD_JS_MODULES";
30constexpr const char *napiLoadableModules[] = {
31 "@ohos.abilityAccessCtrl",
32 "@ohos.accessibility",
33 "@ohos.app.ability.AbilityConstant",
34 "@ohos.app.ability.ConfigurationConstant",
35 "@ohos.app.ability.childProcessManager",
36 "@ohos.app.ability.contextConstant",
37 "@ohos.app.ability.wantConstant",
38 "@ohos.arkui.uiExtension",
39 "@ohos.bluetooth.access",
40 "@ohos.bluetooth.connection",
41 "@ohos.bluetooth.socket",
42 "@ohos.bundle.bundleManager",
43 "@ohos.data.uniformTypeDescriptor",
49 "@ohos.geoLocationManager",
50 "@ohos.graphics.text",
54 "@ohos.multimedia.image",
55 "@ohos.multimedia.media",
56 "@ohos.multimodalInput.inputDevice",
57 "@ohos.multimodalInput.pointer",
58 "@ohos.net.connection",
59 "@ohos.notificationManager",
70 DummyQAbilityPeer() =
default;
72 std::string instanceId()
override;
73 QNapi::Object uiContext()
override;
74 QNapi::Object qAbility()
override;
75 QNapi::Object launchWant()
override;
76 QObjectThreadSafeRef qWindowRef()
override;
77 QOhosOptional<QNapi::Promise> qWindowDestroyPromise()
override;
78 void forceResolveQWindowDestroyPromiseIfPresent(Napi::Env env)
override;
79 std::shared_ptr<
std::atomic_bool> destroyAllowedFlag()
override;
80 bool isTerminating()
override;
82 void setQWindow(Napi::Env env, QObjectThreadSafeRef)
override;
84 void *tryCastWithTypeIdObject(
const void *matchTypeIdObject)
override;
87std::string DummyQAbilityPeer::instanceId()
92QNapi::Object DummyQAbilityPeer::uiContext()
94 return QNapi::Object();
97QNapi::Object DummyQAbilityPeer::qAbility()
99 return QNapi::Object();
102QNapi::Object DummyQAbilityPeer::launchWant()
104 return QNapi::Object();
107QObjectThreadSafeRef DummyQAbilityPeer::qWindowRef()
112QOhosOptional<QNapi::Promise> DummyQAbilityPeer::qWindowDestroyPromise()
117void DummyQAbilityPeer::forceResolveQWindowDestroyPromiseIfPresent(Napi::Env)
121std::shared_ptr<
std::atomic_bool> DummyQAbilityPeer::destroyAllowedFlag()
123 return std::make_shared<
std::atomic_bool>(
false);
126bool DummyQAbilityPeer::isTerminating()
131void DummyQAbilityPeer::setQWindow(Napi::Env, QObjectThreadSafeRef)
135void *DummyQAbilityPeer::tryCastWithTypeIdObject(
const void *)
140template<
typename Task>
141class PreQueuingTasksExecutor
144 void setUnderlyingExecutor(
145 std::function<
void(Task)> executor,
146 std::function<
void(Task)> optSyncFlushExecutor =
nullptr);
147 void invokeTask(Task task);
150 std::recursive_mutex m_executorMutex;
151 std::function<
void(Task)> m_optUnderlyingExecutor;
152 std::deque<Task> m_pendingTasks;
155template<
typename Task>
156void PreQueuingTasksExecutor<Task>::setUnderlyingExecutor(
157 std::function<
void(Task)> executor,
std::function<
void(Task)> optSyncFlushExecutor)
159 std::lock_guard<
std::recursive_mutex> executorLock(m_executorMutex);
160 auto &flushExecutor = optSyncFlushExecutor ? optSyncFlushExecutor : executor;
161 while (!m_pendingTasks.empty()) {
162 auto task =
std::move(m_pendingTasks.front());
163 m_pendingTasks.pop_front();
164 flushExecutor(
std::move(task));
166 m_optUnderlyingExecutor =
std::move(executor);
169template<
typename Task>
170void PreQueuingTasksExecutor<Task>::invokeTask(Task task)
172 std::lock_guard<
std::recursive_mutex> executorLock(m_executorMutex);
173 if (m_optUnderlyingExecutor)
174 m_optUnderlyingExecutor(
std::move(task));
176 m_pendingTasks.push_back(std::move(task));
179using PreQueuingJsTasksExecutor = PreQueuingTasksExecutor<
std::function<
void(
JsState &)>>;
184 std::mutex pendingTasksMutex;
185 bool pendingTasksRunnerActive =
false;
186 std::vector<std::function<
void(JsState &)>> pendingTasks;
187 Napi::ThreadSafeFunction executorThreadSafeFunc;
189 auto ctx =
std::make_shared<Context>();
191 ctx->executorThreadSafeFunc = Napi::ThreadSafeFunction::New(
193 QNapi::Function::New(
195 [ctx = ctx.get()](
const CallbackInfo &cbInfo) {
197 std::vector<std::function<
void(JsState &)>> pendingTasks;
199 std::lock_guard<std::mutex> pendingTasksLock(ctx->pendingTasksMutex);
200 pendingTasks = std::exchange(ctx->pendingTasks, {});
201 if (pendingTasks.empty()) {
202 ctx->pendingTasksRunnerActive =
false;
206 for (
auto &task : pendingTasks)
207 task(cbInfo.jsState());
210 "QCoreOhosTasksExecutorFunc"),
211 "QCoreOhosTasksExecutor", 0, 1);
213 return [ctx](
std::function<
void(
JsState &)> task) {
214 std::lock_guard<
std::mutex> pendingTasksLock(ctx->pendingTasksMutex);
215 ctx->pendingTasks.push_back(
std::move(task));
216 if (!ctx->pendingTasksRunnerActive) {
217 ctx->executorThreadSafeFunc.NonBlockingCall();
218 ctx->pendingTasksRunnerActive =
true;
223QNapi::Symbol getJsWindowsTrackerIsClosingPropSymbol(
JsState &jsState)
229 return jsState.getJsSymbolForType<SymbolTag>();
232QNapi::Object loadJsModuleViaNapiOrFail(napi_env env,
const std::string &moduleName)
234 qOhosPrintfDebug(
"%s: loading napi module '%s' via napi_load_module()", Q_FUNC_INFO, moduleName.c_str());
236 napi_value result =
nullptr;
237 auto status = napi_load_module(env, moduleName.c_str(), &result);
238 if (status != napi_ok) {
239 qOhosReportFatalErrorAndAbort(
240 "%s: napi_load_module failed for module '%s' (status=%d)",
241 Q_FUNC_INFO, moduleName.c_str(),
static_cast<
int>(status));
244 return QNapi::Object(env, result);
247QNapi::Object loadJsModuleViaEtsFactoryOrFail(
248 const QNapi::Function &etsModuleFactory,
const std::string &moduleName)
250 qOhosPrintfDebug(
"%s: loading napi module '%s' via ets factory", Q_FUNC_INFO, moduleName.c_str());
252 auto result = etsModuleFactory.Call({});
253 if (!result.IsObject()) {
254 qOhosReportFatalErrorAndAbort(
255 "%s: ets factory for module '%s' did not return an object",
256 Q_FUNC_INFO, moduleName.c_str());
259 return QNapi::checkedCast<QNapi::Object>(result);
263std::function<T(
JsState &)> makeMemoizingJsValueSupplier(
std::function<T(
JsState &)> baseJsSupplier)
265 auto memoizedValue = moveToSharedPtr(QNapi::Reference<T>());
266 return [baseJsSupplier =
std::move(baseJsSupplier), memoizedValue](
JsState &jsState)
mutable {
267 if (memoizedValue->IsEmpty()) {
268 *memoizedValue = QNapi::Reference<T>::makePersistentFrom(baseJsSupplier(jsState));
269 baseJsSupplier =
nullptr;
271 return memoizedValue->Value();
275class JsStateImpl :
public JsState
278 JsStateImpl() =
default;
281 napi_env env, std::map<std::string, QNapi::Reference<QNapi::Function>> &&etsModulesFactories,
282 std::shared_ptr<AppFunctions> appFunctions, QtRunMode qtRunMode);
286 void addQAbilityPeerInJsThread(
std::shared_ptr<
QAbilityPeer> qAbilityPeer);
287 void removeMatchingQAbilityPeerInJsThread(QNapi::Object qAbility);
289 void dispatchNewWantInJsThread(QNapi::Object want, QNapi::Object launchParam);
291 void invokeTask(
std::function<
void(
JsState &)> &&task);
293 napi_env env()
override;
294 QNapi::Object getModule(
const std::string &moduleName)
override;
296 QNapi::Object appLaunchWant()
override;
297 QOhosOptional<QNapi::Object> optAppLaunchParam()
override;
299 QNapi::Object defaultWindowStageOrEmpty()
override;
300 QNapi::Object defaultUiContextOrEmpty()
override;
303 std::shared_ptr<
QAbilityPeer> tryGetQAbilityPeerByInstanceId(
const std::string &instanceId)
override;
304 std::shared_ptr<
QAbilityPeer> tryGetQAbilityPeerByInstance(QNapi::Object qAbility)
override;
305 std::shared_ptr<
QAbilityPeer> tryGetQAbilityPeerByQWindow(QObjectThreadSafeRef qwindow)
override;
307 void visitEachQAbilityPeer(
const std::function<
void(
std::shared_ptr<
QAbilityPeer>)> &visitor)
override;
309 void startNewQAbilityInstance(
310 std::shared_ptr<
QAbilityPeer> baseQAbilityPeer, QObjectThreadSafeRef qwindow,
311 QNapi::Object optStartOptions,
314 void startAppProcess(
315 const std::string &processId, QNapi::Object requestWant,
316 QNapi::Object optStartOptions)
override;
318 void startAppProcess(
319 const std::string &processId, QNapi::Object requestWant,
320 QNapi::Object optStartOptions,
std::function<
void(
JsState &)> continueFunc)
override;
322 void addNewWantConsumer(QOhosConsumer<JsState &, QNapi::Object, QNapi::Object> wantConsumer)
override;
324 void startNoUiChildProcess(
const std::string &libraryName,
const std::vector<std::string> &args)
override;
329 template<
typename PeerMatchFunc>
330 std::shared_ptr<
QAbilityPeer> tryFindMatchingQAbilityPeer(PeerMatchFunc &&matchFunc);
332 void *getAttachedObjectWithLazyCreate(
333 const std::type_info &objectTypeInfo, QOhosSupplier<std::shared_ptr<
void>> objectFactory)
override;
335 std::tuple<QNapi::Object, std::string> extractModuleFromEvalExpr(
const std::string &expr)
override;
336 QNapi::Number mapOhosEnumToJs(
int enumValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
override;
337 std::optional<
int> tryMapOhosEnumFromJs(QNapi::Number enumJsValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
override;
338 int mapOhosEnumFromJs(QNapi::Number enumJsValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
override;
339 QNapi::Symbol getJsSymbolForType(
const std::type_info &typeInfo)
override;
341 const std::vector<std::pair<
int,
double>> &getOhosEnumEnumerators(
342 const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)());
343 std::vector<std::pair<
int,
double>> resolveOhosEnumEnumerators(
const OhosEnumInfo &enumInfo);
345 void forceLoadAllJsModulesIfRequested();
347 pthread_t m_jsThread = {};
348 napi_env m_env =
nullptr;
349 QNapi::Reference<QNapi::Object> m_appLaunchWant;
350 QNapi::Reference<QNapi::Object> m_optAppLaunchParam;
352 std::map<std::string, std::shared_ptr<QAbilityPeer>> m_qAbilityPeers;
353 std::map<std::string, std::function<QNapi::Object(JsState &)>> m_jsModulesFactories;
355 PreQueuingJsTasksExecutor m_tasksExecutor;
356 std::vector<QOhosConsumer<JsState &, QNapi::Object, QNapi::Object>> m_newWantConsumers;
357 std::map<std::type_index, std::vector<std::pair<
int,
double>>> m_ohosEnumsEnumerators;
358 std::map<std::type_index, QNapi::Reference<QNapi::Symbol>> m_jsSymbolsRefs;
360 std::map<std::type_index, std::shared_ptr<
void>> m_attachedObjects;
363bool JsStateImpl::isJsThread()
365 return pthread_self() == m_jsThread;
368void JsStateImpl::initInJsThread(
369 napi_env env, std::map<std::string, QNapi::Reference<QNapi::Function>> &&etsModulesFactories,
370 std::shared_ptr<AppFunctions> appFunctions, QtRunMode qtRunMode)
372 m_jsThread = pthread_self();
374 m_defaultQAbilityPeer = std::make_shared<DummyQAbilityPeer>();
375 m_appFunctions = appFunctions;
376 m_qtRunMode = qtRunMode;
378 for (
const char *moduleName : napiLoadableModules) {
379 m_jsModulesFactories.emplace(
381 makeMemoizingJsValueSupplier<QNapi::Object>(
382 [moduleName](JsState &jsState) {
383 return loadJsModuleViaNapiOrFail(jsState.env(), moduleName);
386 for (
auto &etsModulesFactoryEntry : etsModulesFactories) {
387 const std::string &moduleName = etsModulesFactoryEntry.first;
388 auto etsModuleFactory = moveToSharedPtr(std::move(etsModulesFactoryEntry.second));
389 m_jsModulesFactories.emplace(
391 makeMemoizingJsValueSupplier<QNapi::Object>(
392 [moduleName, etsModuleFactory](JsState &) {
393 return loadJsModuleViaEtsFactoryOrFail(etsModuleFactory->Value(), moduleName);
396 m_jsModulesFactories.emplace(
398 [](JsState &jsState) -> QNapi::Object {
399 return Napi::Env(jsState.env()).Global();
402 m_tasksExecutor.setUnderlyingExecutor(makeJsThreadTasksExecutor(
this));
405void JsStateImpl::addQAbilityPeerInJsThread(
std::shared_ptr<
QAbilityPeer> qAbilityPeer)
407 forceLoadAllJsModulesIfRequested();
409 if (m_appLaunchWant.IsEmpty())
410 m_appLaunchWant = Napi::Persistent(qAbilityPeer->launchWant());
413 if (m_optAppLaunchParam.IsEmpty() && optQUiAbilityPeer) {
414 m_optAppLaunchParam = QNapi::Reference<>::makePersistentFrom(optQUiAbilityPeer->launchParam());
416 "JsStateImpl: appLaunchParam set from Ability with instanceId='%s'",
417 qAbilityPeer->instanceId().c_str());
420 if (m_qAbilityPeers.empty())
421 m_defaultQAbilityPeer = qAbilityPeer;
423 m_qAbilityPeers.emplace(qAbilityPeer->instanceId(), qAbilityPeer);
426 "JsStateImpl: added QAbilityPeer, instanceId: %s, qwindow: %s",
427 qAbilityPeer->instanceId().c_str(), qAbilityPeer->qWindowRef().refName().c_str());
430void JsStateImpl::removeMatchingQAbilityPeerInJsThread(QNapi::Object qAbility)
432 Napi::HandleScope funcScope{qAbility.Env()};
434 auto foundPeerIter =
std::find_if(
435 m_qAbilityPeers.begin(), m_qAbilityPeers.end(),
436 [&](
const auto &entry) {
437 return entry.second->qAbility() == qAbility;
440 if (foundPeerIter != m_qAbilityPeers.end()) {
441 auto removedPeer = foundPeerIter->second;
442 m_qAbilityPeers.erase(foundPeerIter);
443 if (m_defaultQAbilityPeer == removedPeer) {
444 m_defaultQAbilityPeer =
445 !m_qAbilityPeers.empty()
446 ? m_qAbilityPeers.begin()->second
447 : std::make_shared<DummyQAbilityPeer>();
450 "JsStateImpl: removed QAbilityPeer, instanceId: %s, qwindow: %s",
451 removedPeer->instanceId().c_str(), removedPeer->qWindowRef().refName().c_str());
455void JsStateImpl::dispatchNewWantInJsThread(QNapi::Object want, QNapi::Object launchParam)
457 const auto consumersCount = m_newWantConsumers.size();
458 for (std::size_t i = 0; i < consumersCount; ++i)
459 m_newWantConsumers[i](*
this, want, launchParam);
462void JsStateImpl::invokeTask(
std::function<
void(
JsState &)> &&task)
464 m_tasksExecutor.invokeTask(std::move(task));
467napi_env JsStateImpl::env()
472void JsStateImpl::forceLoadAllJsModulesIfRequested()
474 if (qEnvironmentVariableIntValue(forceLoadJsModulesEnvVariableName) == 0)
477 qOhosPrintfDebug(
"%s: forcibly loading all JS modules", Q_FUNC_INFO);
479 for (
const auto &moduleFactoryEntry : m_jsModulesFactories)
480 getModule(moduleFactoryEntry.first);
483QNapi::Object JsStateImpl::getModule(
const std::string &moduleName)
485 using namespace std::string_literals;
487 auto moduleFactoryIter = m_jsModulesFactories.find(moduleName);
488 if (moduleFactoryIter == m_jsModulesFactories.end())
489 throw QNapi::makeLoggedException(m_env,
"JS module not found: "s + moduleName);
491 return moduleFactoryIter->second(*
this);
494QNapi::Object JsStateImpl::appLaunchWant()
496 return m_appLaunchWant.Value();
499QOhosOptional<QNapi::Object> JsStateImpl::optAppLaunchParam()
501 return !m_optAppLaunchParam.IsEmpty()
502 ? makeQOhosOptional(m_optAppLaunchParam.Value())
503 : makeEmptyQOhosOptional();
506QNapi::Object JsStateImpl::defaultWindowStageOrEmpty()
510 return QNapi::Object();
512 return qUiAbilityPeer->windowStage();
515QNapi::Object JsStateImpl::defaultUiContextOrEmpty()
517 return m_defaultQAbilityPeer->uiContext();
522 return m_defaultQAbilityPeer;
525std::shared_ptr<
QAbilityPeer> JsStateImpl::tryGetQAbilityPeerByInstanceId(
const std::string &instanceId)
527 auto foundIter = m_qAbilityPeers.find(instanceId);
528 return foundIter != m_qAbilityPeers.end() ? foundIter->second :
nullptr;
531std::shared_ptr<
QAbilityPeer> JsStateImpl::tryGetQAbilityPeerByInstance(QNapi::Object qAbility)
533 Napi::HandleScope findPeerScope(qAbility.Env());
534 return tryFindMatchingQAbilityPeer(
535 [&](
const auto &peer) {
536 return peer->qAbility() == qAbility;
540std::shared_ptr<
QAbilityPeer> JsStateImpl::tryGetQAbilityPeerByQWindow(QObjectThreadSafeRef qwindow)
542 return tryFindMatchingQAbilityPeer(
543 [&](
const auto &peer) {
544 return peer->qWindowRef() == qwindow;
548void JsStateImpl::visitEachQAbilityPeer(
const std::function<
void(
std::shared_ptr<
QAbilityPeer>)> &visitor)
550 for (
const auto &qAbilityPeerEntry : m_qAbilityPeers)
551 visitor(qAbilityPeerEntry.second);
554void JsStateImpl::startNewQAbilityInstance(
555 std::shared_ptr<
QAbilityPeer> baseQAbilityPeer, QObjectThreadSafeRef qwindow,
556 QNapi::Object optStartOptions,
559 auto baseQAbility = baseQAbilityPeer->qAbility();
560 if (!baseQAbility.IsEmpty()) {
561 m_appFunctions->startQAbilityInstance(
562 baseQAbility, qwindow, optStartOptions, std::move(startupNotifyFunc));
566void JsStateImpl::startAppProcess(
567 const std::string &processId, QNapi::Object requestWant, QNapi::Object optStartOptions)
569 auto baseQAbility = defaultQAbilityPeer()->qAbility();
570 if (!baseQAbility.IsEmpty())
571 m_appFunctions->startAppProcess(baseQAbility, processId, requestWant, optStartOptions);
574void JsStateImpl::startAppProcess(
575 const std::string &processId, QNapi::Object requestWant,
576 QNapi::Object optStartOptions,
std::function<
void(
JsState &)> continueFunc)
578 auto baseQAbility = defaultQAbilityPeer()->qAbility();
579 if (!baseQAbility.IsEmpty()) {
580 m_appFunctions->startAppProcess(
581 baseQAbility, processId, requestWant, optStartOptions, std::move(continueFunc));
583 invokeTask(
std::move(continueFunc));
587void JsStateImpl::addNewWantConsumer(QOhosConsumer<JsState &, QNapi::Object, QNapi::Object> wantConsumer)
589 m_newWantConsumers.push_back(std::move(wantConsumer));
592void JsStateImpl::startNoUiChildProcess(
const std::string &libraryName,
const std::vector<std::string> &args)
594 m_appFunctions->startNoUiChildProcess(*
this, libraryName, args);
602template<
typename PeerMatchFunc>
603std::shared_ptr<
QAbilityPeer> JsStateImpl::tryFindMatchingQAbilityPeer(PeerMatchFunc &&matchFunc)
605 auto foundIter =
std::find_if(
606 m_qAbilityPeers.begin(), m_qAbilityPeers.end(),
607 [&](
const auto &entry) {
608 return matchFunc(entry.second);
610 return foundIter != m_qAbilityPeers.end() ? foundIter->second :
nullptr;
613void *JsStateImpl::getAttachedObjectWithLazyCreate(
614 const std::type_info &objectTypeInfo, QOhosSupplier<std::shared_ptr<
void>> objectFactory)
616 auto objectIter = m_attachedObjects.find(objectTypeInfo);
617 if (objectIter == m_attachedObjects.end())
618 std::tie(objectIter, std::ignore) = m_attachedObjects.emplace(objectTypeInfo, objectFactory());
620 return objectIter->second.get();
623std::tuple<QNapi::Object, std::string> JsStateImpl::extractModuleFromEvalExpr(
const std::string &expr)
625 using namespace std::string_literals;
627 std::size_t longestModulePrefixSize = 0;
628 auto considerAsLongestPrefix = [&](
const std::string &name) {
629 if (name.size() > longestModulePrefixSize
630 && expr.compare(0, name.size(), name) == 0
631 && (expr.size() == name.size() || expr[name.size()] ==
'.')) {
632 longestModulePrefixSize = name.size();
635 for (
const auto &moduleFactoryEntry : m_jsModulesFactories)
636 considerAsLongestPrefix(moduleFactoryEntry.first);
638 if (longestModulePrefixSize == 0) {
639 throw QNapi::makeLoggedException(
640 env(),
"global expression doesn't start with known module path: '"s + expr +
"'");
643 return std::make_tuple(
644 getModule(expr.substr(0, longestModulePrefixSize)),
645 longestModulePrefixSize < expr.size()
646 ? expr.substr(longestModulePrefixSize + 1)
650QNapi::Number JsStateImpl::mapOhosEnumToJs(
651 int enumValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
653 using namespace std::string_literals;
655 const auto &enumeratorsValues = getOhosEnumEnumerators(enumTypeInfo, ohosEnumInfoFactory);
657 auto valueIter =
std::find_if(
658 enumeratorsValues.begin(), enumeratorsValues.end(),
659 [&](
const auto &valuePair) {
660 return valuePair.first == enumValue;
663 if (valueIter == enumeratorsValues.end()) {
664 throw QNapi::makeLoggedException(
666 "Illegal C++ value of enumerator for enum '"s + enumTypeInfo.name() +
"': "
667 + std::to_string(enumValue));
670 return QNapi::Number::New(env(), valueIter->second);
673std::optional<
int> JsStateImpl::tryMapOhosEnumFromJs(
674 QNapi::Number enumJsValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
676 double enumDoubleJsValue = enumJsValue.DoubleValue();
678 const auto &enumeratorsValues = getOhosEnumEnumerators(enumTypeInfo, ohosEnumInfoFactory);
680 auto valueIter =
std::find_if(
681 enumeratorsValues.begin(), enumeratorsValues.end(),
682 [&](
const auto &valuePair) {
683 return valuePair.second == enumDoubleJsValue;
686 return valueIter != enumeratorsValues.end()
687 ?
std::optional(valueIter->first)
691int JsStateImpl::mapOhosEnumFromJs(
692 QNapi::Number enumJsValue,
const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
694 using namespace std::string_literals;
696 auto optEnumValue = tryMapOhosEnumFromJs(enumJsValue, enumTypeInfo, ohosEnumInfoFactory);
697 if (!optEnumValue.has_value()) {
698 throw QNapi::makeLoggedException(
700 "Illegal Napi value of enumerator for enum '"s + enumTypeInfo.name() +
"': "
701 + std::to_string(enumJsValue.DoubleValue()));
703 return optEnumValue.value();
706QNapi::Symbol JsStateImpl::getJsSymbolForType(
const std::type_info &typeInfo)
708 using namespace std::string_literals;
710 auto jsSymbolRefIter = m_jsSymbolsRefs.find(typeInfo);
711 if (jsSymbolRefIter == m_jsSymbolsRefs.end()) {
712 std::tie(jsSymbolRefIter, std::ignore) = m_jsSymbolsRefs.emplace(
714 QNapi::Reference<>::makePersistentFrom(
715 QNapi::Symbol::New(env(),
"_io_qt_"s + typeInfo.name())));
717 return jsSymbolRefIter->second.Value();
720const std::vector<std::pair<
int,
double>> &JsStateImpl::getOhosEnumEnumerators(
721 const std::type_info &enumTypeInfo, OhosEnumInfo (*ohosEnumInfoFactory)())
723 auto enumeratorsIter = m_ohosEnumsEnumerators.find(enumTypeInfo);
724 if (enumeratorsIter == m_ohosEnumsEnumerators.end()) {
725 std::tie(enumeratorsIter, std::ignore) = m_ohosEnumsEnumerators.emplace(
726 enumTypeInfo, resolveOhosEnumEnumerators(ohosEnumInfoFactory()));
729 return enumeratorsIter->second;
732std::vector<std::pair<
int,
double>> JsStateImpl::resolveOhosEnumEnumerators(
const OhosEnumInfo &enumInfo)
734 using namespace std::string_literals;
736 QNapi::Object enumObj;
738 enumObj = eval<QNapi::Object>(enumInfo.fullTypeName);
739 }
catch (
const Napi::Error &e) {
740 throw QNapi::makeLoggedException(
741 env(),
"JS enum '"s + enumInfo.fullTypeName +
"' not found: "s + e.what());
744 std::vector<std::pair<
int,
double>> enumeratorsValues;
746 for (
const auto &enumNamePair : enumInfo.enumeratorsNames) {
747 const auto *enumeratorName = enumNamePair.second;
749 QNapi::Value jsEnumeratorValue = QNapi::getPropOrUndefined(enumObj, enumeratorName);
751 if (!jsEnumeratorValue.IsNumber()) {
752 throw QNapi::makeLoggedException(
753 env(),
"JS enum '"s + enumInfo.fullTypeName +
"' doesn't contain: "s + enumeratorName);
756 enumeratorsValues.emplace_back(
757 enumNamePair.first, QNapi::checkedCast<QNapi::Number>(jsEnumeratorValue).DoubleValue());
760 return enumeratorsValues;
763JsStateImpl &getJsStateImpl()
765 static JsStateImpl jsStateImpl;
771 static auto qtJsBlockingCallsGateway = QOhosMtBlockingCallsGateway<JsState>::makeInstance(
772 invokeInQtThread, invokeInJsThread);
773 return *qtJsBlockingCallsGateway;
776class JsThreadOpsImpl :
public ::QOhosJsThreadOps
779 QOhosJsState &jsState()
override;
781 void invoke(std::function<
void(QOhosJsState &)> task)
override;
782 void invokeAndWaitForContinue(
783 std::function<
void(QOhosJsState &, std::function<
void()>)> &&task)
override;
784 void runAndWait(
const std::function<
void(QOhosJsState &)> &task)
override;
787QOhosJsState &JsThreadOpsImpl::jsState()
789 return getJsStateImpl();
792void JsThreadOpsImpl::invoke(std::function<
void(QOhosJsState &)> task)
797void JsThreadOpsImpl::invokeAndWaitForContinue(
798 std::function<
void(QOhosJsState &, std::function<
void()>)> &&task)
803void JsThreadOpsImpl::runAndWait(
const std::function<
void(QOhosJsState &)> &task)
824 return matchTypeIdObject == &typeIdObject ?
this :
nullptr;
834 qAbilityPeer->tryCastWithTypeIdObject(&typeIdObject));
836 return qUiAbilityPeer !=
nullptr
837 ? qUiAbilityPeer->shared_from_this()
851 return getJsStateImpl();
856 qOhosPrintfDebug(
"Tagging JS Window as closing (from: %s)", logContext);
857 Napi::HandleScope funcScope{jsWindow.Env()};
859 getJsWindowsTrackerIsClosingPropSymbol(getJsStateImpl()),
860 QNapi::Boolean::New(jsWindow.Env(),
true));
865 Napi::HandleScope funcScope{jsWindow.Env()};
866 auto isClosingPropSymbol = getJsWindowsTrackerIsClosingPropSymbol(getJsStateImpl());
867 auto isClosingValue = jsWindow.Has(isClosingPropSymbol)
868 ? jsWindow.Get(isClosingPropSymbol)
870 return isClosingValue.IsBoolean() && QNapi::checkedCast<QNapi::Boolean>(isClosingValue).Value();
877 static JsThreadOpsImpl jsThreadOpsImpl;
878 QOhosJsThreadOps::registerInstance(&jsThreadOpsImpl);
880 getJsStateImpl().initInJsThread(env,
std::move(jsModulesFactories), appFunctions, qtRunMode);
885 getJsStateImpl().addQAbilityPeerInJsThread(qAbilityPeer);
890 getJsStateImpl().removeMatchingQAbilityPeerInJsThread(qAbility);
895 getJsStateImpl().dispatchNewWantInJsThread(want, launchParam);
900 getJsStateImpl().invokeTask(
std::move(task));
904 std::function<
void(
JsState &, QOhosTaskPromise<>)> &&task,
905 std::string callerContextName)
907 using namespace std::string_literals;
909 const auto funcInfo = Q_FUNC_INFO;
911 struct JsTaskCompletionState
914 bool outerTaskPromiseSettled =
false;
915 QOhosOptional<std::string> jsThreadExceptionMsg;
917 auto completionState =
std::make_shared<JsTaskCompletionState>();
919 auto catchingTaskWrapper =
920 [task = std::move(task), completionState, callerContextName, funcInfo](
921 JsState &jsState, QOhosTaskPromise<> outerTaskPromise) {
922 auto sharedOuterTaskPromise =
923 std::make_shared<QOhosTaskPromise<>>(std::move(outerTaskPromise));
925 auto settleOuterTaskPromise =
926 [completionState, sharedOuterTaskPromise](QOhosOptional<std::string> exceptionMsg) {
928 std::lock_guard<
std::mutex> lock(completionState->mutex);
929 if (completionState->outerTaskPromiseSettled)
931 completionState->outerTaskPromiseSettled =
true;
932 completionState->jsThreadExceptionMsg =
std::move(exceptionMsg);
934 (*sharedOuterTaskPromise)();
937 auto innerTaskPromise = QOhosTaskPromise<>(
938 [settleOuterTaskPromise]() {
939 settleOuterTaskPromise({});
941 [funcInfo, callerContextName]() {
942 if (!
std::uncaught_exception()) {
943 qOhosReportFatalErrorAndAbort(
944 "%s: promise destroyed without notifying the caller: %s",
945 funcInfo, callerContextName.c_str());
951 task(jsState,
std::move(innerTaskPromise));
952 }
catch (
const std::exception &e) {
953 settleOuterTaskPromise(QOhosOptional<std::string>(e.what()));
955 settleOuterTaskPromise(QOhosOptional<std::string>(
"<unknown exception>"));
959 if (getQtState().isQtThread()) {
960 getQtJsBlockingCallsGateway().runInSlaveThreadAndWaitForContinue(
961 std::move(catchingTaskWrapper), callerContextName);
963 auto taskReadyPromise =
std::make_shared<
std::promise<
void>>();
964 auto taskReadyFuture = taskReadyPromise->get_future();
966 auto sharedTaskPromise = std::make_shared<QOhosTaskPromise<>>(
967 [taskReadyPromise]() {
968 taskReadyPromise->set_value();
970 [funcInfo, callerContextName]() {
971 qOhosReportFatalErrorAndAbort(
972 "%s: promise destroyed without notifying the caller: %s",
973 funcInfo, callerContextName.c_str());
978 [catchingTaskWrapper = std::move(catchingTaskWrapper), sharedTaskPromise](
JsState &jsState) {
979 catchingTaskWrapper(jsState,
std::move(*sharedTaskPromise));
981 taskReadyFuture.wait();
984 if (completionState->jsThreadExceptionMsg.has_value()) {
985 throw std::runtime_error(
986 "Got exception from task invoked in JS thread"s
987 +
" (caller: \""s + callerContextName +
"\"): "
988 + completionState->jsThreadExceptionMsg.value());
993 const std::function<
void(
JsState &)> &task,
std::string callerContextName)
995 if (getJsStateImpl().isJsThread()) {
996 task(getJsStateImpl());
998 invokeInJsThreadAndWaitForContinue(
999 [&](
auto &jsState,
auto taskPromise) {
1003 std::move(callerContextName));
1008 std::function<
void(
std::function<
void()>)> &&task,
1009 std::chrono::nanoseconds timeout)
1011 if (getJsStateImpl().isJsThread()) {
1012 auto result = getQtJsBlockingCallsGateway().tryInvokeInMasterThreadAndTryWaitForContinue(
std::move(task), timeout);
1013 return result == QOhosMtBlockingCallsGateway<JsState>::MasterThreadTaskResult::Finished;
1015 auto taskReadyPromise =
std::make_shared<
std::promise<
void>>();
1016 auto taskReadyFuture = taskReadyPromise->get_future();
1018 auto continueFunc = [taskReadyPromise]() {
1019 taskReadyPromise->set_value();
1023 [task = std::move(task), continueFunc = std::move(continueFunc)]()
mutable {
1024 task(
std::move(continueFunc));
1026 auto waitResult = taskReadyFuture.wait_for(timeout);
1028 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 runInJsThreadAndWait(const std::function< void(JsState &)> &task, std::string callerContextName={})
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 invokeInJsThreadAndWaitForContinue(std::function< void(JsState &, QOhosTaskPromise<>)> &&task, std::string callerContextName={})
void addJsQAbilityPeer(std::shared_ptr< QAbilityPeer > qAbilityPeer)