5#include <QtCore/private/qnapi_p.h>
6#include <qohosjsenv_p.h>
7#include <QtCore/private/qohoslogger_p.h>
8#include <QtCore/qcryptographichash.h>
9#include <QtGui/qwindow.h>
14#include <qohosdeviceinfo_p.h>
15#include <qohosjsutils.h>
16#include <qohosplatformwindow.h>
17#include <render/qohosview.h>
25using namespace std::chrono_literals;
31const auto qtInternalRequestIdWantParamKey =
"io.qt.private.internalRequestId";
32const auto qAbilityInstanceIdWantParamKey =
"io.qt.private.abilityInstanceId";
33const auto callerAbilityNameWantParamKey =
"ohos.aafwk.param.callerAbilityName";
34const auto callerBundleNameWantParamKey =
"ohos.aafwk.param.callerBundleName";
35const auto callerPidWantParamKey =
"ohos.aafwk.param.callerPid";
37std::string getUtf8QtInternalRequestIdForThisProcess()
39 static const auto internalRequestIdCommonData =
40 QByteArray::fromHex(
"eba08604c86e45ae3690621757b3f4e9");
42 std::int32_t thisProcessPid = ::getpid();
44 QCryptographicHash hash{QCryptographicHash::Sha1};
45 hash.addData(internalRequestIdCommonData);
46 hash.addData(QByteArray::number(thisProcessPid));
48 return hash.result().toHex().toStdString();
51QNapi::Symbol getAbilityLaunchParamPropSymbol(
JsState &jsState)
57 return jsState.getJsSymbolForType<SymbolTag>();
60void sendWantToSelfQAbility(
62 QNapi::Object baseQAbility, QNapi::Object optStartOptions,
const std::string &instanceId,
65 auto qAbilityInfo = abilityEngine->readAbilityInfo(baseQAbility);
67 std::vector<QNapi::ValueWrapper> startAbilityArgs = {
71 {
"bundleName", qAbilityInfo.bundleName},
72 {
"moduleName", qAbilityInfo.moduleName},
73 {
"abilityName", qAbilityInfo.name},
79 {qtInternalRequestIdWantParamKey, getUtf8QtInternalRequestIdForThisProcess()},
80 {qAbilityInstanceIdWantParamKey, instanceId},
86 if (!optStartOptions.IsEmpty())
87 startAbilityArgs.push_back(optStartOptions);
89 baseQAbility.call<QNapi::Promise>(
"context.startAbility", startAbilityArgs)
91 [instanceId](
const CallbackInfo &cbInfo) {
92 logJsCallbackError(cbInfo,
"context.startAbility()");
93 qOhosReportFatalErrorAndAbort(
"Failed to start the Ability with instance id: %s", instanceId.c_str());
96 [continueFunc = std::move(continueFunc)](
const CallbackInfo &cbInfo) {
97 continueFunc(cbInfo.jsState());
101QOhosSupplier<std::string> makeAbilityInstanceIdsGenerator()
103 return [idsCounter =
std::uint64_t(0)]()
mutable {
104 auto instanceId =
std::to_string(idsCounter);
115 std::string instanceId, QObjectThreadSafeRef qwindow,
116 QNapi::Object uiAbility);
118 ~QAbilityPeerImpl() =
default;
120 std::string instanceId()
final;
121 QNapi::Object qAbility()
final;
122 QObjectThreadSafeRef qWindowRef()
final;
123 QOhosOptional<QNapi::Promise> qWindowDestroyPromise()
final;
124 std::shared_ptr<
std::atomic_bool> destroyAllowedFlag()
final;
126 void setQWindow(Napi::Env env, QObjectThreadSafeRef qwindow)
final;
129 struct QWindowDestroyPromiseData
131 QNapi::Promise::Deferred deferred;
132 QNapi::Reference<QNapi::Promise> promise;
136 std::string m_instanceId;
137 QObjectThreadSafeRef m_qwindow;
138 std::shared_ptr<QWindowDestroyPromiseData> m_optQWindowDestroyPromiseData;
139 std::shared_ptr<
std::atomic_bool> m_destroyAllowedFlag;
140 QNapi::Reference<QNapi::Object> m_uiAbility;
143QAbilityPeerImpl::QAbilityPeerImpl(
145 std::string instanceId, QObjectThreadSafeRef qwindow,
146 QNapi::Object uiAbility)
147 : m_abilityEngine(abilityEngine)
148 , m_instanceId(
std::move(instanceId))
149 , m_destroyAllowedFlag(std::make_shared<std::atomic_bool>(
false))
150 , m_uiAbility(Napi::Persistent(uiAbility))
152 setQWindow(uiAbility.Env(), qwindow);
155std::string QAbilityPeerImpl::instanceId()
160QNapi::Object QAbilityPeerImpl::qAbility()
162 return m_uiAbility.Value();
165QObjectThreadSafeRef QAbilityPeerImpl::qWindowRef()
170QOhosOptional<QNapi::Promise> QAbilityPeerImpl::qWindowDestroyPromise()
172 if (!m_optQWindowDestroyPromiseData || m_optQWindowDestroyPromiseData->promise.IsEmpty())
175 auto promiseValue = m_optQWindowDestroyPromiseData->promise.Value();
176 m_optQWindowDestroyPromiseData->promise.Reset();
178 return makeQOhosOptional(promiseValue);
181std::shared_ptr<
std::atomic_bool> QAbilityPeerImpl::destroyAllowedFlag()
183 return m_destroyAllowedFlag;
186void QAbilityPeerImpl::setQWindow(Napi::Env env, QObjectThreadSafeRef qwindow)
188 if (qwindow == m_qwindow)
192 "%s: setting QAbilityPeer's QWindow: %s / %s",
193 Q_FUNC_INFO, m_instanceId.c_str(), m_qwindow.refName().c_str());
195 if (m_qwindow != QObjectThreadSafeRef()) {
196 qOhosReportFatalErrorAndAbort(
197 "QAbilityPeerImpl: overwriting previously set qwindow: %s => %s",
198 m_qwindow.refName().c_str(), qwindow.refName().c_str());
203 auto qWindowDestroyPromiseDeferred = QNapi::Promise::Deferred::New(env);
204 m_optQWindowDestroyPromiseData = std::make_shared<QWindowDestroyPromiseData>(
205 QWindowDestroyPromiseData{
206 .deferred = qWindowDestroyPromiseDeferred,
207 .promise = QNapi::Reference<QNapi::Promise>::makePersistentFrom(
208 qWindowDestroyPromiseDeferred.Promise()),
211 auto resolveQWindowDestroyPromiseFunc =
212 [qWindowRef = qwindow, weakPromiseData = QtOhos::makeWeakPtr(m_optQWindowDestroyPromiseData)]() {
214 "%s: sending request to resolve QWindow destroy notify Promise if needed: %s",
215 Q_FUNC_INFO, qWindowRef.refName().c_str());
217 [qWindowRef, weakPromiseData](
JsState &jsState) {
218 auto promiseData = weakPromiseData.lock();
221 "%s: resolving QWindow destroy notify Promise: %s",
222 Q_FUNC_INFO, qWindowRef.refName().c_str());
223 auto napiUndefined = Napi::Env(jsState.env()).Undefined();
224 promiseData->deferred.Resolve(napiUndefined);
227 "%s: QAbilityPeer already removed on QWindow's destroy notification: %s",
228 Q_FUNC_INFO, qWindowRef.refName().c_str());
233 QtOhos::invokeInQtThread(
234 [qWindowRef = qwindow, resolveQWindowDestroyPromiseFunc]() {
235 auto *qWindow =
static_cast<QWindow *>(qWindowRef.data().data());
236 auto *platformWindow = qWindow !=
nullptr ? QOhosPlatformWindow::fromQWindowOrNull(qWindow) :
nullptr;
237 auto *view = platformWindow !=
nullptr ? platformWindow->ownedViewOrNull() :
nullptr;
239 if (view !=
nullptr) {
241 QtForOhos,
"%s: connecting to QOhosView's 'destroyed', win: %s",
242 Q_FUNC_INFO, qWindowRef.refName().c_str());
244 view, &QObject::destroyed,
245 [qWindowRef, resolveQWindowDestroyPromiseFunc](QObject *) {
247 QtForOhos,
"%s: got QOhosView's 'destroyed' signal, win: %s",
248 Q_FUNC_INFO, qWindowRef.refName().c_str());
249 resolveQWindowDestroyPromiseFunc();
253 QtForOhos,
"%s: QOhosView already destroyed, win: %s",
254 Q_FUNC_INFO, qWindowRef.refName().c_str());
255 resolveQWindowDestroyPromiseFunc();
265 std::string instanceId, QObjectThreadSafeRef qwindow,
266 QNapi::Object uiAbility, QNapi::Object windowStage);
268 QNapi::Object uiContext()
override;
269 QNapi::Object launchWant()
override;
270 QNapi::Object launchParam()
override;
271 QNapi::Object windowStage()
override;
272 QNapi::Object window()
override;
273 bool isTerminating()
override;
275 QNapi::Promise handleCloseRequestFromSystem(
277 std::function<QNapi::Value(JsState &, CloseAbilityRequestResolution)> promiseValueFactory)
override;
279 void handleOnContinueRequestFromSystem(
280 JsState &jsState, QNapi::Object wantParamsObj,
281 QOhosConsumer<JsState &, QOhosAbilityOnContinueResult> resultConsumer)
override;
283 void setOnContinueRequestsHandler(
284 std::function<
void(
JsState &, QNapi::Object, QOhosConsumer<JsState &, QOhosAbilityOnContinueResult>)> requestsHandler)
override;
287 QNapi::Reference<QNapi::Object> m_windowStage;
288 QNapi::Reference<QNapi::Object> m_window;
289 QNapi::Reference<QNapi::Object> m_launchParam;
290 std::shared_ptr<
void> m_windowWillCloseCallbackHandle;
291 std::function<
void(
JsState &, QNapi::Object, QOhosConsumer<JsState &, QOhosAbilityOnContinueResult>)> m_onContinueRequestsHandler;
294QUiAbilityPeerImpl::QUiAbilityPeerImpl(
296 std::string instanceId, QObjectThreadSafeRef qwindow,
297 QNapi::Object uiAbility, QNapi::Object windowStage)
298 : QAbilityPeerImpl(jsState, abilityEngine, instanceId, qwindow, uiAbility)
299 , m_windowStage(Napi::Persistent(windowStage))
300 , m_window(Napi::Persistent(windowStage.call<QNapi::Object>(
"getMainWindowSync")))
302 auto optLaunchParam = QNapi::getOptionalPropOrEmpty<QNapi::Object>(
303 uiAbility, getAbilityLaunchParamPropSymbol(jsState));
304 if (optLaunchParam.IsEmpty()) {
305 qOhosReportFatalErrorAndAbort(
306 "Ability with instanceId='%s' doesn't have 'launchParam' set. It shouldn't happen!",
309 m_launchParam = QNapi::Reference<>::makePersistentFrom(optLaunchParam);
312 m_windowWillCloseCallbackHandle = registerOnOffMethodsBasedEventHandler(
313 window(),
"windowWillClose",
314 [
this](
const CallbackInfo &cbInfo) {
315 JsState &jsState = cbInfo.jsState();
316 return handleCloseRequestFromSystem(
317 jsState,
"Window.windowWillClose",
318 CloseAbilityRequestSource::WindowWillClose,
319 [](JsState &jsState, CloseAbilityRequestResolution ohosRequestResolution) {
320 return QNapi::Boolean::New(
322 ohosRequestResolution == CloseAbilityRequestResolution::DontClose);
326 .optEventSourceAliveCheckFunc =
327 [qAbilityWeakRef = moveToSharedPtr(Napi::Weak(uiAbility))](QNapi::Object window) {
328 bool isWindowClosing = evalInJsThread(
330 return JsWindowsTracker::isWindowClosing(window);
332 auto qAbilityValue = qAbilityWeakRef->Value();
335 && qAbilityValue.IsObject()
336 && !QNapi::checkedCast<QNapi::Object>(qAbilityValue).eval<QNapi::Boolean>(
"context.isTerminating()");
342QNapi::Object QUiAbilityPeerImpl::uiContext()
344 return window().call<QNapi::Object>(
"getUIContext");
347QNapi::Object QUiAbilityPeerImpl::launchWant()
349 return qAbility().get<QNapi::Object>(
"launchWant");
352QNapi::Object QUiAbilityPeerImpl::launchParam()
354 return m_launchParam.Value();
357QNapi::Object QUiAbilityPeerImpl::windowStage()
359 return m_windowStage.Value();
362QNapi::Object QUiAbilityPeerImpl::window()
364 return m_window.Value();
367bool QUiAbilityPeerImpl::isTerminating()
369 return qAbility().call<QNapi::Boolean>(
"context.isTerminating");
372QOhosCloseEventContext::CloseRootCause mapCloseAbilityRequestSourceToRootCause(
376 using CloseRootCause = QOhosCloseEventContext::CloseRootCause;
378 switch (requestSource) {
380 return CloseRootCause::OnPrepareToTerminate;
382 return CloseRootCause::WindowStageClose;
385 qOhosReportFatalErrorAndAbort(
"Unsupported CloseAbilityRequestSource: %d",
static_cast<
int>(requestSource));
388QNapi::Promise QUiAbilityPeerImpl::handleCloseRequestFromSystem(
390 std::function<QNapi::Value(JsState &, CloseAbilityRequestResolution)> promiseValueFactory)
392 auto instanceId =
this->instanceId();
395 "%s: got close request from source %d, id: '%s'",
396 logContextStr.c_str(),
static_cast<
int>(requestSource), instanceId.c_str());
398 if (destroyAllowedFlag()->load()) {
400 "%s: resolving immediately with 'false', id: '%s'",
401 Q_FUNC_INFO, instanceId.c_str());
402 JsWindowsTracker::tagWindowAsClosing(
403 window(), printfToString(
"%s => false", logContextStr.c_str()).c_str());
404 return makeResolvedPromise(
405 promiseValueFactory(jsState, CloseAbilityRequestResolution::Close));
408 struct PromiseResolveFuncContext
410 std::string logContextStr;
412 std::shared_ptr<QNapi::Promise::Deferred> promiseDeferred;
413 std::string instanceId;
414 std::function<QNapi::Value(JsState &, CloseAbilityRequestResolution)> promiseValueFactory;
417 auto promiseResolveFuncContext = moveToSharedPtr(
418 PromiseResolveFuncContext{
419 .logContextStr = logContextStr,
420 .weakQAbilityPeer = makeWeakPtr(shared_from_this()),
421 .promiseDeferred = std::make_shared<QNapi::Promise::Deferred>(jsState.env()),
422 .instanceId = instanceId,
423 .promiseValueFactory = std::move(promiseValueFactory),
426 auto promiseResolveFunc = moveToSharedPtr(
429 auto qAbilityPeer = context->weakQAbilityPeer.lock();
431 JsWindowsTracker::tagWindowAsClosing(
432 qAbilityPeer->window(), printfToString(
"%s => false", context->logContextStr.c_str()).c_str());
434 context->promiseDeferred->Resolve(
435 context->promiseValueFactory(jsState, ohosRequestResolution));
437 "%s: promise resolved as %d, id: '%s'",
438 context->logContextStr.c_str(),
static_cast<
int>(ohosRequestResolution), context->instanceId.c_str());
441 constexpr auto promiseAutoresolveTimeout = 9s;
444 [logContextStr, promiseResolveFunc, instanceId](
const CallbackInfo &cbInfo) {
447 "%s: promise resolved by timeout, id: '%s'",
448 logContextStr.c_str(), instanceId.c_str());
451 promiseAutoresolveTimeout);
453 qWindowRef().visitInQtThreadIfAlive(
454 [logContextStr, requestSource, promiseResolveFunc, instanceId](
auto &qWindowObj) {
456 "%s: calling close(), id: '%s'",
457 logContextStr.c_str(), instanceId.c_str());
458 QOhosCloseEventContext::runWithCloseRootCauseAndCloseResolutionConsumerSet(
459 mapCloseAbilityRequestSourceToRootCause(requestSource),
460 [logContextStr, promiseResolveFunc, instanceId](QOhosCloseEventContext::CloseResolution closeResolution) {
461 auto ohosRequestResolution =
462 closeResolution == QOhosCloseEventContext::CloseResolution::Close
463 ? CloseAbilityRequestResolution::Close
464 : CloseAbilityRequestResolution::DontClose;
466 "%s: requesting promise resolve to %d from Qt thread, id: '%s'",
467 logContextStr.c_str(),
static_cast<
int>(ohosRequestResolution), instanceId.c_str());
468 QtOhos::invokeInJsThread(
469 [logContextStr, promiseResolveFunc, ohosRequestResolution, instanceId](JsState &jsState) {
470 bool resolveCalled = (*promiseResolveFunc)(jsState, ohosRequestResolution);
472 "%s: promise resolve to %d, called: %s, id: '%s'",
473 logContextStr.c_str(),
static_cast<
int>(ohosRequestResolution),
474 mapBoolToTrueFalseStr(resolveCalled), instanceId.c_str());
478 static_cast<QWindow &>(qWindowObj).close();
482 qOhosPrintfInfo(
"%s: returning Promise, id: '%s'", logContextStr.c_str(), instanceId.c_str());
484 return promiseResolveFuncContext->promiseDeferred->Promise();
487void QUiAbilityPeerImpl::handleOnContinueRequestFromSystem(
488 JsState &jsState, QNapi::Object wantParamsObj, QOhosConsumer<JsState &, QOhosAbilityOnContinueResult> resultConsumer)
490 if (m_onContinueRequestsHandler) {
491 m_onContinueRequestsHandler(jsState, wantParamsObj, std::move(resultConsumer));
493 qOhosPrintfDebug(
"%s: no handler set, rejecting", Q_FUNC_INFO);
495 [resultConsumer = std::move(resultConsumer)](
JsState &jsState) {
501void QUiAbilityPeerImpl::setOnContinueRequestsHandler(
502 std::function<
void(
JsState &, QNapi::Object, QOhosConsumer<JsState &, QOhosAbilityOnContinueResult>)> requestsHandler)
504 if (m_onContinueRequestsHandler) {
505 qOhosReportFatalErrorAndAbort(
506 "%s: overwriting previously set requests handler, id='%s'",
507 Q_FUNC_INFO, instanceId().c_str());
510 m_onContinueRequestsHandler = std::move(requestsHandler);
516 explicit QAbilityInstancesManagerImpl(
522 bool isWantFromThisApp(QNapi::Object appQAbility, QNapi::Object want)
const override;
524 QOhosOptional<
std::string> tryGetQAbilityInstanceIdFromWant(QNapi::Object appQAbility, QNapi::Object want)
const override;
525 std::string getQAbilityInstanceIdOrPendingAutoStartedId(QNapi::Object qAbility)
const override;
529 void registerPendingAutoStartedInstance()
override;
531 void startNewInstance(
532 QNapi::Object baseQAbility, QObjectThreadSafeRef qwindow,
533 QNapi::Object optStartOptions,
536 void handleStartedUiInstance(
JsState &jsState, QNapi::Object qAbility, QNapi::Object windowStage)
override;
541 struct InstanceStartParams
543 QObjectThreadSafeRef qwindow;
547 void handleStartedInstance(
548 JsState &jsState,
const std::string &abilityInstanceId,
549 const std::function<
std::shared_ptr<
QAbilityPeer>(
const std::string &,
const InstanceStartParams &)> &qAbilityPeerFactory);
553 std::shared_ptr<std::map<QUiAbilityPeer *, std::weak_ptr<QUiAbilityPeerImpl>>> m_ownedUiAbilityPeers;
554 QOhosSupplier<std::string> m_abilityInstanceIdsGenerator;
555 std::map<std::string, InstanceStartParams> m_pendingInstancesStartParams;
559QAbilityInstancesManagerImpl::QAbilityInstancesManagerImpl(
562 : m_abilityEngine(abilityEngine)
563 , m_autoStartedInstanceStartupNotifyFunc(moveToSharedPtr(std::move(autoStartedInstanceStartupNotifyFunc)))
564 , m_ownedUiAbilityPeers(std::make_shared<std::map<QUiAbilityPeer *, std::weak_ptr<QUiAbilityPeerImpl>>>())
565 , m_abilityInstanceIdsGenerator(makeAbilityInstanceIdsGenerator())
567 registerPendingAutoStartedInstance();
570QOhosOptional<
std::string> QAbilityInstancesManagerImpl::tryGetQAbilityInstanceIdFromWant(
571 QNapi::Object appQAbility, QNapi::Object want)
const
573 auto wantParameters = QNapi::getPropOrUndefined(want,
"parameters");
574 if (wantParameters.IsUndefined())
577 auto instanceIdParam = QNapi::getPropOrUndefined(wantParameters, qAbilityInstanceIdWantParamKey);
579 bool validInstanceId = isWantFromThisApp(appQAbility, want) && instanceIdParam.IsString();
581 return validInstanceId
582 ? makeQOhosOptional(QNapi::checkedCast<QNapi::String>(instanceIdParam).Utf8Value())
586std::string QAbilityInstancesManagerImpl::getQAbilityInstanceIdOrPendingAutoStartedId(QNapi::Object qAbility)
const
588 auto optInstanceId = tryGetQAbilityInstanceIdFromWant(
589 qAbility, qAbility.get<QNapi::Object>(
"launchWant"));
591 if (!optInstanceId.hasValue() && !m_pendingAutoStartedInstanceId.hasValue()) {
592 qOhosReportFatalErrorAndAbort(
593 "Got Ability without '%s' param and no pending auto-started instance",
594 qAbilityInstanceIdWantParamKey);
597 return optInstanceId.hasValue()
598 ? optInstanceId.value()
599 : m_pendingAutoStartedInstanceId.value();
604 return m_abilityEngine;
607bool QAbilityInstancesManagerImpl::isWantFromThisApp(QNapi::Object appQAbility, QNapi::Object want)
const
609 auto wantParameters = QNapi::getPropOrUndefined(want,
"parameters");
610 if (wantParameters.IsUndefined())
613 auto callerAbilityNameParam = QNapi::getPropOrUndefined(wantParameters, callerAbilityNameWantParamKey);
614 auto callerBundleNameParam = QNapi::getPropOrUndefined(wantParameters, callerBundleNameWantParamKey);
616 auto appAbilityInfo = m_abilityEngine->readAbilityInfo(appQAbility);
619 callerAbilityNameParam.IsString()
620 && QNapi::checkedCast<QNapi::String>(callerAbilityNameParam).Utf8Value() == appAbilityInfo.name
621 && callerBundleNameParam.IsString()
622 && QNapi::checkedCast<QNapi::String>(callerBundleNameParam).Utf8Value() == appAbilityInfo.bundleName;
625QOhosOptional<
std::string> QAbilityInstancesManagerImpl::pendingAutoStartedInstanceId()
const
627 return m_pendingAutoStartedInstanceId;
630void QAbilityInstancesManagerImpl::registerPendingAutoStartedInstance()
632 if (m_pendingAutoStartedInstanceId.hasValue()) {
633 qOhosReportFatalErrorAndAbort(
634 "%s: pending auto-started instance already registered (id='%s')",
635 Q_FUNC_INFO, m_pendingAutoStartedInstanceId.value().c_str());
638 m_pendingAutoStartedInstanceId = m_abilityInstanceIdsGenerator();
641 "Registered pending auto-started Ability instance, id='%s'",
642 m_pendingAutoStartedInstanceId.value().c_str());
645void QAbilityInstancesManagerImpl::startNewInstance(
646 QNapi::Object baseQAbility, QObjectThreadSafeRef qwindow,
647 QNapi::Object optStartOptions,
653 bool sendWantFinished =
false;
656 void callNotifyIfReady(
JsState &jsState)
659 "%s: sendWantFinished=%s, startedAbilityPeer=%s",
661 if (sendWantFinished && optAbilityPeer)
662 startupNotifyFunc(jsState, optAbilityPeer);
666 auto context =
std::make_shared<Context>();
667 context->startupNotifyFunc =
std::move(startupNotifyFunc);
669 auto instanceId = m_abilityInstanceIdsGenerator();
671 InstanceStartParams startParams;
672 startParams.qwindow = qwindow;
674 context->optAbilityPeer = abilityPeer;
675 context->callNotifyIfReady(jsState);
677 m_pendingInstancesStartParams.emplace(instanceId, std::move(startParams));
679 sendWantToSelfQAbility(
680 m_abilityEngine, baseQAbility, optStartOptions, instanceId,
682 context->sendWantFinished =
true;
683 context->callNotifyIfReady(jsState);
687void QAbilityInstancesManagerImpl::handleStartedUiInstance(
688 JsState &jsState, QNapi::Object qAbility, QNapi::Object windowStage)
690 handleStartedInstance(
691 jsState, getQAbilityInstanceIdOrPendingAutoStartedId(qAbility),
692 [&](
const std::string &instanceId,
const InstanceStartParams &startParams) {
693 auto uiAbilityPeer =
std::make_shared<QUiAbilityPeerImpl>(
694 jsState, m_abilityEngine, instanceId, startParams.qwindow, qAbility, windowStage);
696 m_ownedUiAbilityPeers->emplace(uiAbilityPeerKey, uiAbilityPeer);
697 return makeSharedPtrWithAttachedExtraData(
700 [ownedUiAbilityPeers = m_ownedUiAbilityPeers, uiAbilityPeerKey]() {
701 ownedUiAbilityPeers->erase(uiAbilityPeerKey);
709 auto ownedPeerEntryIter = m_ownedUiAbilityPeers->find(uiAbilityPeer.get());
711 ownedPeerEntryIter != m_ownedUiAbilityPeers->end()
712 ? ownedPeerEntryIter->second.lock()
715 if (!ownedPeerImpl) {
716 qOhosReportFatalErrorAndAbort(
717 "%s: got unknown QUiAbilityPeer %p with id='%s'",
718 Q_FUNC_INFO, uiAbilityPeer.get(), uiAbilityPeer->instanceId().c_str());
721 return ownedPeerImpl;
724void QAbilityInstancesManagerImpl::handleStartedInstance(
725 JsState &jsState,
const std::string &abilityInstanceId,
726 const std::function<
std::shared_ptr<
QAbilityPeer>(
const std::string &,
const InstanceStartParams &)> &qAbilityPeerFactory)
728 auto startParamsIter = m_pendingInstancesStartParams.find(abilityInstanceId);
729 if (startParamsIter != m_pendingInstancesStartParams.end()) {
730 auto startParams =
std::move(startParamsIter->second);
731 m_pendingInstancesStartParams.erase(startParamsIter);
733 auto qAbilityPeer = qAbilityPeerFactory(abilityInstanceId, startParams);
735 startParams.startupNotifyFunc(jsState, qAbilityPeer);
736 }
else if (abilityInstanceId == m_pendingAutoStartedInstanceId) {
737 m_pendingAutoStartedInstanceId.reset();
739 InstanceStartParams startParams = {
740 .qwindow = QObjectThreadSafeRef(),
741 .startupNotifyFunc = [startupNotifyFunc = m_autoStartedInstanceStartupNotifyFunc](
JsState &jsState,
std::shared_ptr<
QAbilityPeer> qAbilityPeer) {
742 (*startupNotifyFunc)(jsState, qAbilityPeer);
745 auto qAbilityPeer = qAbilityPeerFactory(abilityInstanceId, startParams);
747 startParams.startupNotifyFunc(jsState, qAbilityPeer);
750 "%s: got unexpected started instance notification with id='%s', ignoring",
751 Q_FUNC_INFO, abilityInstanceId.c_str());
765 auto optWantParameters = QNapi::getOptionalPropOrEmpty<QNapi::Object>(want,
"parameters");
767 auto optCallerPid = QNapi::getOptionalPropOrEmpty<QNapi::Number>(
768 optWantParameters, callerPidWantParamKey);
769 auto optQtInternalRequestId = QNapi::getOptionalPropOrEmpty<QNapi::String>(
770 optWantParameters, qtInternalRequestIdWantParamKey);
772 std::int32_t thisProcessPid = ::getpid();
775 !optCallerPid.IsEmpty() && optCallerPid.Int32Value() == thisProcessPid
776 && !optQtInternalRequestId.IsEmpty()
777 && optQtInternalRequestId.Utf8Value() == getUtf8QtInternalRequestIdForThisProcess();
781 JsState &jsState, QNapi::Object ability, QNapi::Object launchParam)
783 ability.set(getAbilityLaunchParamPropSymbol(jsState), launchParam);
790 return std::make_shared<QAbilityInstancesManagerImpl>(abilityEngine,
std::move(autoStartedInstanceStartupNotifyFunc));
std::enable_if_t< qohosplugincore_h_detail::isQOhosOptional< QOhosInvokeResult< Func, T > >, QOhosInvokeResult< Func, T > > andThen(Func &&func) const
JsState & jsState() const
CloseAbilityRequestSource
CloseAbilityRequestResolution
std::string const char * mapBoolToTrueFalseStr(bool value)
void invokeInJsThread(std::function< void(JsState &)> task)
std::shared_ptr< QAbilityInstancesManager > makeQAbilityInstancesManager(std::shared_ptr< QAbilityEngine > abilityEngine, std::function< void(JsState &, std::shared_ptr< QAbilityPeer >)> autoStartedInstanceStartupNotifyFunc)
QOhosAbilityOnContinueResult
void addJsQAbilityPeer(std::shared_ptr< QAbilityPeer > qAbilityPeer)
QOhosOptional< void > makeEmptyQOhosOptional()
QAbilityInstancesManager()
virtual ~QAbilityInstancesManager()