37 static QQmlRefPointer<QQmlContextData> createRefCounted(
38 const QQmlRefPointer<QQmlContextData> &parent)
40 return QQmlRefPointer<QQmlContextData>(
new QQmlContextData(RefCounted,
nullptr, parent),
41 QQmlRefPointer<QQmlContextData>::Adopt);
45 static QQmlRefPointer<QQmlContextData> createChild(
46 const QQmlRefPointer<QQmlContextData> &parent)
48 Q_ASSERT(!parent.isNull());
49 return QQmlRefPointer<QQmlContextData>(
new QQmlContextData(OwnedByParent,
nullptr, parent));
52 void addref()
const { ++m_refCount; }
53 void release()
const {
if (--m_refCount == 0)
delete this; }
54 int count()
const {
return m_refCount; }
55 int refCount()
const {
return m_refCount; }
57 QQmlRefPointer<QV4::ExecutableCompilationUnit> typeCompilationUnit()
const
59 return m_typeCompilationUnit;
61 void setTypeCompilationUnit(
const QQmlRefPointer<QV4::ExecutableCompilationUnit> &unit);
63 void initFromTypeCompilationUnit(
const QQmlRefPointer<QV4::ExecutableCompilationUnit> &unit,
64 int subComponentIndex);
66 static QQmlRefPointer<QQmlContextData> get(QQmlContext *context) {
67 return QQmlContextPrivate::get(context)->m_data;
70 void emitDestruction();
71 void clearExpressions();
72 void clearContextRecursively();
73 void clearChildrenAndSiblings();
74 void clearImportedScripts();
75 void clearOwnedObjects();
76 void clearContextGuards();
82 return m_engine && (!m_isInternal || !m_contextObject
83 || !QObjectPrivate::get(m_contextObject)->wasDeleted);
86 bool isInternal()
const {
return m_isInternal; }
87 void setInternal(
bool isInternal) { m_isInternal = isInternal; }
89 bool isJSContext()
const {
return m_isJSContext; }
90 void setJSContext(
bool isJSContext) { m_isJSContext = isJSContext; }
92 bool isPragmaLibraryContext()
const {
return m_isPragmaLibraryContext; }
93 void setPragmaLibraryContext(
bool library) { m_isPragmaLibraryContext = library; }
95 QQmlRefPointer<QQmlContextData> parent()
const {
return m_parent; }
102 if (m_ownedByParent) {
103 m_ownedByParent =
false;
108 void refreshExpressions();
110 void addOwnedObject(QQmlData *ownedObject);
111 QQmlData *ownedObjects()
const {
return m_ownedObjects; }
112 void setOwnedObjects(QQmlData *ownedObjects) { m_ownedObjects = ownedObjects; }
118 void installContext(QQmlData *ddata, QmlObjectKind kind);
120 QUrl resolvedUrl(
const QUrl &)
const;
124 QQmlContext *asQQmlContext()
126 if (!m_publicContext)
127 m_publicContext =
new QQmlContext(*
new QQmlContextPrivate(
this));
128 return m_publicContext;
131 QQmlContextPrivate *asQQmlContextPrivate()
133 return QQmlContextPrivate::get(asQQmlContext());
136 QObject *contextObject()
const {
return m_contextObject; }
137 void setContextObject(QObject *contextObject) { m_contextObject = contextObject; }
139 template<
typename HandleSelf,
typename HandleLinked>
140 void deepClearContextObject(
141 QObject *contextObject, HandleSelf &&handleSelf, HandleLinked &&handleLinked) {
142 for (QQmlContextData *lc = m_linkedContext.data(); lc; lc = lc->m_linkedContext.data()) {
144 if (lc->m_contextObject == contextObject)
145 lc->m_contextObject =
nullptr;
149 if (m_contextObject == contextObject)
150 m_contextObject =
nullptr;
153 void deepClearContextObject(QObject *contextObject)
155 deepClearContextObject(
157 [](QQmlContextData *self) { self->emitDestruction(); },
158 [](QQmlContextData *){});
161 QQmlEngine *engine()
const {
return m_engine; }
162 void setEngine(QQmlEngine *engine) { m_engine = engine; }
164 QQmlContext *publicContext()
const {
return m_publicContext; }
165 void clearPublicContext()
167 if (!m_publicContext)
170 m_publicContext =
nullptr;
171 if (m_ownedByPublicContext) {
172 m_ownedByPublicContext =
false;
177 int propertyIndex(
const QString &name)
const
179 ensurePropertyNames();
180 return m_propertyNameCache.value(name);
183 int propertyIndex(QV4::Heap::String *name)
const
185 ensurePropertyNames();
186 return m_propertyNameCache.value(name);
189 QString propertyName(
int index)
const
191 ensurePropertyNames();
192 return m_propertyNameCache.key<QString>(index);
195 void addPropertyNameAndIndex(
const QString &name,
int index)
197 Q_ASSERT(m_propertyNameCache.isValid());
198 m_propertyNameCache.add(name, index);
201 void setExpressions(QQmlJavaScriptExpression *expressions) { m_expressions = expressions; }
202 QQmlJavaScriptExpression *takeExpressions()
204 QQmlJavaScriptExpression *expressions = m_expressions;
205 m_expressions =
nullptr;
209 void setChildContexts(
const QQmlRefPointer<QQmlContextData> &childContexts)
211 m_childContexts = childContexts.data();
213 QQmlRefPointer<QQmlContextData> childContexts()
const {
return m_childContexts; }
214 QQmlRefPointer<QQmlContextData> takeChildContexts()
216 QQmlRefPointer<QQmlContextData> childContexts = m_childContexts;
217 m_childContexts =
nullptr;
218 return childContexts;
220 QQmlRefPointer<QQmlContextData> nextChild()
const {
return m_nextChild; }
222 int numIdValues()
const {
return m_idValueCount; }
223 void setIdValue(
int index, QObject *idValue);
224 bool isIdValueSet(
int index)
const {
return m_idValues[index].wasSet(); }
225 QQmlNotifier *idValueBindings(
int index)
const {
return m_idValues[index].bindings(); }
226 QObject *idValue(
int index)
const {
return m_idValues[index].data(); }
229 QString findObjectId(
const QObject *obj)
const;
237 QString urlString()
const;
239 void setBaseUrlString(
const QString &baseUrlString) { m_baseUrlString = baseUrlString; }
240 QString baseUrlString()
const
242 for (
const QQmlContextData *data =
this; data; data = data->m_parent) {
243 if (!data->m_baseUrlString.isEmpty())
244 return data->m_baseUrlString;
245 if (data->m_typeCompilationUnit)
246 return data->m_typeCompilationUnit->finalUrlString();
251 void setBaseUrl(
const QUrl &baseUrl) { m_baseUrl = baseUrl; }
254 for (
const QQmlContextData *data =
this; data; data = data->m_parent) {
255 if (!data->m_baseUrl.isEmpty())
256 return data->m_baseUrl;
257 if (data->m_typeCompilationUnit)
258 return data->m_typeCompilationUnit->finalUrl();
263 QQmlRefPointer<QQmlTypeNameCache> imports()
const {
return m_imports; }
264 void setImports(
const QQmlRefPointer<QQmlTypeNameCache> &imports) { m_imports = imports; }
266 QQmlIncubatorPrivate *incubator()
const {
return m_hasExtraObject ?
nullptr : m_incubator; }
267 void setIncubator(QQmlIncubatorPrivate *incubator)
269 Q_ASSERT(!m_hasExtraObject || m_extraObject ==
nullptr);
270 m_hasExtraObject =
false;
271 m_incubator = incubator;
274 QObject *extraObject()
const {
return m_hasExtraObject ? m_extraObject :
nullptr; }
275 void setExtraObject(QObject *extraObject)
277 Q_ASSERT(m_hasExtraObject || m_incubator ==
nullptr);
278 m_hasExtraObject =
true;
279 m_extraObject = extraObject;
282 bool isRootObjectInCreation()
const {
return m_isRootObjectInCreation; }
283 void setRootObjectInCreation(
bool rootInCreation) { m_isRootObjectInCreation = rootInCreation; }
285 QV4::ReturnedValue importedScripts()
const
287 if (m_hasWeakImportedScripts)
288 return m_weakImportedScripts.value();
290 return m_importedScripts.value();
292 void setImportedScripts(QV4::ExecutionEngine *engine, QV4::Value scripts) {
294 Q_ASSERT(!m_hasWeakImportedScripts);
295 m_importedScripts.set(engine, scripts);
299
300
301
302
303 void setImportedScripts(QV4::ExecutionEngine *engine, QV4::ReturnedValue scripts) {
305 Q_ASSERT(!m_hasWeakImportedScripts);
306 m_importedScripts.set(engine, scripts);
309 QQmlRefPointer<QQmlContextData> linkedContext()
const {
return m_linkedContext; }
310 void setLinkedContext(
const QQmlRefPointer<QQmlContextData> &context) { m_linkedContext = context; }
312 bool hasUnresolvedNames()
const {
return m_unresolvedNames; }
313 void setUnresolvedNames(
bool hasUnresolvedNames) { m_unresolvedNames = hasUnresolvedNames; }
315 QQmlComponentAttached *componentAttacheds()
const {
return m_componentAttacheds; }
316 void addComponentAttached(QQmlComponentAttached *attached);
318 void addExpression(QQmlJavaScriptExpression *expression);
320 bool valueTypesAreAddressable()
const {
321 return m_typeCompilationUnit && m_typeCompilationUnit->valueTypesAreAddressable();
324 bool valueTypesAreAssertable()
const {
325 return m_typeCompilationUnit && m_typeCompilationUnit->valueTypesAreAssertable();
329 friend class QQmlGuardedContextData;
330 friend class QQmlContextPrivate;
339 struct ContextGuard :
public QQmlGuard<QObject>
346 inline ContextGuard() : QQmlGuard<QObject>(&ContextGuard::objectDestroyedImpl,
nullptr), m_context(
nullptr) {}
347 inline ContextGuard &operator=(QObject *obj);
349 inline bool wasSet()
const;
351 QQmlNotifier *bindings() {
return &m_bindings; }
352 void setContext(
const QQmlRefPointer<QQmlContextData> &context)
354 m_context = context.data();
358 inline static void objectDestroyedImpl(QQmlGuardImpl *);
360 QTaggedPointer<QQmlContextData, Tag> m_context;
361 QQmlNotifier m_bindings;
367 Ownership ownership, QQmlContext *publicContext,
368 const QQmlRefPointer<QQmlContextData> &parent, QQmlEngine *engine =
nullptr)
369 : m_parent(parent.data()),
370 m_engine(engine ? engine : (parent.isNull() ?
nullptr : parent->engine())),
371 m_isInternal(
false), m_isJSContext(
false), m_isPragmaLibraryContext(
false),
372 m_unresolvedNames(
false), m_hasEmittedDestruction(
false), m_isRootObjectInCreation(
false),
373 m_ownedByParent(ownership == OwnedByParent),
374 m_ownedByPublicContext(ownership == OwnedByPublicContext), m_hasExtraObject(
false),
375 m_hasWeakImportedScripts(
false), m_dummy(0), m_publicContext(publicContext), m_incubator(
nullptr)
377 Q_ASSERT(!m_ownedByParent || !m_ownedByPublicContext);
381 m_nextChild = m_parent->m_childContexts;
383 m_nextChild->m_prevChild = &m_nextChild;
384 m_prevChild = &m_parent->m_childContexts;
385 m_parent->m_childContexts =
this;
390 bool hasExpressionsToRun(
bool isGlobalRefresh)
const
392 return m_expressions && (!isGlobalRefresh || m_unresolvedNames);
395 void refreshExpressionsRecursive(
bool isGlobal);
396 void refreshExpressionsRecursive(QQmlJavaScriptExpression *);
397 void initPropertyNames()
const;
399 void ensurePropertyNames()
const
401 if (!m_propertyNameCache.isValid())
403 Q_ASSERT(m_propertyNameCache.isValid());
407 QQmlContextData *m_parent =
nullptr;
408 QQmlEngine *m_engine =
nullptr;
410 mutable quint32 m_refCount = 1;
411 quint32 m_isInternal:1;
412 quint32 m_isJSContext:1;
413 quint32 m_isPragmaLibraryContext:1;
414 quint32 m_unresolvedNames:1;
415 quint32 m_hasEmittedDestruction:1;
416 quint32 m_isRootObjectInCreation:1;
417 quint32 m_ownedByParent:1;
418 quint32 m_ownedByPublicContext:1;
419 quint32 m_hasExtraObject:1;
420 quint32 m_hasWeakImportedScripts:1;
421 Q_DECL_UNUSED_MEMBER quint32 m_dummy:22;
422 QQmlContext *m_publicContext =
nullptr;
426 QQmlIncubatorPrivate *m_incubator;
428 QObject *m_extraObject;
432 QQmlRefPointer<QV4::ExecutableCompilationUnit> m_typeCompilationUnit;
435 int m_componentObjectIndex = -1;
438 mutable QV4::IdentifierHash m_propertyNameCache;
441 QObject *m_contextObject =
nullptr;
446
447
448
449 QV4::PersistentValue m_importedScripts = {};
450 QV4::WeakValue m_weakImportedScripts;
454 QString m_baseUrlString;
457 QQmlRefPointer<QQmlTypeNameCache> m_imports;
460 QQmlContextData *m_childContexts =
nullptr;
463 QQmlContextData *m_nextChild =
nullptr;
464 QQmlContextData **m_prevChild =
nullptr;
467 QQmlJavaScriptExpression *m_expressions =
nullptr;
470 QQmlData *m_ownedObjects =
nullptr;
473 QQmlGuardedContextData *m_contextGuards =
nullptr;
475 ContextGuard *m_idValues =
nullptr;
476 int m_idValueCount = 0;
479 QQmlRefPointer<QQmlContextData> m_linkedContext;
482 QQmlComponentAttached *m_componentAttacheds =
nullptr;