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);
109void QQmlIncubatorPrivate::clear()
114 if (requiredPropertiesFromComponent)
115 requiredPropertiesFromComponent =
decltype(requiredPropertiesFromComponent){};
116 compilationUnit.reset();
117 if (next.isInList()) {
119 enginePriv->incubatorCount--;
120 QQmlIncubationController *controller = enginePriv->incubationController;
122 controller->incubatingObjectCountChanged(enginePriv->incubatorCount);
124 enginePriv =
nullptr;
125 if (!rootContext.isNull()) {
126 if (rootContext->incubator())
127 rootContext->setIncubator(
nullptr);
128 rootContext.setContextData({});
131 if (nextWaitingFor.isInList()) {
132 Q_ASSERT(waitingOnMe);
133 nextWaitingFor.remove();
134 waitingOnMe =
nullptr;
138 while (waitingFor.first()) {
139 QQmlIncubator * i =
static_cast<QQmlIncubatorPrivate*>(waitingFor.first())->q;
144 bool guardOk = vmeGuard.isOK();
147 if (creator && guardOk)
149 creator.reset(
nullptr);
252void QQmlIncubatorPrivate::incubate(QQmlInstantiationInterrupt &i)
254 if (!compilationUnit)
257 QExplicitlySharedDataPointer<QQmlIncubatorPrivate> protectThis(
this);
259 QRecursionWatcher<QQmlIncubatorPrivate, &QQmlIncubatorPrivate::recursion> watcher(
this);
261 QQmlEnginePrivate *enginePriv =
this->enginePriv;
264 enum { EstimatedSizeInV4Frames = 2 };
265 QV4::ExecutionEngineCallDepthRecorder<EstimatedSizeInV4Frames> callDepthRecorder(
266 compilationUnit->engine);
267 if (callDepthRecorder.hasOverflow()) {
269 error.setMessageType(QtCriticalMsg);
270 error.setUrl(compilationUnit->url());
271 error.setDescription(QQmlComponent::tr(
"Maximum call stack size exceeded."));
273 progress = QQmlIncubatorPrivate::Completed;
277 if (!vmeGuard.isOK()) {
279 error.setMessageType(QtInfoMsg);
280 error.setUrl(compilationUnit->url());
281 error.setDescription(QQmlComponent::tr(
"Object or context destroyed during incubation"));
283 progress = QQmlIncubatorPrivate::Completed;
290 if (progress == QQmlIncubatorPrivate::Execute) {
291 enginePriv->referenceScarceResources();
292 QObject *tresult =
nullptr;
293 tresult = creator->create(subComponentToCreate,
nullptr, &i);
295 errors = creator->errors;
297 RequiredProperties* requiredProperties = creator->requiredProperties();
298 for (
auto it = initialProperties.cbegin(); it != initialProperties.cend(); ++it) {
299 auto component = tresult;
300 auto name = it.key();
301 QQmlProperty prop = QQmlComponentPrivate::removePropertyFromRequired(
302 component, name, requiredProperties, QQmlEnginePrivate::get(enginePriv));
303 if (!prop.isValid() || !prop.write(it.value())) {
305 error.setUrl(compilationUnit->url());
306 error.setDescription(QLatin1String(
"Could not set property %1").arg(name));
307 errors.push_back(error);
311 enginePriv->dereferenceScarceResources();
313 if (watcher.hasRecursed())
317 if (errors.isEmpty() && result ==
nullptr)
321 QQmlData *ddata = QQmlData::get(result);
324 ddata->indestructible =
true;
325 ddata->explicitIndestructibleSet =
true;
326 ddata->rootObjectInCreation =
false;
328 q->setInitialState(result);
329 if (creator && !creator->requiredProperties()->empty()) {
330 const RequiredProperties *unsetRequiredProperties = creator->requiredProperties();
331 for (
const auto& unsetRequiredProperty: *unsetRequiredProperties)
332 errors << QQmlComponentPrivate::unsetRequiredPropertyToQQmlError(unsetRequiredProperty);
337 if (watcher.hasRecursed())
340 if (errors.isEmpty())
341 progress = QQmlIncubatorPrivate::Completing;
343 progress = QQmlIncubatorPrivate::Completed;
345 changeStatus(calculateStatus());
347 if (watcher.hasRecursed())
350 if (i.shouldInterrupt())
354 if (progress == QQmlIncubatorPrivate::Completing) {
356 if (watcher.hasRecursed())
359 if (creator->finalize(i)) {
360 rootContext = creator->rootContext();
361 progress = QQmlIncubatorPrivate::Completed;
364 }
while (!i.shouldInterrupt());
368 if (progress == QQmlIncubatorPrivate::Completed && waitingFor.isEmpty()) {
369 QExplicitlySharedDataPointer<QQmlIncubatorPrivate> isWaiting = waitingOnMe;
373 QRecursionWatcher<QQmlIncubatorPrivate, &QQmlIncubatorPrivate::recursion> watcher(isWaiting.data());
374 changeStatus(calculateStatus());
375 if (!watcher.hasRecursed())
376 isWaiting->incubate(i);
378 changeStatus(calculateStatus());
381 enginePriv->inProgressCreations--;
383 if (0 == enginePriv->inProgressCreations) {
384 while (enginePriv->erroredBindings)
385 enginePriv->warning(enginePriv->erroredBindings->removeError());
387 }
else if (!creator.isNull()) {
388 vmeGuard.guard(creator.data());
398void QQmlIncubatorPrivate::incubateCppBasedComponent(QQmlComponent *component, QQmlContext *context)
400 auto compPriv = QQmlComponentPrivate::get(component);
401 Q_ASSERT(compPriv->loadedType().isCreatable());
402 std::unique_ptr<QObject> object(component->beginCreate(context));
403 component->setInitialProperties(object.get(), initialProperties);
404 if (
auto props = compPriv->requiredProperties()) {
405 requiredPropertiesFromComponent = props;
406 requiredPropertiesFromComponent.setTag(HadTopLevelRequired::Yes);
408 q->setInitialState(object.get());
409 if (requiredPropertiesFromComponent && !requiredPropertiesFromComponent->isEmpty()) {
410 for (
const RequiredPropertyInfo &unsetRequiredProperty :
411 std::as_const(*requiredPropertiesFromComponent)) {
412 errors << QQmlComponentPrivate::unsetRequiredPropertyToQQmlError(unsetRequiredProperty);
415 compPriv->completeCreate();
416 result = object.release();
417 progress = QQmlIncubatorPrivate::Completed;
419 changeStatus(calculateStatus());
606void QQmlIncubator::clear()
608 QRecursionWatcher<QQmlIncubatorPrivate, &QQmlIncubatorPrivate::recursion> watcher(d);
615 QQmlEnginePrivate *enginePriv = d->enginePriv;
617 Q_ASSERT(d->compilationUnit);
618 if (d->result) d->result->deleteLater();
624 Q_ASSERT(d->compilationUnit.isNull());
625 Q_ASSERT(d->waitingOnMe.data() ==
nullptr);
626 Q_ASSERT(d->waitingFor.isEmpty());
629 d->progress = QQmlIncubatorPrivate::Execute;
633 Q_ASSERT(enginePriv);
635 enginePriv->inProgressCreations--;
636 if (0 == enginePriv->inProgressCreations) {
637 while (enginePriv->erroredBindings)
638 enginePriv->warning(enginePriv->erroredBindings->removeError());
642 d->changeStatus(Null);