11void QQmlEnginePrivate::incubate(
12 QQmlIncubator &i,
const QQmlRefPointer<QQmlContextData> &forContext)
14 QExplicitlySharedDataPointer<QQmlIncubatorPrivate> p(i.d);
16 QQmlIncubator::IncubationMode mode = i.incubationMode();
18 if (!incubationController)
19 mode = QQmlIncubator::Synchronous;
21 if (mode == QQmlIncubator::AsynchronousIfNested) {
22 mode = QQmlIncubator::Synchronous;
25 QExplicitlySharedDataPointer<QQmlIncubatorPrivate> parentIncubator;
26 QQmlRefPointer<QQmlContextData> cctxt = forContext;
28 if (QQmlIncubatorPrivate *incubator = cctxt->incubator()) {
29 parentIncubator = incubator;
32 cctxt = cctxt->parent();
35 if (parentIncubator && parentIncubator->isAsynchronous) {
36 mode = QQmlIncubator::Asynchronous;
37 p->waitingOnMe = parentIncubator;
38 parentIncubator->waitingFor.insert(p.data());
42 p->isAsynchronous = (mode != QQmlIncubator::Synchronous);
44 inProgressCreations++;
46 if (mode == QQmlIncubator::Synchronous) {
47 QRecursionWatcher<QQmlIncubatorPrivate, &QQmlIncubatorPrivate::recursion> watcher(p.data());
49 p->changeStatus(QQmlIncubator::Loading);
51 if (!watcher.hasRecursed()) {
52 QQmlInstantiationInterrupt i;
56 incubatorList.insert(p.data());
59 p->vmeGuard.guard(p->creator.data());
60 p->changeStatus(QQmlIncubator::Loading);
62 if (incubationController)
63 incubationController->incubatingObjectCountChanged(incubatorCount);
104void QQmlIncubatorPrivate::clear()
107 if (requiredPropertiesFromComponent)
108 requiredPropertiesFromComponent =
decltype(requiredPropertiesFromComponent){};
109 compilationUnit.reset();
110 if (next.isInList()) {
112 enginePriv->incubatorCount--;
113 QQmlIncubationController *controller = enginePriv->incubationController;
115 controller->incubatingObjectCountChanged(enginePriv->incubatorCount);
117 enginePriv =
nullptr;
118 if (!rootContext.isNull()) {
119 if (rootContext->incubator())
120 rootContext->setIncubator(
nullptr);
121 rootContext.setContextData({});
124 if (nextWaitingFor.isInList()) {
125 Q_ASSERT(waitingOnMe);
126 nextWaitingFor.remove();
127 waitingOnMe =
nullptr;
131 while (waitingFor.first()) {
132 QQmlIncubator * i =
static_cast<QQmlIncubatorPrivate*>(waitingFor.first())->q;
137 bool guardOk = vmeGuard.isOK();
140 if (creator && guardOk)
142 creator.reset(
nullptr);
245void QQmlIncubatorPrivate::incubate(QQmlInstantiationInterrupt &i)
247 if (!compilationUnit)
250 QExplicitlySharedDataPointer<QQmlIncubatorPrivate> protectThis(
this);
252 QRecursionWatcher<QQmlIncubatorPrivate, &QQmlIncubatorPrivate::recursion> watcher(
this);
254 QQmlEnginePrivate *enginePriv =
this->enginePriv;
257 enum { EstimatedSizeInV4Frames = 2 };
258 QV4::ExecutionEngineCallDepthRecorder<EstimatedSizeInV4Frames> callDepthRecorder(
259 compilationUnit->engine);
260 if (callDepthRecorder.hasOverflow()) {
262 error.setMessageType(QtCriticalMsg);
263 error.setUrl(compilationUnit->url());
264 error.setDescription(QQmlComponent::tr(
"Maximum call stack size exceeded."));
266 progress = QQmlIncubatorPrivate::Completed;
270 if (!vmeGuard.isOK()) {
272 error.setMessageType(QtInfoMsg);
273 error.setUrl(compilationUnit->url());
274 error.setDescription(QQmlComponent::tr(
"Object or context destroyed during incubation"));
276 progress = QQmlIncubatorPrivate::Completed;
283 if (progress == QQmlIncubatorPrivate::Execute) {
284 enginePriv->referenceScarceResources();
285 QObject *tresult =
nullptr;
286 tresult = creator->create(subComponentToCreate,
nullptr, &i);
288 errors = creator->errors;
290 RequiredProperties* requiredProperties = creator->requiredProperties();
291 for (
auto it = initialProperties.cbegin(); it != initialProperties.cend(); ++it) {
292 auto component = tresult;
293 auto name = it.key();
294 QQmlProperty prop = QQmlComponentPrivate::removePropertyFromRequired(
295 component, name, requiredProperties, QQmlEnginePrivate::get(enginePriv));
296 if (!prop.isValid() || !prop.write(it.value())) {
298 error.setUrl(compilationUnit->url());
299 error.setDescription(QLatin1String(
"Could not set property %1").arg(name));
300 errors.push_back(error);
304 enginePriv->dereferenceScarceResources();
306 if (watcher.hasRecursed())
310 if (errors.isEmpty() && result ==
nullptr)
314 QQmlData *ddata = QQmlData::get(result);
317 ddata->indestructible =
true;
318 ddata->explicitIndestructibleSet =
true;
319 ddata->rootObjectInCreation =
false;
321 q->setInitialState(result);
322 if (creator && !creator->requiredProperties()->empty()) {
323 const RequiredProperties *unsetRequiredProperties = creator->requiredProperties();
324 for (
const auto& unsetRequiredProperty: *unsetRequiredProperties)
325 errors << QQmlComponentPrivate::unsetRequiredPropertyToQQmlError(unsetRequiredProperty);
330 if (watcher.hasRecursed())
333 if (errors.isEmpty())
334 progress = QQmlIncubatorPrivate::Completing;
336 progress = QQmlIncubatorPrivate::Completed;
338 changeStatus(calculateStatus());
340 if (watcher.hasRecursed())
343 if (i.shouldInterrupt())
347 if (progress == QQmlIncubatorPrivate::Completing) {
349 if (watcher.hasRecursed())
352 if (creator->finalize(i)) {
353 rootContext = creator->rootContext();
354 progress = QQmlIncubatorPrivate::Completed;
357 }
while (!i.shouldInterrupt());
361 if (progress == QQmlIncubatorPrivate::Completed && waitingFor.isEmpty()) {
362 QExplicitlySharedDataPointer<QQmlIncubatorPrivate> isWaiting = waitingOnMe;
366 QRecursionWatcher<QQmlIncubatorPrivate, &QQmlIncubatorPrivate::recursion> watcher(isWaiting.data());
367 changeStatus(calculateStatus());
368 if (!watcher.hasRecursed())
369 isWaiting->incubate(i);
371 changeStatus(calculateStatus());
374 enginePriv->inProgressCreations--;
376 if (0 == enginePriv->inProgressCreations) {
377 while (enginePriv->erroredBindings)
378 enginePriv->warning(enginePriv->erroredBindings->removeError());
380 }
else if (!creator.isNull()) {
381 vmeGuard.guard(creator.data());
391void QQmlIncubatorPrivate::incubateCppBasedComponent(QQmlComponent *component, QQmlContext *context)
393 auto compPriv = QQmlComponentPrivate::get(component);
394 Q_ASSERT(compPriv->loadedType().isCreatable());
395 std::unique_ptr<QObject> object(component->beginCreate(context));
396 component->setInitialProperties(object.get(), initialProperties);
397 if (
auto props = compPriv->requiredProperties()) {
398 requiredPropertiesFromComponent = props;
399 requiredPropertiesFromComponent.setTag(HadTopLevelRequired::Yes);
401 q->setInitialState(object.get());
402 if (requiredPropertiesFromComponent && !requiredPropertiesFromComponent->isEmpty()) {
403 for (
const RequiredPropertyInfo &unsetRequiredProperty :
404 std::as_const(*requiredPropertiesFromComponent)) {
405 errors << QQmlComponentPrivate::unsetRequiredPropertyToQQmlError(unsetRequiredProperty);
408 compPriv->completeCreate();
409 result = object.release();
410 progress = QQmlIncubatorPrivate::Completed;
412 changeStatus(calculateStatus());
599void QQmlIncubator::clear()
601 QRecursionWatcher<QQmlIncubatorPrivate, &QQmlIncubatorPrivate::recursion> watcher(d);
608 QQmlEnginePrivate *enginePriv = d->enginePriv;
610 Q_ASSERT(d->compilationUnit);
611 if (d->result) d->result->deleteLater();
617 Q_ASSERT(d->compilationUnit.isNull());
618 Q_ASSERT(d->waitingOnMe.data() ==
nullptr);
619 Q_ASSERT(d->waitingFor.isEmpty());
622 d->progress = QQmlIncubatorPrivate::Execute;
626 Q_ASSERT(enginePriv);
628 enginePriv->inProgressCreations--;
629 if (0 == enginePriv->inProgressCreations) {
630 while (enginePriv->erroredBindings)
631 enginePriv->warning(enginePriv->erroredBindings->removeError());
635 d->changeStatus(Null);