36 static QQmlRefPointer<QQmlContextData> createRefCounted(
37 const QQmlRefPointer<QQmlContextData> &parent)
39 return QQmlRefPointer<QQmlContextData>(
new QQmlContextData(RefCounted,
nullptr, parent),
40 QQmlRefPointer<QQmlContextData>::Adopt);
44 static QQmlRefPointer<QQmlContextData> createChild(
45 const QQmlRefPointer<QQmlContextData> &parent)
47 Q_ASSERT(!parent.isNull());
48 return QQmlRefPointer<QQmlContextData>(
new QQmlContextData(OwnedByParent,
nullptr, parent));
51 void addref()
const { ++m_refCount; }
52 void release()
const {
if (--m_refCount == 0)
delete this; }
53 int count()
const {
return m_refCount; }
54 int refCount()
const {
return m_refCount; }
56 QQmlRefPointer<QV4::ExecutableCompilationUnit> typeCompilationUnit()
const
58 return m_typeCompilationUnit;
60 void initFromTypeCompilationUnit(
const QQmlRefPointer<QV4::ExecutableCompilationUnit> &unit,
61 int subComponentIndex);
63 static QQmlRefPointer<QQmlContextData> get(QQmlContext *context) {
64 return QQmlContextPrivate::get(context)->m_data;
67 void emitDestruction();
68 void clearExpressions();
69 void clearContextRecursively();
70 void clearChildrenAndSiblings();
71 void clearImportedScripts();
72 void clearOwnedObjects();
73 void clearContextGuards();
79 return m_engine && (!m_isInternal || !m_contextObject
80 || !QObjectPrivate::get(m_contextObject)->wasDeleted);
83 bool isInternal()
const {
return m_isInternal; }
84 void setInternal(
bool isInternal) { m_isInternal = isInternal; }
86 bool isJSContext()
const {
return m_isJSContext; }
87 void setJSContext(
bool isJSContext) { m_isJSContext = isJSContext; }
89 bool isPragmaLibraryContext()
const {
return m_isPragmaLibraryContext; }
90 void setPragmaLibraryContext(
bool library) { m_isPragmaLibraryContext = library; }
92 QQmlRefPointer<QQmlContextData> parent()
const {
return m_parent; }
99 if (m_ownedByParent) {
100 m_ownedByParent =
false;
105 void refreshExpressions();
107 void addOwnedObject(QQmlData *ownedObject);
108 QQmlData *ownedObjects()
const {
return m_ownedObjects; }
109 void setOwnedObjects(QQmlData *ownedObjects) { m_ownedObjects = ownedObjects; }
115 void installContext(QQmlData *ddata, QmlObjectKind kind);
117 QUrl resolvedUrl(
const QUrl &)
const;
121 QQmlContext *asQQmlContext()
123 if (!m_publicContext)
124 m_publicContext =
new QQmlContext(*
new QQmlContextPrivate(
this));
125 return m_publicContext;
128 QQmlContextPrivate *asQQmlContextPrivate()
130 return QQmlContextPrivate::get(asQQmlContext());
133 QObject *contextObject()
const {
return m_contextObject; }
134 void setContextObject(QObject *contextObject) { m_contextObject = contextObject; }
136 template<
typename HandleSelf,
typename HandleLinked>
137 void deepClearContextObject(
138 QObject *contextObject, HandleSelf &&handleSelf, HandleLinked &&handleLinked) {
139 for (QQmlContextData *lc = m_linkedContext.data(); lc; lc = lc->m_linkedContext.data()) {
141 if (lc->m_contextObject == contextObject)
142 lc->m_contextObject =
nullptr;
146 if (m_contextObject == contextObject)
147 m_contextObject =
nullptr;
150 void deepClearContextObject(QObject *contextObject)
152 deepClearContextObject(
154 [](QQmlContextData *self) { self->emitDestruction(); },
155 [](QQmlContextData *){});
158 QQmlEngine *engine()
const {
return m_engine; }
159 void setEngine(QQmlEngine *engine) { m_engine = engine; }
161 QQmlContext *publicContext()
const {
return m_publicContext; }
162 void clearPublicContext()
164 if (!m_publicContext)
167 m_publicContext =
nullptr;
168 if (m_ownedByPublicContext) {
169 m_ownedByPublicContext =
false;
174 int propertyIndex(
const QString &name)
const
176 ensurePropertyNames();
177 return m_propertyNameCache.value(name);
180 int propertyIndex(QV4::String *name)
const
182 ensurePropertyNames();
183 return m_propertyNameCache.value(name);
186 QString propertyName(
int index)
const
188 ensurePropertyNames();
189 return m_propertyNameCache.findId(index);
192 void addPropertyNameAndIndex(
const QString &name,
int index)
194 Q_ASSERT(!m_propertyNameCache.isEmpty());
195 m_propertyNameCache.add(name, index);
198 void setExpressions(QQmlJavaScriptExpression *expressions) { m_expressions = expressions; }
199 QQmlJavaScriptExpression *takeExpressions()
201 QQmlJavaScriptExpression *expressions = m_expressions;
202 m_expressions =
nullptr;
206 void setChildContexts(
const QQmlRefPointer<QQmlContextData> &childContexts)
208 m_childContexts = childContexts.data();
210 QQmlRefPointer<QQmlContextData> childContexts()
const {
return m_childContexts; }
211 QQmlRefPointer<QQmlContextData> takeChildContexts()
213 QQmlRefPointer<QQmlContextData> childContexts = m_childContexts;
214 m_childContexts =
nullptr;
215 return childContexts;
217 QQmlRefPointer<QQmlContextData> nextChild()
const {
return m_nextChild; }
219 int numIdValues()
const {
return m_idValueCount; }
220 void setIdValue(
int index, QObject *idValue);
221 bool isIdValueSet(
int index)
const {
return m_idValues[index].wasSet(); }
222 QQmlNotifier *idValueBindings(
int index)
const {
return m_idValues[index].bindings(); }
223 QObject *idValue(
int index)
const {
return m_idValues[index].data(); }
226 QString findObjectId(
const QObject *obj)
const;
234 QString urlString()
const;
236 void setBaseUrlString(
const QString &baseUrlString) { m_baseUrlString = baseUrlString; }
237 QString baseUrlString()
const
239 for (
const QQmlContextData *data =
this; data; data = data->m_parent) {
240 if (!data->m_baseUrlString.isEmpty())
241 return data->m_baseUrlString;
242 if (data->m_typeCompilationUnit)
243 return data->m_typeCompilationUnit->finalUrlString();
248 void setBaseUrl(
const QUrl &baseUrl) { m_baseUrl = baseUrl; }
251 for (
const QQmlContextData *data =
this; data; data = data->m_parent) {
252 if (!data->m_baseUrl.isEmpty())
253 return data->m_baseUrl;
254 if (data->m_typeCompilationUnit)
255 return data->m_typeCompilationUnit->finalUrl();
260 QQmlRefPointer<QQmlTypeNameCache> imports()
const {
return m_imports; }
261 void setImports(
const QQmlRefPointer<QQmlTypeNameCache> &imports) { m_imports = imports; }
263 QQmlIncubatorPrivate *incubator()
const {
return m_hasExtraObject ?
nullptr : m_incubator; }
264 void setIncubator(QQmlIncubatorPrivate *incubator)
266 Q_ASSERT(!m_hasExtraObject || m_extraObject ==
nullptr);
267 m_hasExtraObject =
false;
268 m_incubator = incubator;
271 QObject *extraObject()
const {
return m_hasExtraObject ? m_extraObject :
nullptr; }
272 void setExtraObject(QObject *extraObject)
274 Q_ASSERT(m_hasExtraObject || m_incubator ==
nullptr);
275 m_hasExtraObject =
true;
276 m_extraObject = extraObject;
279 bool isRootObjectInCreation()
const {
return m_isRootObjectInCreation; }
280 void setRootObjectInCreation(
bool rootInCreation) { m_isRootObjectInCreation = rootInCreation; }
282 QV4::ReturnedValue importedScripts()
const
284 if (m_hasWeakImportedScripts)
285 return m_weakImportedScripts.value();
287 return m_importedScripts.value();
289 void setImportedScripts(QV4::ExecutionEngine *engine, QV4::Value scripts) {
291 Q_ASSERT(!m_hasWeakImportedScripts);
292 m_importedScripts.set(engine, scripts);
296
297
298
299
300 void setImportedScripts(QV4::ExecutionEngine *engine, QV4::ReturnedValue scripts) {
302 Q_ASSERT(!m_hasWeakImportedScripts);
303 m_importedScripts.set(engine, scripts);
306 QQmlRefPointer<QQmlContextData> linkedContext()
const {
return m_linkedContext; }
307 void setLinkedContext(
const QQmlRefPointer<QQmlContextData> &context) { m_linkedContext = context; }
309 bool hasUnresolvedNames()
const {
return m_unresolvedNames; }
310 void setUnresolvedNames(
bool hasUnresolvedNames) { m_unresolvedNames = hasUnresolvedNames; }
312 QQmlComponentAttached *componentAttacheds()
const {
return m_componentAttacheds; }
313 void addComponentAttached(QQmlComponentAttached *attached);
315 void addExpression(QQmlJavaScriptExpression *expression);
317 bool valueTypesAreAddressable()
const {
318 return m_typeCompilationUnit && m_typeCompilationUnit->valueTypesAreAddressable();
321 bool valueTypesAreAssertable()
const {
322 return m_typeCompilationUnit && m_typeCompilationUnit->valueTypesAreAssertable();
326 friend class QQmlGuardedContextData;
327 friend class QQmlContextPrivate;
336 struct ContextGuard :
public QQmlGuard<QObject>
343 inline ContextGuard() : QQmlGuard<QObject>(&ContextGuard::objectDestroyedImpl,
nullptr), m_context(
nullptr) {}
344 inline ContextGuard &operator=(QObject *obj);
346 inline bool wasSet()
const;
348 QQmlNotifier *bindings() {
return &m_bindings; }
349 void setContext(
const QQmlRefPointer<QQmlContextData> &context)
351 m_context = context.data();
355 inline static void objectDestroyedImpl(QQmlGuardImpl *);
357 QTaggedPointer<QQmlContextData, Tag> m_context;
358 QQmlNotifier m_bindings;
364 Ownership ownership, QQmlContext *publicContext,
365 const QQmlRefPointer<QQmlContextData> &parent, QQmlEngine *engine =
nullptr)
366 : m_parent(parent.data()),
367 m_engine(engine ? engine : (parent.isNull() ?
nullptr : parent->engine())),
368 m_isInternal(
false), m_isJSContext(
false), m_isPragmaLibraryContext(
false),
369 m_unresolvedNames(
false), m_hasEmittedDestruction(
false), m_isRootObjectInCreation(
false),
370 m_ownedByParent(ownership == OwnedByParent),
371 m_ownedByPublicContext(ownership == OwnedByPublicContext), m_hasExtraObject(
false),
372 m_hasWeakImportedScripts(
false), m_dummy(0), m_publicContext(publicContext), m_incubator(
nullptr)
374 Q_ASSERT(!m_ownedByParent || !m_ownedByPublicContext);
378 m_nextChild = m_parent->m_childContexts;
380 m_nextChild->m_prevChild = &m_nextChild;
381 m_prevChild = &m_parent->m_childContexts;
382 m_parent->m_childContexts =
this;
387 bool hasExpressionsToRun(
bool isGlobalRefresh)
const
389 return m_expressions && (!isGlobalRefresh || m_unresolvedNames);
392 void refreshExpressionsRecursive(
bool isGlobal);
393 void refreshExpressionsRecursive(QQmlJavaScriptExpression *);
394 void initPropertyNames()
const;
396 void ensurePropertyNames()
const
398 if (m_propertyNameCache.isEmpty())
400 Q_ASSERT(!m_propertyNameCache.isEmpty());
404 QQmlContextData *m_parent =
nullptr;
405 QQmlEngine *m_engine =
nullptr;
407 mutable quint32 m_refCount = 1;
408 quint32 m_isInternal:1;
409 quint32 m_isJSContext:1;
410 quint32 m_isPragmaLibraryContext:1;
411 quint32 m_unresolvedNames:1;
412 quint32 m_hasEmittedDestruction:1;
413 quint32 m_isRootObjectInCreation:1;
414 quint32 m_ownedByParent:1;
415 quint32 m_ownedByPublicContext:1;
416 quint32 m_hasExtraObject:1;
417 quint32 m_hasWeakImportedScripts:1;
418 Q_DECL_UNUSED_MEMBER quint32 m_dummy:22;
419 QQmlContext *m_publicContext =
nullptr;
423 QQmlIncubatorPrivate *m_incubator;
425 QObject *m_extraObject;
429 QQmlRefPointer<QV4::ExecutableCompilationUnit> m_typeCompilationUnit;
432 int m_componentObjectIndex = -1;
435 mutable QV4::IdentifierHash m_propertyNameCache;
438 QObject *m_contextObject =
nullptr;
443
444
445
446 QV4::PersistentValue m_importedScripts = {};
447 QV4::WeakValue m_weakImportedScripts;
451 QString m_baseUrlString;
454 QQmlRefPointer<QQmlTypeNameCache> m_imports;
457 QQmlContextData *m_childContexts =
nullptr;
460 QQmlContextData *m_nextChild =
nullptr;
461 QQmlContextData **m_prevChild =
nullptr;
464 QQmlJavaScriptExpression *m_expressions =
nullptr;
467 QQmlData *m_ownedObjects =
nullptr;
470 QQmlGuardedContextData *m_contextGuards =
nullptr;
472 ContextGuard *m_idValues =
nullptr;
473 int m_idValueCount = 0;
476 QQmlRefPointer<QQmlContextData> m_linkedContext;
479 QQmlComponentAttached *m_componentAttacheds =
nullptr;