// Copyright (C) 2019 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include "qqmljsscope_p.h" #include "qqmljstypereader_p.h" #include "qqmljsimporter_p.h" #include "qqmljsutils_p.h" #include #include #include #include #include QT_BEGIN_NAMESPACE /*! \class QQmlJSScope \internal \brief Tracks the types for the QmlCompiler QQmlJSScope tracks the types used in qml for the QmlCompiler. Multiple QQmlJSScope objects might be created for the same conceptual type, except when reused due to extensive caching. Two QQmlJSScope objects are considered equal when they are backed by the same implementation, that is, they have the same internalName. The qualifiedName of the QQmlJSScope for a type imported from multiple modules will contain the name of one of the modules that imported it, which is not unique and might change depending on the caching in . */ using namespace Qt::StringLiterals; void QQmlJSScope::reparent(const QQmlJSScope::Ptr &parentScope, const QQmlJSScope::Ptr &childScope) { if (const QQmlJSScope::Ptr parent = childScope->m_parentScope.toStrongRef()) parent->m_childScopes.removeOne(childScope); if (parentScope) parentScope->m_childScopes.append(childScope); childScope->m_parentScope = parentScope; } QQmlJSScope::Ptr QQmlJSScope::clone(const ConstPtr &origin) { if (origin.isNull()) return QQmlJSScope::Ptr(); QQmlJSScope::Ptr cloned = create(); *cloned = *origin; if (QQmlJSScope::Ptr parent = cloned->parentScope()) parent->m_childScopes.append(cloned); return cloned; } void QQmlJSScope::insertJSIdentifier(const QString &name, const JavaScriptIdentifier &identifier) { Q_ASSERT(m_scopeType != QQmlJSScope::QMLScope); if (identifier.kind == JavaScriptIdentifier::LexicalScoped || identifier.kind == JavaScriptIdentifier::Injected || m_scopeType == QQmlJSScope::JSFunctionScope) { m_jsIdentifiers.insert(name, identifier); } else { auto targetScope = parentScope(); while (targetScope->m_scopeType != QQmlJSScope::JSFunctionScope) targetScope = targetScope->parentScope(); targetScope->m_jsIdentifiers.insert(name, identifier); } } void QQmlJSScope::insertPropertyIdentifier(const QQmlJSMetaProperty &property) { addOwnProperty(property); QQmlJSMetaMethod method(property.propertyName() + u"Changed"_s, u"void"_s); method.setMethodType(QQmlJSMetaMethod::Signal); method.setIsImplicitQmlPropertyChangeSignal(true); addOwnMethod(method); } bool QQmlJSScope::isIdInCurrentScope(const QString &id) const { return isIdInCurrentQmlScopes(id) || isIdInCurrentJSScopes(id); } bool QQmlJSScope::hasMethod(const QString &name) const { return QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode == QQmlJSScope::ExtensionNamespace) return false; return scope->m_methods.contains(name); }); } /*! Returns all methods visible from this scope including those of base types and extensions. \note Methods that get shadowed are not included and only the version visible from this scope is contained. Additionally method overrides are not included either, only the first visible version of any method is included. */ QHash QQmlJSScope::methods() const { QHash results; QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode == QQmlJSScope::ExtensionNamespace) return false; for (auto it = scope->m_methods.constBegin(); it != scope->m_methods.constEnd(); it++) { if (!results.contains(it.key())) results.insert(it.key(), it.value()); } return false; }); return results; } QList QQmlJSScope::methods(const QString &name) const { QList results; QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode == QQmlJSScope::ExtensionNamespace) return false; results.append(scope->ownMethods(name)); return false; }); return results; } QList QQmlJSScope::methods(const QString &name, QQmlJSMetaMethod::Type type) const { QList results; QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode == QQmlJSScope::ExtensionNamespace) return false; const auto ownMethods = scope->ownMethods(name); for (const auto &method : ownMethods) { if (method.methodType() == type) results.append(method); } return false; }); return results; } bool QQmlJSScope::hasEnumeration(const QString &name) const { return QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope) { return scope->m_enumerations.contains(name); }); } bool QQmlJSScope::hasEnumerationKey(const QString &name) const { return QQmlJSUtils::searchBaseAndExtensionTypes(this, [&](const QQmlJSScope *scope) { for (const auto &e : scope->m_enumerations) { if (e.keys().contains(name)) return true; } return false; }); } QQmlJSMetaEnum QQmlJSScope::enumeration(const QString &name) const { QQmlJSMetaEnum result; QQmlJSUtils::searchBaseAndExtensionTypes(this, [&](const QQmlJSScope *scope) { const auto it = scope->m_enumerations.find(name); if (it == scope->m_enumerations.end()) return false; result = *it; return true; }); return result; } QHash QQmlJSScope::enumerations() const { QHash results; QQmlJSUtils::searchBaseAndExtensionTypes(this, [&](const QQmlJSScope *scope) { for (auto it = scope->m_enumerations.constBegin(); it != scope->m_enumerations.constEnd(); it++) { if (!results.contains(it.key())) results.insert(it.key(), it.value()); } return false; }); return results; } QString QQmlJSScope::prettyName(QAnyStringView name) { const auto internal = "$internal$."_L1; const QString anonymous = "$anonymous$."_L1; QString pretty = name.toString(); if (pretty.startsWith(internal)) pretty = pretty.mid(internal.size()); else if (pretty.startsWith(anonymous)) pretty = pretty.mid(anonymous.size()); if (pretty == u"std::nullptr_t") return u"null"_s; if (pretty == u"void") return u"undefined"_s; return pretty; } /*! Returns if assigning \a assignedType to \a property would require an implicit component wrapping. */ bool QQmlJSScope::causesImplicitComponentWrapping(const QQmlJSMetaProperty &property, const QQmlJSScope::ConstPtr &assignedType) { // See QQmlComponentAndAliasResolver::findAndRegisterImplicitComponents() // for the logic in qqmltypecompiler // Note: unlike findAndRegisterImplicitComponents() we do not check whether // the property type is *derived* from QQmlComponent at some point because // this is actually meaningless (and in the case of QQmlComponent::create() // gets rejected in QQmlPropertyValidator): if the type is not a // QQmlComponent, we have a type mismatch because of assigning a Component // object to a non-Component property const bool propertyVerdict = property.type()->internalName() == u"QQmlComponent"; const bool assignedTypeVerdict = [&assignedType]() { // Note: nonCompositeBaseType covers the case when assignedType itself // is non-composite auto cppBase = nonCompositeBaseType(assignedType); Q_ASSERT(cppBase); // any QML type has (or must have) a C++ base type // See isUsableComponent() in qqmltypecompiler.cpp: along with checking // whether a type has a QQmlComponent static meta object (which we // substitute here with checking the first non-composite base for being // a QQmlComponent), it also excludes QQmlAbstractDelegateComponent // subclasses from implicit wrapping if (cppBase->internalName() == u"QQmlComponent") return false; for (; cppBase; cppBase = cppBase->baseType()) { if (cppBase->internalName() == u"QQmlAbstractDelegateComponent") return false; } return true; }(); return propertyVerdict && assignedTypeVerdict; } /*! \internal Returns true if the scope is the outermost element of a separate Component Either because it has been implicitly wrapped, e.g. due to an assignment to a Component property, or because it is the first (and only) child of a Component. For visitors: This method should only be called after implicit components are detected, that is, after QQmlJSImportVisitor::endVisit(UiProgram *) was called. */ bool QQmlJSScope::isComponentRootElement() const { if (m_flags.testFlag(WrappedInImplicitComponent)) return true; auto base = nonCompositeBaseType(parentScope()); // handles null parentScope() if (!base) return false; return base->internalName() == u"QQmlComponent"; } bool QQmlJSScope::isIdInCurrentQmlScopes(const QString &id) const { if (m_scopeType == QQmlJSScope::QMLScope) return m_properties.contains(id) || m_methods.contains(id) || m_enumerations.contains(id); const auto qmlScope = findCurrentQMLScope(parentScope()); return qmlScope->m_properties.contains(id) || qmlScope->m_methods.contains(id) || qmlScope->m_enumerations.contains(id); } bool QQmlJSScope::isIdInCurrentJSScopes(const QString &id) const { if (m_scopeType != QQmlJSScope::QMLScope && m_jsIdentifiers.contains(id)) return true; for (auto jsScope = parentScope(); jsScope; jsScope = jsScope->parentScope()) { if (jsScope->m_scopeType != QQmlJSScope::QMLScope && jsScope->m_jsIdentifiers.contains(id)) return true; } return false; } bool QQmlJSScope::isIdInjectedFromSignal(const QString &id) const { const auto found = findJSIdentifier(id); return found.has_value() && found->kind == JavaScriptIdentifier::Injected; } std::optional QQmlJSScope::findJSIdentifier(const QString &id) const { for (const auto *scope = this; scope; scope = scope->parentScope().data()) { if (scope->m_scopeType == QQmlJSScope::JSFunctionScope || scope->m_scopeType == QQmlJSScope::JSLexicalScope) { auto it = scope->m_jsIdentifiers.find(id); if (it != scope->m_jsIdentifiers.end()) return *it; } } return std::optional{}; } std::optional QQmlJSScope::JSIdentifier(const QString &id) const { auto it = m_jsIdentifiers.find(id); if (it != m_jsIdentifiers.end()) return *it; return std::optional{}; } static QQmlJSScope::ImportedScope qFindInlineComponents(QStringView typeName, const QQmlJSScope::ContextualTypes &contextualTypes) { const int separatorIndex = typeName.lastIndexOf(u'.'); // do not crash in typeName.sliced() when it starts or ends with an '.'. if (separatorIndex < 1 || separatorIndex >= typeName.size() - 1) return {}; const auto parentIt = contextualTypes.types().constFind(typeName.first(separatorIndex).toString()); if (parentIt == contextualTypes.types().constEnd()) return {}; auto inlineComponentParent = *parentIt; // find the inline components using BFS, as inline components defined in childrens are also // accessible from other qml documents. Same for inline components defined in a base class of // the parent. Use BFS over DFS as the inline components are probably not deeply-nested. QStringView inlineComponentName = typeName.sliced(separatorIndex + 1); QQueue candidatesForInlineComponents; candidatesForInlineComponents.enqueue(inlineComponentParent.scope); while (candidatesForInlineComponents.size()) { QQmlJSScope::ConstPtr current = candidatesForInlineComponents.dequeue(); if (!current) // if some type was not resolved, ignore it instead of crashing continue; if (current->isInlineComponent() && current->inlineComponentName() == inlineComponentName) { return { current, inlineComponentParent.revision }; } // check alternatively the inline components at layer 1 in current and basetype, then at // layer 2, etc... candidatesForInlineComponents.append(current->childScopes()); if (const auto base = current->baseType()) candidatesForInlineComponents.enqueue(base); } return {}; } QQmlJSScope::ImportedScope QQmlJSScope::findType( const QString &name, const QQmlJSScope::ContextualTypes &contextualTypes, QSet *usedTypes) { const auto useType = [&]() { if (usedTypes != nullptr) usedTypes->insert(name); }; auto type = contextualTypes.types().constFind(name); if (type != contextualTypes.types().constEnd()) { useType(); return *type; } const auto findListType = [&](const QString &prefix, const QString &postfix) -> ImportedScope { if (name.startsWith(prefix) && name.endsWith(postfix)) { const qsizetype prefixLength = prefix.length(); const QString &elementName = name.mid(prefixLength, name.length() - prefixLength - postfix.length()); const ImportedScope element = findType(elementName, contextualTypes, usedTypes); if (element.scope) { useType(); return { element.scope->listType(), element.revision }; } } return {}; }; switch (contextualTypes.context()) { case ContextualTypes::INTERNAL: { if (const auto listType = findListType(u"QList<"_s, u">"_s); listType.scope && !listType.scope->isReferenceType()) { return listType; } if (const auto listType = findListType(u"QQmlListProperty<"_s, u">"_s); listType.scope && listType.scope->isReferenceType()) { return listType; } // look for c++ namescoped enums! const auto colonColon = name.lastIndexOf(QStringLiteral("::")); if (colonColon == -1) break; const QString outerTypeName = name.left(colonColon); const auto outerType = contextualTypes.types().constFind(outerTypeName); if (outerType == contextualTypes.types().constEnd()) break; for (const auto &innerType : std::as_const(outerType->scope->m_childScopes)) { if (innerType->m_internalName == name) { useType(); return { innerType, outerType->revision }; } } break; } case ContextualTypes::QML: { // look after inline components const auto inlineComponent = qFindInlineComponents(name, contextualTypes); if (inlineComponent.scope) { useType(); return inlineComponent; } if (const auto listType = findListType(u"list<"_s, u">"_s); listType.scope) return listType; break; } } return {}; } QTypeRevision QQmlJSScope::resolveType( const QQmlJSScope::Ptr &self, const QQmlJSScope::ContextualTypes &context, QSet *usedTypes) { if (self->accessSemantics() == AccessSemantics::Sequence && self->internalName().startsWith(u"QQmlListProperty<"_s)) { self->setIsListProperty(true); } const QString baseTypeName = self->baseTypeName(); const auto baseType = findType(baseTypeName, context, usedTypes); if (!self->m_baseType.scope && !baseTypeName.isEmpty()) self->m_baseType = { baseType.scope, baseType.revision }; if (!self->m_attachedType && !self->m_attachedTypeName.isEmpty()) self->m_attachedType = findType(self->m_attachedTypeName, context, usedTypes).scope; if (!self->m_valueType && !self->m_valueTypeName.isEmpty()) self->m_valueType = findType(self->m_valueTypeName, context, usedTypes).scope; if (!self->m_extensionType) { if (self->m_extensionTypeName.isEmpty()) { if (self->accessSemantics() == AccessSemantics::Sequence) { // All sequence types are implicitly extended by JS Array. self->setExtensionTypeName(u"Array"_s); self->m_extensionType = context.arrayType(); } } else { self->m_extensionType = findType(self->m_extensionTypeName, context, usedTypes).scope; } } for (auto it = self->m_properties.begin(), end = self->m_properties.end(); it != end; ++it) { const QString typeName = it->typeName(); if (it->type() || typeName.isEmpty()) continue; if (const auto type = findType(typeName, context, usedTypes); type.scope) { it->setType(it->isList() ? type.scope->listType() : type.scope); continue; } const auto enumeration = self->m_enumerations.find(typeName); if (enumeration != self->m_enumerations.end()) { it->setType(it->isList() ? enumeration->type()->listType() : QQmlJSScope::ConstPtr(enumeration->type())); } } for (auto it = self->m_methods.begin(), end = self->m_methods.end(); it != end; ++it) { const QString returnTypeName = it->returnTypeName(); if (!it->returnType() && !returnTypeName.isEmpty()) { const auto returnType = findType(returnTypeName, context, usedTypes); it->setReturnType(returnType.scope); } auto parameters = it->parameters(); for (int i = 0, length = parameters.size(); i < length; ++i) { auto ¶meter = parameters[i]; if (const QString typeName = parameter.typeName(); !parameter.type() && !typeName.isEmpty()) { auto type = findType(typeName, context, usedTypes); if (type.scope && parameter.isList()) { type.scope = type.scope->listType(); parameter.setIsList(false); parameter.setIsPointer(false); parameter.setTypeName(type.scope ? type.scope->internalName() : QString()); } else if (type.scope && type.scope->isReferenceType()) { parameter.setIsPointer(true); } parameter.setType({ type.scope }); } } it->setParameters(parameters); } return baseType.revision; } void QQmlJSScope::updateChildScope( const QQmlJSScope::Ptr &childScope, const QQmlJSScope::Ptr &self, const QQmlJSScope::ContextualTypes &contextualTypes, QSet *usedTypes) { switch (childScope->scopeType()) { case QQmlJSScope::GroupedPropertyScope: QQmlJSUtils::searchBaseAndExtensionTypes( self.data(), [&](const QQmlJSScope *type, QQmlJSScope::ExtensionKind mode) { if (mode == QQmlJSScope::ExtensionNamespace) return false; const auto propertyIt = type->m_properties.find(childScope->internalName()); if (propertyIt != type->m_properties.end()) { childScope->m_baseType.scope = QQmlJSScope::ConstPtr(propertyIt->type()); if (propertyIt->type()) childScope->m_semantics = propertyIt->type()->accessSemantics(); childScope->setBaseTypeName(propertyIt->typeName()); return true; } return false; }); break; case QQmlJSScope::AttachedPropertyScope: if (const auto attachedBase = findType( childScope->internalName(), contextualTypes, usedTypes).scope) { childScope->m_baseType.scope = attachedBase->attachedType(); childScope->setBaseTypeName(attachedBase->attachedTypeName()); } break; default: break; } } template static QTypeRevision resolveTypesInternal( Resolver resolve, ChildScopeUpdater update, const QQmlJSScope::Ptr &self, const QQmlJSScope::ContextualTypes &contextualTypes, QSet *usedTypes) { const QTypeRevision revision = resolve(self, contextualTypes, usedTypes); // NB: constness ensures no detach const auto childScopes = self->childScopes(); for (auto it = childScopes.begin(), end = childScopes.end(); it != end; ++it) { const auto childScope = *it; update(childScope, self, contextualTypes, usedTypes); resolveTypesInternal(resolve, update, childScope, contextualTypes, usedTypes); // recursion } return revision; } QTypeRevision QQmlJSScope::resolveTypes( const QQmlJSScope::Ptr &self, const QQmlJSScope::ContextualTypes &contextualTypes, QSet *usedTypes) { const auto resolveAll = [](const QQmlJSScope::Ptr &self, const QQmlJSScope::ContextualTypes &contextualTypes, QSet *usedTypes) { resolveEnums(self, contextualTypes, usedTypes); resolveList(self, contextualTypes.arrayType()); return resolveType(self, contextualTypes, usedTypes); }; return resolveTypesInternal(resolveAll, updateChildScope, self, contextualTypes, usedTypes); } void QQmlJSScope::resolveNonEnumTypes( const QQmlJSScope::Ptr &self, const QQmlJSScope::ContextualTypes &contextualTypes, QSet *usedTypes) { resolveTypesInternal(resolveType, updateChildScope, self, contextualTypes, usedTypes); } /*! \internal Resolves all enums of self. Some enums happen to have an alias, e.g. when an enum is used as a flag, the enum will exist in two versions, once as enum (e.g. Qt::MouseButton) and once as a flag (e.g. Qt::MouseButtons). In this case, normally only the flag is exposed to the qt metatype system and tools like qmltc will have troubles when encountering the enum in signal parameters etc. To solve this problem, resolveEnums() will create a QQmlJSMetaEnum copy for the alias in case the 'self'-scope already does not have an enum called like the alias. */ void QQmlJSScope::resolveEnums( const QQmlJSScope::Ptr &self, const QQmlJSScope::ContextualTypes &contextualTypes, QSet *usedTypes) { // temporary hash to avoid messing up m_enumerations while iterators are active on it QHash toBeAppended; for (auto it = self->m_enumerations.begin(), end = self->m_enumerations.end(); it != end; ++it) { if (it->type()) continue; QQmlJSScope::Ptr enumScope = QQmlJSScope::create(); reparent(self, enumScope); enumScope->m_scopeType = EnumScope; QString typeName = it->typeName(); if (typeName.isEmpty()) typeName = QStringLiteral("int"); enumScope->setBaseTypeName(typeName); const auto type = findType(typeName, contextualTypes, usedTypes); enumScope->m_baseType = { type.scope, type.revision }; enumScope->m_semantics = AccessSemantics::Value; enumScope->m_internalName = self->internalName() + QStringLiteral("::") + it->name(); if (QString alias = it->alias(); !alias.isEmpty() && self->m_enumerations.constFind(alias) == self->m_enumerations.constEnd()) { auto aliasScope = QQmlJSScope::clone(enumScope); aliasScope->m_internalName = self->internalName() + QStringLiteral("::") + alias; QQmlJSMetaEnum cpy(*it); cpy.setType(QQmlJSScope::ConstPtr(aliasScope)); toBeAppended.insert(alias, cpy); } it->setType(QQmlJSScope::ConstPtr(enumScope)); } // no more iterators active on m_enumerations, so it can be changed safely now self->m_enumerations.insert(toBeAppended); } void QQmlJSScope::resolveList(const QQmlJSScope::Ptr &self, const QQmlJSScope::ConstPtr &arrayType) { if (self->listType() || self->accessSemantics() == AccessSemantics::Sequence) return; Q_ASSERT(!arrayType.isNull()); QQmlJSScope::Ptr listType = QQmlJSScope::create(); listType->setAccessSemantics(AccessSemantics::Sequence); listType->setValueTypeName(self->internalName()); if (self->isComposite()) { // There is no internalName for this thing. Just set the value type right away listType->setInternalName(u"QQmlListProperty<>"_s); listType->m_valueType = QQmlJSScope::ConstPtr(self); } else if (self->isReferenceType()) { listType->setInternalName(u"QQmlListProperty<%2>"_s.arg(self->internalName())); // Do not set a filePath on the list type, so that we have to generalize it // even in direct mode. } else { listType->setInternalName(u"QList<%2>"_s.arg(self->internalName())); listType->setFilePath(self->filePath()); } const QQmlJSImportedScope element = {self, QTypeRevision()}; const QQmlJSImportedScope array = {arrayType, QTypeRevision()}; QQmlJSScope::ContextualTypes contextualTypes( QQmlJSScope::ContextualTypes::INTERNAL, { { self->internalName(), element }, }, arrayType); QQmlJSScope::resolveTypes(listType, contextualTypes); Q_ASSERT(listType->valueType() == self); self->m_listType = listType; } void QQmlJSScope::resolveGeneralizedGroup( const Ptr &self, const ConstPtr &baseType, const QQmlJSScope::ContextualTypes &contextualTypes, QSet *usedTypes) { Q_ASSERT(baseType); // Generalized group properties are always composite, // which means we expect contextualTypes to be QML names. Q_ASSERT(self->isComposite()); self->m_baseType.scope = baseType; self->m_semantics = baseType->accessSemantics(); resolveNonEnumTypes(self, contextualTypes, usedTypes); } QQmlJSScope::ConstPtr QQmlJSScope::findCurrentQMLScope(const QQmlJSScope::ConstPtr &scope) { auto qmlScope = scope; while (qmlScope && qmlScope->m_scopeType != QQmlJSScope::QMLScope) qmlScope = qmlScope->parentScope(); return qmlScope; } bool QQmlJSScope::hasProperty(const QString &name) const { return QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode == QQmlJSScope::ExtensionNamespace) return false; return scope->m_properties.contains(name); }); } QQmlJSMetaProperty QQmlJSScope::property(const QString &name) const { QQmlJSMetaProperty prop; QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode == QQmlJSScope::ExtensionNamespace) return false; const auto it = scope->m_properties.find(name); if (it == scope->m_properties.end()) return false; prop = *it; return true; }); return prop; } /*! Returns all properties visible from this scope including those of base types and extensions. \note Properties that get shadowed are not included and only the version visible from this scope is contained. */ QHash QQmlJSScope::properties() const { QHash results; QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode == QQmlJSScope::ExtensionNamespace) return false; for (auto it = scope->m_properties.constBegin(); it != scope->m_properties.constEnd(); it++) { if (!results.contains(it.key())) results.insert(it.key(), it.value()); } return false; }); return results; } QQmlJSScope::AnnotatedScope QQmlJSScope::ownerOfProperty(const QQmlJSScope::ConstPtr &self, const QString &name) { QQmlJSScope::AnnotatedScope owner; QQmlJSUtils::searchBaseAndExtensionTypes( self, [&](const QQmlJSScope::ConstPtr &scope, QQmlJSScope::ExtensionKind mode) { if (mode == QQmlJSScope::ExtensionNamespace) return false; if (scope->hasOwnProperty(name)) { owner = { scope, mode }; return true; } return false; }); return owner; } void QQmlJSScope::setPropertyLocallyRequired(const QString &name, bool isRequired) { if (!isRequired) m_requiredPropertyNames.removeOne(name); else if (!m_requiredPropertyNames.contains(name)) m_requiredPropertyNames.append(name); } bool QQmlJSScope::isPropertyRequired(const QString &name) const { bool isRequired = false; QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (scope->isPropertyLocallyRequired(name)) { isRequired = true; return true; } // the hasOwnProperty() below only makes sense if our scope is // not an extension namespace if (mode == QQmlJSScope::ExtensionNamespace) return false; // If it has a property of that name, and that is not required, then none of the // base types matter. You cannot make a derived type's property required with // a "required" specification in a base type. return scope->hasOwnProperty(name); }); return isRequired; } bool QQmlJSScope::isPropertyLocallyRequired(const QString &name) const { return m_requiredPropertyNames.contains(name); } static_assert(QTypeInfo::isRelocatable, "We really want T to be relocatable as it improves QList performance"); void QQmlJSScope::addOwnPropertyBindingInQmlIROrder(const QQmlJSMetaPropertyBinding &binding, BindingTargetSpecifier specifier) { // the order: // * ordinary bindings are prepended to the binding array // * list bindings are properly ordered within each other, so basically // prepended "in bulk" // * bindings to default properties (which are not explicitly mentioned in // binding expression) are inserted by source location's offset switch (specifier) { case BindingTargetSpecifier::SimplePropertyTarget: { m_propertyBindingsArray.emplaceFront(binding.propertyName(), binding.sourceLocation().offset); break; } case BindingTargetSpecifier::ListPropertyTarget: { const auto bindingOnTheSameProperty = [&](const QQmlJSScope::QmlIRCompatibilityBindingData &x) { return x.propertyName == binding.propertyName(); }; // fake "prepend in bulk" by appending a list binding to the sequence of // bindings to the same property. there's an implicit QML language // guarantee that such sequence does not contain arbitrary in-between // bindings that do not belong to the same list property auto pos = std::find_if_not(m_propertyBindingsArray.begin(), m_propertyBindingsArray.end(), bindingOnTheSameProperty); Q_ASSERT(pos == m_propertyBindingsArray.begin() || std::prev(pos)->propertyName == binding.propertyName()); m_propertyBindingsArray.emplace(pos, binding.propertyName(), binding.sourceLocation().offset); break; } case BindingTargetSpecifier::UnnamedPropertyTarget: { // see QmlIR::PoolList<>::findSortedInsertionPoint() const auto findInsertionPoint = [this](const QQmlJSMetaPropertyBinding &x) { qsizetype pos = -1; for (auto it = m_propertyBindingsArray.cbegin(); it != m_propertyBindingsArray.cend(); ++it) { if (!(it->sourceLocationOffset <= x.sourceLocation().offset)) break; ++pos; } return pos; }; // see QmlIR::PoolList<>::insertAfter() const auto insertAfter = [this](qsizetype pos, const QQmlJSMetaPropertyBinding &x) { if (pos == -1) { m_propertyBindingsArray.emplaceFront(x.propertyName(), x.sourceLocation().offset); } else if (pos == m_propertyBindingsArray.size()) { m_propertyBindingsArray.emplaceBack(x.propertyName(), x.sourceLocation().offset); } else { // since we insert *after*, use (pos + 1) as insertion point m_propertyBindingsArray.emplace(pos + 1, x.propertyName(), x.sourceLocation().offset); } }; const qsizetype insertionPos = findInsertionPoint(binding); insertAfter(insertionPos, binding); break; } default: { Q_UNREACHABLE(); break; } } } QList QQmlJSScope::ownPropertyBindingsInQmlIROrder() const { QList qmlIrOrdered; qmlIrOrdered.reserve(m_propertyBindingsArray.size()); for (const auto &data : m_propertyBindingsArray) { const auto [first, last] = m_propertyBindings.equal_range(data.propertyName); Q_ASSERT(first != last); auto binding = std::find_if(first, last, [&](const QQmlJSMetaPropertyBinding &x) { return x.sourceLocation().offset == data.sourceLocationOffset; }); Q_ASSERT(binding != last); qmlIrOrdered.append(*binding); } return qmlIrOrdered; } bool QQmlJSScope::hasPropertyBindings(const QString &name) const { return QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode != QQmlJSScope::NotExtension) { Q_ASSERT(!scope->hasOwnPropertyBindings(name)); return false; } return scope->hasOwnPropertyBindings(name); }); } QList QQmlJSScope::propertyBindings(const QString &name) const { QList bindings; QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode != QQmlJSScope::NotExtension) { Q_ASSERT(!scope->hasOwnPropertyBindings(name)); return false; } const auto range = scope->ownPropertyBindings(name); for (auto it = range.first; it != range.second; ++it) bindings.append(*it); return false; }); return bindings; } bool QQmlJSScope::hasInterface(const QString &name) const { return QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode != QQmlJSScope::NotExtension) return false; return scope->m_interfaceNames.contains(name); }); } bool QQmlJSScope::isNameDeferred(const QString &name) const { bool isDeferred = false; QQmlJSUtils::searchBaseAndExtensionTypes(this, [&](const QQmlJSScope *scope) { const QStringList immediate = scope->ownImmediateNames(); if (!immediate.isEmpty()) { isDeferred = !immediate.contains(name); return true; } const QStringList deferred = scope->ownDeferredNames(); if (!deferred.isEmpty()) { isDeferred = deferred.contains(name); return true; } return false; }); return isDeferred; } QString QQmlJSScope::qualifiedNameFrom(const QString &moduleName, const QString &typeName, const QTypeRevision &firstRevision, const QTypeRevision &lastRevision) { QString qualifiedName = u"%1/%2 %3.%4"_s.arg(moduleName, typeName) .arg(firstRevision.hasMajorVersion() ? firstRevision.majorVersion() : 0) .arg(firstRevision.hasMinorVersion() ? firstRevision.minorVersion() : 0); if (firstRevision != lastRevision) { qualifiedName += u"-%1.%2"_s .arg(lastRevision.hasMajorVersion() ? lastRevision.majorVersion() : 0) .arg(lastRevision.hasMinorVersion() ? lastRevision.minorVersion() : 0); } return qualifiedName; } void QQmlJSScope::setBaseTypeName(const QString &baseTypeName) { m_flags.setFlag(HasBaseTypeError, false); m_baseTypeNameOrError = baseTypeName; } QString QQmlJSScope::baseTypeName() const { return m_flags.testFlag(HasBaseTypeError) ? QString() : m_baseTypeNameOrError; } void QQmlJSScope::setBaseTypeError(const QString &baseTypeError) { m_flags.setFlag(HasBaseTypeError); m_baseTypeNameOrError = baseTypeError; } QString QQmlJSScope::baseTypeError() const { return m_flags.testFlag(HasBaseTypeError) ? m_baseTypeNameOrError : QString(); } QString QQmlJSScope::attachedTypeName() const { QString name; QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode != QQmlJSScope::NotExtension) return false; if (scope->ownAttachedType().isNull()) return false; name = scope->ownAttachedTypeName(); return true; }); return name; } QQmlJSScope::ConstPtr QQmlJSScope::attachedType() const { QQmlJSScope::ConstPtr ptr; QQmlJSUtils::searchBaseAndExtensionTypes( this, [&](const QQmlJSScope *scope, QQmlJSScope::ExtensionKind mode) { if (mode != QQmlJSScope::NotExtension) return false; if (scope->ownAttachedType().isNull()) return false; ptr = scope->ownAttachedType(); return true; }); return ptr; } bool QQmlJSScope::isResolved() const { const bool nameIsEmpty = (m_scopeType == ScopeType::AttachedPropertyScope || m_scopeType == ScopeType::GroupedPropertyScope) ? m_internalName.isEmpty() : m_baseTypeNameOrError.isEmpty(); if (nameIsEmpty) return true; if (m_baseType.scope.isNull()) return false; if (isComposite() && !nonCompositeBaseType(baseType())) return false; return true; } QString QQmlJSScope::defaultPropertyName() const { QString name; QQmlJSUtils::searchBaseAndExtensionTypes(this, [&](const QQmlJSScope *scope) { name = scope->ownDefaultPropertyName(); return !name.isEmpty(); }); return name; } QString QQmlJSScope::parentPropertyName() const { QString name; QQmlJSUtils::searchBaseAndExtensionTypes(this, [&](const QQmlJSScope *scope) { name = scope->ownParentPropertyName(); return !name.isEmpty(); }); return name; } bool QQmlJSScope::isFullyResolved() const { bool baseResolved = true; QQmlJSUtils::searchBaseAndExtensionTypes(this, [&](const QQmlJSScope *scope) { if (!scope->isResolved()) { baseResolved = false; return true; } return false; }); return baseResolved; } QQmlJSScope::Import::Import(QString prefix, QString name, QTypeRevision version, bool isFile, bool isDependency) : m_prefix(std::move(prefix)), m_name(std::move(name)), m_version(version), m_isFile(isFile), m_isDependency(isDependency) { } bool QQmlJSScope::Import::isValid() const { return !m_name.isEmpty(); } QQmlJSScope::Export::Export( QString package, QString type, QTypeRevision version, QTypeRevision revision) : m_package(std::move(package)) , m_type(std::move(type)) , m_version(std::move(version)) , m_revision(std::move(revision)) { } bool QQmlJSScope::Export::isValid() const { return m_version.isValid() || !m_package.isEmpty() || !m_type.isEmpty(); } void QDeferredFactory::populate(const QSharedPointer &scope) const { scope->setQualifiedName(m_qualifiedName); scope->setModuleName(m_moduleName); QQmlJSTypeReader typeReader(m_importer, m_filePath); typeReader(scope); m_importer->m_globalWarnings.append(typeReader.errors()); scope->setInternalName(internalName()); QQmlJSScope::resolveEnums(scope, m_importer->builtinInternalNames()); QQmlJSScope::resolveList(scope, m_importer->builtinInternalNames().arrayType()); if (m_isSingleton && !scope->isSingleton()) { m_importer->m_globalWarnings.append( { QStringLiteral( "Type %1 declared as singleton in qmldir but missing pragma Singleton") .arg(scope->internalName()), QtCriticalMsg, QQmlJS::SourceLocation() }); scope->setIsSingleton(true); } else if (!m_isSingleton && scope->isSingleton()) { m_importer->m_globalWarnings.append( { QStringLiteral("Type %1 not declared as singleton in qmldir " "but using pragma Singleton") .arg(scope->internalName()), QtCriticalMsg, QQmlJS::SourceLocation() }); scope->setIsSingleton(false); } } bool QQmlJSScope::canAssign(const QQmlJSScope::ConstPtr &derived) const { if (!derived) return false; // expect this and derived types to have non-composite bases Q_ASSERT(!isComposite() || nonCompositeBaseType(baseType())); Q_ASSERT(nonCompositeBaseType(derived)); // the logic with isBaseComponent (as well as the way we set this flag) // feels wrong - QTBUG-101940 const bool isBaseComponent = [this]() { if (internalName() == u"QQmlComponent") return true; else if (isComposite()) return false; for (auto cppBase = nonCompositeBaseType(baseType()); cppBase; cppBase = cppBase->baseType()) { if (cppBase->internalName() == u"QQmlAbstractDelegateComponent") return true; } return false; }(); QDuplicateTracker seen; for (auto scope = derived; !scope.isNull() && !seen.hasSeen(scope); scope = scope->baseType()) { if (isSameType(scope)) return true; if (isBaseComponent && scope->internalName() == u"QObject"_s) return true; } if (internalName() == u"QVariant"_s || internalName() == u"QJSValue"_s) return true; return isListProperty() && valueType()->canAssign(derived); } bool QQmlJSScope::isInCustomParserParent() const { for (const auto *scope = this; scope; scope = scope->parentScope().get()) { if (!scope->baseType().isNull() && scope->baseType()->hasCustomParser()) return true; } return false; } /*! * \internal * if this->isInlineComponent(), then this getter returns the name of the inline * component. */ std::optional QQmlJSScope::inlineComponentName() const { Q_ASSERT(isInlineComponent() == m_inlineComponentName.has_value()); return m_inlineComponentName; } /*! * \internal * If this type is part of an inline component, return its name. Otherwise, if this type * is part of the document root, return the document root name. */ QQmlJSScope::InlineComponentOrDocumentRootName QQmlJSScope::enclosingInlineComponentName() const { for (auto *type = this; type; type = type->parentScope().get()) { if (type->isInlineComponent()) return *type->inlineComponentName(); } return RootDocumentNameType(); } /*! \internal Returns true if the current type is creatable by checking all the required base classes. "Uncreatability" is only inherited from base types for composite types (in qml) and not for non-composite types (c++). For the exact definition: A type is uncreatable if and only if one of its composite base type or its first non-composite base type matches following criteria: \list \li the base type is a singleton, or \li the base type is an attached type, or \li the base type is a C++ type with the QML_UNCREATABLE or QML_ANONYMOUS macro, or \li the base type is a type without default constructor (in that case, it really needs QML_UNCREATABLE or QML_ANONYMOUS) \endlist */ bool QQmlJSScope::isCreatable() const { auto isCreatableNonRecursive = [](const QQmlJSScope *scope) { return scope->hasCreatableFlag() && !scope->isSingleton() && scope->scopeType() == QMLScope; }; for (const QQmlJSScope* scope = this; scope; scope = scope->baseType().get()) { if (!scope->isComposite()) { // just check the first nonComposite (c++) base for isCreatableNonRecursive() and then stop return isCreatableNonRecursive(scope); } else { // check all composite (qml) bases for isCreatableNonRecursive(). if (isCreatableNonRecursive(scope)) return true; } } // no uncreatable bases found return false; } QT_END_NAMESPACE