12#if defined(__GNUC__
) && __GNUC__
>= 11
13# pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
26#include <QtQml/private/qqmljslexer_p.h>
27#include <QtQml/private/qqmljsparser_p.h>
28#include <QtQml/private/qqmljsengine_p.h>
29#include <QtQml/private/qqmljsastvisitor_p.h>
30#include <QtQml/private/qqmljsast_p.h>
32#include <QtCore/QScopeGuard>
33#include <QtCore/QRegularExpression>
35#include <QtCore/QBasicMutex>
43using namespace Qt::StringLiterals;
52 QString version = QString::number(majorVersion);
53 if (majorVersion == Version::Latest)
54 version = QLatin1String(
"Latest");
55 else if (majorVersion == Version::Undefined)
57 QRegularExpression moduleRe(QLatin1String(R"(\A\w+(?:\.\w+)*\Z)"));
58 auto m = moduleRe.match(uri);
61 .error(Path::tr(
"Invalid module name in import %1").arg(uri))
62 .handle(errorHandler);
69 .withField(Fields::moduleIndexWithUri)
71 .withKey(version.majorSymbolicString())
72 .withField(Fields::moduleScope)
73 .withKey(version.minorString());
78 Version v = Version::fromString(version);
81 return moduleScopePath(uri, v, errorHandler);
95 cont = cont && self.invokeVisitorOnField(visitor, Fields::comments, m_comments);
102 updatePathFromOwnerMultiMap(m_enumerations, newPath.withField(Fields::enumerations));
103 updatePathFromOwnerQList(m_objects, newPath.withField(Fields::objects));
112 bool cont = CommentableDomElement::iterateDirectSubpaths(self, visitor);
113 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::name), name());
114 cont = cont && self.invokeVisitorOnField(visitor, Fields::enumerations, m_enumerations);
115 cont = cont && self.invokeVisitorOnField(visitor, Fields::objects, m_objects);
116 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isSingleton),
118 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isCreatable),
120 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isComposite),
122 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::attachedTypeName),
124 cont = cont && self.invokeVisitorOnReference(visitor, Fields::attachedType,
attachedTypePath(self
));
130 if (name == Fields::name)
131 return self.wrap(
PathEls::Field(Fields::name), m_name);
132 if (name == Fields::objects)
133 return self.wrap(
PathEls::Field(Fields::objects), m_objects);
135 return DomBase::field(self, name);
140 return appendUpdatableElementInQList(
pathFromOwner().withField(Fields::objects), m_objects, object,
146 bool cont = Component::iterateDirectSubpaths(self, visitor);
147 cont = cont && self.invokeVisitorOnField(visitor, Fields::ids, m_ids);
148 cont = cont && self.invokeVisitorOnLazyField(visitor, Fields::subComponents, [
this, &self]() {
149 return this->subComponents(self);
151 if (m_nameIdentifiers) {
152 cont = cont && visitor(
PathEls::Field(Fields::nameIdentifiers), [
this, &self]() {
153 return self.subScriptElementWrapperItem(m_nameIdentifiers);
162 updatePathFromOwnerMultiMap(m_ids, newPath.withField(Fields::annotations));
167 if (name().contains(QLatin1Char(
'.'))) {
171 .writeRegion(fLoc, ComponentKeywordRegion)
173 .writeRegion(fLoc, IdentifierRegion, name().split(QLatin1Char(
'.')).last())
174 .writeRegion(fLoc, ColonTokenRegion)
177 self.field(Fields::objects).index(0).writeOut(lw);
183 const QSet<QString> cNames = components.keys();
185 if (!myNameDot.isEmpty())
186 myNameDot += QLatin1Char(
'.');
187 QList<QString> subNames;
188 for (
const QString &cName : cNames)
189 if (cName.startsWith(myNameDot)
190 && !QStringView(cName).mid(myNameDot.size()).contains(QLatin1Char(
'.'))
192 subNames.append(cName);
193 std::sort(subNames.begin(), subNames.end());
201 for (
const QString &cName : subComponentsNames(self))
202 for (
const DomItem &comp : components.key(cName).values())
210 return Version(Latest, Latest);
211 QRegularExpression r(
212 QRegularExpression::anchoredPattern(QStringLiteral(uR"(([0-9]*)(?:\.([0-9]*))?)")));
213 auto m = r.matchView(v);
216 int majorV = m.capturedView(1).toInt(&ok);
218 majorV = Version::Undefined;
219 int minorV = m.capturedView(2).toInt(&ok);
221 minorV = Version::Undefined;
222 return Version(majorV, minorV);
231 return majorVersion == Latest && minorVersion == Latest;
236 return majorVersion >= 0 && minorVersion >= 0;
243 if (minorVersion < 0) {
244 if (majorVersion < 0)
245 return QLatin1String(
".");
247 return QString::number(majorVersion);
249 if (majorVersion < 0)
250 return QLatin1String(
".") + QString::number(minorVersion);
251 return QString::number(majorVersion) + QChar::fromLatin1(
'.') + QString::number(minorVersion);
257 cont = cont && self.invokeVisitorOnField(visitor, Fields::majorVersion, majorVersion);
258 cont = cont && self.invokeVisitorOnField(visitor, Fields::minorVersion, minorVersion);
259 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isLatest),
isLatest());
260 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isValid),
isValid());
261 cont = cont && self.invokeVisitorOnLazyField(visitor, Fields::stringValue, [
this]() {
262 return this->stringValue();
269 static QRegularExpression res(QRegularExpression::anchoredPattern(QStringLiteral(
270 uR"((?<uri>\w+(?:\.\w+)*)(?:\W+(?<version>[0-9]+(?:\.[0-9]*)?))?(?:\W+as\W+(?<id>\w+))?$)")));
277 auto m = importRe().match(importStr);
279 if (v.majorVersion == Version::Undefined && v.minorVersion == Version::Undefined)
280 v = Version::fromString(m.captured(2));
281 else if (!m.captured(u"version").isEmpty())
283 .warning(tr(
"Version %1 in import string '%2' overridden by explicit "
285 .arg(m.captured(2), importStr, v.stringValue())
)
287 QString resolvedImportId;
288 if (importId.isEmpty()) {
289 resolvedImportId = m.captured(u"importId");
291 if (!m.captured(u"importId").isEmpty()) {
293 .warning(tr(
"namespace %1 in import string '%2' overridden by explicit "
295 .arg(m.captured(u"importId"), importStr, importId)
)
298 resolvedImportId = importId;
301 return Import(QmlUri::fromUriString(m.captured(u"uri").trimmed()), v, resolvedImportId);
304 .error(tr(
"Unexpected URI format in import '%1'").arg(importStr)
)
310 const QString &importStr,
const QString &importId,
const ErrorHandler &)
312 return Import(QmlUri::fromDirectoryString(importStr),
Version(), importId);
318 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::uri), uri.toString());
319 cont = cont && self.invokeVisitorOnField(visitor, Fields::version, version);
320 if (!importId.isEmpty())
321 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::importId), importId);
323 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::implicit),
implicit);
324 cont = cont && self.invokeVisitorOnField(visitor, Fields::comments, comments);
335 if (std::shared_ptr<QmlFile> qmlFilePtr = self.ownerAs<QmlFile>())
336 code = qmlFilePtr->code();
343 quint32 start = fLoc->info().fullRegion.offset;
344 if (size_t(code.size()) >= start) {
346 QChar c = code.at(--start);
348 if (++preNewlines == 2)
350 }
else if (!c.isSpace())
354 if (preNewlines == 0)
357 ow.ensureNewline(preNewlines);
358 ow.writeRegion(fLoc, ImportTokenRegion).ensureSpace();
359 ow.writeRegion(fLoc, ImportUriRegion, uri.toString());
360 if (uri.isModule()) {
361 QString vString = version.stringValue();
362 if (!vString.isEmpty())
363 ow.ensureSpace().writeRegion(fLoc, VersionRegion, vString);
365 if (!importId.isEmpty()) {
367 .writeRegion(fLoc, AsTokenRegion)
369 .writeRegion(fLoc, IdNameRegion, importId);
378 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::name), name);
379 cont = cont && self.invokeVisitorOnReference(visitor, Fields::referredObject, referredObjectPath);
380 cont = cont && self.invokeVisitorOnField(visitor, Fields::comments, comments);
381 cont = cont && self.invokeVisitorOnField(visitor, Fields::annotations, annotations);
382 cont = cont && self.invokeVisitorOnField(visitor, Fields::value, value);
388 updatePathFromOwnerQList(annotations, newPath.withField(Fields::annotations));
393 return appendUpdatableElementInQList(selfPathFromOwner.withField(Fields::annotations), annotations,
401 bool cont = CommentableDomElement::iterateDirectSubpaths(self, visitor);
402 if (!idStr().isEmpty())
403 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::idStr), idStr());
404 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::name), name());
405 if (!prototypePaths().isEmpty())
406 cont = cont && self.invokeVisitorOnReferences(visitor, Fields::prototypes, m_prototypePaths);
408 cont = cont && self.invokeVisitorOnReference(visitor, Fields::nextScope,
nextScopePath());
409 cont = cont && self.invokeVisitorOnField(visitor, Fields::propertyDefs, m_propertyDefs);
410 cont = cont && self.invokeVisitorOnField(visitor, Fields::bindings, m_bindings);
411 cont = cont && self.invokeVisitorOnField(visitor, Fields::methods, m_methods);
412 cont = cont && self.invokeVisitorOnField(visitor, Fields::children, m_children);
413 cont = cont && self.invokeVisitorOnField(visitor, Fields::annotations, m_annotations);
414 cont = cont && visitor(
PathEls::Field(Fields::propertyInfos), [
this, &self]() {
415 return self.subMapItem(Map(
416 pathFromOwner().withField(Fields::propertyInfos),
417 [&self](
const DomItem &map,
const QString &k) {
418 auto pInfo = self.propertyInfoWithName(k);
419 return map.wrap(PathEls::Key(k), pInfo);
421 [&self](
const DomItem &) {
return self.propertyInfoNames(); },
422 QLatin1String(
"PropertyInfo")));
424 if (m_nameIdentifiers) {
425 cont = cont && visitor(
PathEls::Field(Fields::nameIdentifiers), [
this, &self]() {
426 return self.subScriptElementWrapperItem(m_nameIdentifiers);
434 static QList<QString> myFields(
435 { Fields::comments.toString(), Fields::idStr.toString(),
436 Fields::name.toString(), Fields::prototypes.toString(),
437 Fields::nextScope.toString(), Fields::propertyDefs.toString(),
438 Fields::bindings.toString(), Fields::methods.toString(),
439 Fields::children.toString(), Fields::annotations.toString(),
440 Fields::propertyInfos.toString() });
446 bool cont = iterateBaseDirectSubpaths(self, visitor);
447 cont = cont && self.invokeVisitorOnLazyField(visitor, Fields::defaultPropertyName,
448 [
this, &self]() {
return defaultPropertyName(self); });
454 if (name == Fields::name)
455 return self.subDataItem(
PathEls::Field(Fields::name),
this->name());
456 if (name == Fields::idStr) {
457 if (idStr().isEmpty())
459 return self.subDataItem(
PathEls::Field(Fields::idStr), idStr());
461 if (name == Fields::methods)
462 return self.wrap(
PathEls::Field(Fields::methods), m_methods);
463 if (name == Fields::bindings)
464 return self.wrap(
PathEls::Field(Fields::bindings), m_bindings);
465 if (name == Fields::comments)
466 return CommentableDomElement::field(self, name);
467 if (name == Fields::children)
468 return self.wrap(
PathEls::Field(Fields::children), m_children);
470 if (name == Fields::nextScope) {
476 if (name == Fields::prototypes) {
477 if (prototypePaths().isEmpty())
479 return self.subReferencesItem(
PathEls::Field(Fields::prototypes), m_prototypePaths);
481 if (name == Fields::annotations)
482 return self.wrap(
PathEls::Field(Fields::annotations), m_annotations);
483 if (name == Fields::propertyDefs)
484 return self.wrap(
PathEls::Field(Fields::propertyDefs), m_propertyDefs);
485 if (name == Fields::propertyInfos) {
489 [copiedSelf = self](
const DomItem &map,
const QString &k) {
490 return map.wrap(PathEls::Key(k), copiedSelf.propertyInfoWithName(k));
492 [copiedSelf = self](
const DomItem &) {
return copiedSelf.propertyInfoNames(); },
493 QLatin1String(
"PropertyInfo"))
);
495 if (name == Fields::nameIdentifiers && m_nameIdentifiers) {
498 if (name == Fields::defaultPropertyName) {
499 return self.subDataItem(
PathEls::Field(Fields::defaultPropertyName),
500 defaultPropertyName(self));
502 static QStringList knownLookups({ Fields::fileLocationsTree.toString() });
503 if (!knownLookups.contains(name)) {
504 qCWarning(domLog()) <<
"Asked non existing field " << name <<
" in QmlObject "
513 updatePathFromOwnerMultiMap(m_propertyDefs, newPath.withField(Fields::propertyDefs));
514 updatePathFromOwnerMultiMap(m_bindings, newPath.withField(Fields::bindings));
515 updatePathFromOwnerMultiMap(m_methods, newPath.withField(Fields::methods));
516 updatePathFromOwnerQList(m_children, newPath.withField(Fields::children));
517 updatePathFromOwnerQList(m_annotations, newPath.withField(Fields::annotations));
522 if (!m_defaultPropertyName.isEmpty())
523 return m_defaultPropertyName;
524 for (
const PropertyDefinition &pDef : m_propertyDefs)
525 if (pDef.isDefaultMember)
532 QString dProp = localDefaultPropertyName();
533 if (!dProp.isEmpty())
535 QString res = QStringLiteral(u"data");
536 self.visitPrototypeChain(
539 QString dProp = objPtr->localDefaultPropertyName();
540 if (!dProp.isEmpty()) {
553 bool cont = self.field(Fields::bindings).visitKeys([visitor](
const QString &,
const DomItem &bs) {
554 return bs.visitIndexes([visitor](
const DomItem &b) {
555 DomItem v = b.field(Fields::value);
556 if (std::shared_ptr<ScriptExpression> vPtr = v.ownerAs<ScriptExpression>()) {
559 return v.iterateSubOwners(visitor);
564 cont = cont && self.field(Fields::children).visitIndexes([visitor](
const DomItem &qmlObj) {
566 return qmlObjPtr->iterateSubOwners(qmlObj, visitor);
577 AST::Node *node = (expr ? expr->ast() :
nullptr);
579 switch (node->kind) {
580 case AST::Node::Kind_IdentifierExpression: {
581 AST::IdentifierExpression *id = AST::cast<AST::IdentifierExpression *>(node);
582 res.prepend(id->name.toString());
585 case AST::Node::Kind_FieldMemberExpression: {
586 AST::FieldMemberExpression *id = AST::cast<AST::FieldMemberExpression *>(node);
587 res.prepend(id->name.toString());
592 qCDebug(writeOutLog).noquote() <<
"Could not convert dot expression to list for:\n"
593 << expr->astRelocatableDump();
594 return QStringList();
603 QStringList accessSequenceList = dotExpressionToList(accessSequence);
610 QSet<QString> visitedAlias;
611 if (accessSequence.isEmpty()) {
613 }
else if (accessSequence.size() > 3) {
617 QString idName = accessSequence.first();
618 DomItem idTarget = self.component()
622 .field(Fields::referredObject)
626 res.baseObject = idTarget;
627 res.accessedPath = accessSequence.mid(1);
628 res.typeName = idTarget.name();
631 while (!res.accessedPath.isEmpty()) {
632 QString pNow = res.accessedPath.first();
633 DomItem defNow = res.baseObject.propertyDefs().key(pNow).index(0);
636 res.typeName = QString();
639 if (visitedAlias.contains(aliasPath)) {
643 visitedAlias.insert(aliasPath);
644 DomItem valNow = res.baseObject.bindings().key(pNow).index(0);
645 if (std::shared_ptr<ScriptExpression> exp =
646 valNow.field(Fields::value).ownerAs<ScriptExpression>()) {
647 QStringList expList = dotExpressionToList(exp);
648 if (expList.isEmpty()) {
651 }
else if (expList.size() > 3) {
655 idName = expList.first();
656 idTarget = self.component()
660 .field(Fields::referredObject)
662 res.baseObject = idTarget;
663 res.accessedPath = expList.mid(1) + res.accessedPath.mid(1);
669 res.typeName = idTarget.name();
675 res.localPropertyDef = defNow;
676 res.typeName = defNowPtr->typeName;
677 res.accessedPath = res.accessedPath.mid(1);
678 DomItem valNow = res.baseObject.bindings().key(pNow).index(0).field(Fields::value);
680 res.baseObject = valNow;
681 res.typeName = valNow.name();
701 tr(
"Repeated PropertyDefinition with name %1").arg(propertyDef.name)
));
708 if (p && p
.last().headIndex(0) > 1)
709 self.owningItemPtr()->addErrorLocal(
719 self.owningItemPtr()->addErrorLocal(
726 if (!idStr().isEmpty()) {
727 DomItem myId = self.component().field(Fields::ids).key(idStr()).index(0);
731 .writeRegion(fLoc, IdTokenRegion)
732 .writeRegion(fLoc, IdColonTokenRegion)
734 .writeRegion(fLoc, IdNameRegion, idStr());
735 if (ow.lineWriter.options().attributesSequence
736 == LineWriterOptions::AttributesSequence::Normalize) {
747 const DomItem &component)
const
751 return l->info().fullRegion;
752 return SourceLocation(posOfNewElements, 0, 0, 0);
755 QList<std::pair<SourceLocation, DomItem>> attribs;
758 if (component && objLocPtr)
759 componentLoc = objLocPtr->parent()->parent();
763 const auto values = base.values();
764 for (
const auto &els : values) {
765 FileLocations::Tree elsLoc = FileLocations::find(baseLoc, els.pathFromOwner().last());
766 const auto elsValues = els.values();
767 for (
const auto &el : elsValues) {
768 FileLocations::Tree elLoc = FileLocations::find(elsLoc, el.pathFromOwner().last());
769 attribs.append(std::make_pair(startLoc(elLoc), el));
773 auto addMyMMap = [
this, &objLocPtr, &self, &addMMap](QStringView fieldName) {
774 DomItem base =
this->field(self, fieldName);
777 auto addSingleLevel = [&attribs, &startLoc](
const DomItem &base,
781 const auto baseValues = base.values();
782 for (
const auto &el : baseValues) {
783 FileLocations::Tree elLoc = FileLocations::find(baseLoc, el.pathFromOwner().last());
784 attribs.append(std::make_pair(startLoc(elLoc), el));
788 DomItem enums = component.field(Fields::enumerations);
791 addMyMMap(Fields::propertyDefs);
792 addMyMMap(Fields::bindings);
793 addMyMMap(Fields::methods);
794 DomItem children = field(self, Fields::children);
797 DomItem subCs = component.field(Fields::subComponents);
798 for (
const DomItem &c : subCs.values()) {
799 const auto subLocPtr = FileLocations::treeOf(c);
801 attribs.append(std::make_pair(startLoc(subLocPtr), c));
804 std::stable_sort(attribs.begin(), attribs.end(),
805 [](
const std::pair<SourceLocation, DomItem> &el1,
806 const std::pair<SourceLocation, DomItem> &el2) {
807 if (el1.first.offset < el2.first.offset)
809 if (el1.first.offset > el2.first.offset)
811 int i =
int(el1.second.internalKind()) -
int(el2.second.internalKind());
818 const QString &code)
const
820 const QList<std::pair<SourceLocation, DomItem>> attribs = orderOfAttributes(self, component);
822 while (iAttr != attribs.size()) {
823 auto &el = attribs[iAttr++];
826 quint32 start = el.first.offset;
827 if (start != posOfNewElements && size_t(code.size()) >= start) {
829 QChar c = code.at(--start);
831 if (++preNewlines == 2)
833 }
else if (!c.isSpace())
837 if (preNewlines == 0)
839 ow.ensureNewline(preNewlines);
841 && el.first.offset != ~quint32(0)) {
843 auto &bPair = attribs[iAttr];
845 && bPair.first.begin() < el.first.end()
846 && bPair.second.name() == el.second.name()) {
851 el.second.writeOut(ow);
854 ow.writeRegion(fLoc, ColonTokenRegion);
859 qWarning() <<
"Internal error casting binding to Binding in"
861 ow.writeRegion(fLoc, LeftBraceRegion).writeRegion(fLoc, RightBraceRegion);
866 el.second.writeOut(ow);
873 const QStringList &keys)
875 QStringList originalOrderedKeys;
876 for (
const auto &attrib : attribs) {
877 QString defName = attrib.second.name();
878 if (keys.contains(defName) && !originalOrderedKeys.contains(defName))
879 originalOrderedKeys.append(std::move(defName));
881 return originalOrderedKeys;
885 const QStringView &field,
const DomItem &refItem)
887 DomItem item = refItem.field(field);
888 QStringList keys = keepKeysOrder(attribs, item.sortedKeys());
889 QList<DomItem> values = item.values();
891 QList<DomItem> originalValuesOrder;
892 for (
const auto &attrib : attribs) {
893 DomItem itemValue = item.key(attrib.second.name());
894 if (values.contains(itemValue) && !originalValuesOrder.contains(itemValue))
895 originalValuesOrder.append(std::move(itemValue));
897 return originalValuesOrder;
902 for (
const auto &enumDescs : descs) {
903 const auto values = enumDescs.values();
904 for (
const auto &enumDesc : values) {
906 enumDesc.writeOut(ow);
913 QSet<QString> &mergedDefBinding,
914 const QStringList &keys)
const
916 DomItem propertyDefs = field(self, Fields::propertyDefs);
917 DomItem bindings = field(self, Fields::bindings);
919 for (
const QString &defName : keys) {
920 const auto pDefs = propertyDefs.key(defName).values();
921 for (
const auto &pDef : pDefs) {
922 const PropertyDefinition *pDefPtr = pDef.as<PropertyDefinition>();
925 bool uniqueDeclarationWithThisName = pDefs.size() == 1;
926 if (uniqueDeclarationWithThisName && !pDefPtr->isRequired)
927 bindings.key(pDef.name()).visitIndexes([&b, pDefPtr](
const DomItem &el) {
928 const Binding *elPtr = el.as<Binding>();
929 if (elPtr && elPtr->bindingType() == BindingType::Normal) {
930 switch (elPtr->valueKind()) {
931 case BindingValueKind::ScriptExpression:
934 case BindingValueKind::Array:
935 if (!pDefPtr->isDefaultMember && pDefPtr->isParametricType())
938 case BindingValueKind::Object:
939 if (!pDefPtr->isDefaultMember && !pDefPtr->isParametricType())
942 case BindingValueKind::Empty:
950 mergedDefBinding.insert(defName);
955 const auto fLoc = FileLocations::treeOf(b);
956 ow.writeRegion(fLoc, ColonTokenRegion);
958 if (
const Binding *bPtr = b.as<Binding>())
959 bPtr->writeOutValue(b, ow);
961 qWarning() <<
"Internal error casting binding to Binding in"
962 << b.canonicalPath();
963 ow.writeRegion(fLoc, LeftBraceRegion).writeRegion(fLoc, RightBraceRegion);
1018 const DomItem &component)
const
1020 const QList<std::pair<SourceLocation, DomItem>> attribs = orderOfAttributes(self, component);
1023 quint32 counter = ow.counter();
1026 writeOutSortedEnumerations(
1027 ow.lineWriter.options().groupAttributesTogether
1028 ? keepDomItemsOrder(attribs, Fields::enumerations, component)
1029 : component.field(Fields::enumerations).values(),
1032 if (counter != ow.counter() || !idStr().isEmpty())
1033 spacerId = ow.addNewlinesAutospacerCallback(2);
1035 QSet<QString> mergedDefBinding;
1036 QStringList propertyDefsKeys = field(self, Fields::propertyDefs).sortedKeys();
1037 writeOutSortedPropertyDefinition(self, ow, mergedDefBinding,
1038 ow.lineWriter.options().groupAttributesTogether
1039 ? keepKeysOrder(attribs, propertyDefsKeys)
1040 : propertyDefsKeys);
1042 ow.removeTextAddCallback(spacerId);
1043 if (counter != ow.counter())
1044 spacerId = ow.addNewlinesAutospacerCallback(2);
1046 const auto [signalList, methodList] =
1047 splitSignalsAndMethods(ow.lineWriter.options().groupAttributesTogether
1048 ? keepDomItemsOrder(attribs, Fields::methods, self)
1049 : field(self, Fields::methods).values());
1050 for (
const auto &sig : std::as_const(signalList)) {
1056 ow.removeTextAddCallback(spacerId);
1057 if (counter != ow.counter())
1058 spacerId = ow.addNewlinesAutospacerCallback(2);
1061 for (
const auto &method : std::as_const(methodList)) {
1062 if (!first && ow.lineWriter.options().functionsSpacing) {
1067 method.writeOut(ow);
1070 ow.removeTextAddCallback(spacerId);
1072 DomItem bindings = field(self, Fields::bindings);
1073 QStringList bindingsKeys = bindings.sortedKeys();
1074 const auto [normalBindings, signalHandlers, delayedBindings] = splitBindings(
1075 bindings, mergedDefBinding,
1076 ow.lineWriter.options().groupAttributesTogether ? keepKeysOrder(attribs, bindingsKeys)
1079 if (counter != ow.counter())
1080 spacerId = ow.addNewlinesAutospacerCallback(2);
1081 for (
const auto &b : std::as_const(normalBindings))
1083 ow.removeTextAddCallback(spacerId);
1085 if (counter != ow.counter())
1086 spacerId = ow.addNewlinesAutospacerCallback(2);
1087 for (
const auto &b : std::as_const(delayedBindings))
1089 ow.removeTextAddCallback(spacerId);
1091 if (counter != ow.counter())
1092 spacerId = ow.addNewlinesAutospacerCallback(2);
1093 for (
const auto &b : std::as_const(signalHandlers))
1095 ow.removeTextAddCallback(spacerId);
1097 if (counter != ow.counter())
1098 spacerId = ow.addNewlinesAutospacerCallback(2);
1101 const auto values = field(self, Fields::children).values();
1102 for (
const auto &c : values) {
1103 if (!first && ow.lineWriter.options().objectsSpacing) {
1104 ow.newline().newline();
1110 ow.removeTextAddCallback(spacerId);
1114 DomItem subComps = component.field(Fields::subComponents);
1115 if (counter != ow.counter())
1116 spacerId = ow.addNewlinesAutospacerCallback(2);
1117 const auto values = subComps.values();
1118 for (
const auto &subC : values) {
1122 ow.removeTextAddCallback(spacerId);
1124 if (counter != ow.counter() || !ow.lineWriter.options().singleLineEmptyObjects)
1131 &&
pathFromOwner()[0] == Path::fromField(Fields::components)
1134 ow.writeRegion(fLoc, IdentifierRegion, name());
1135 if (!onTarget.isEmpty()) {
1137 .writeRegion(fLoc, OnTokenRegion)
1139 .writeRegion(fLoc, OnTargetRegion, onTarget);
1142 ow.writeRegion(fLoc, LeftBraceRegion);
1150 if (ow.lineWriter.options().attributesSequence
1151 == LineWriterOptions::AttributesSequence::Preserve) {
1153 if (std::shared_ptr<QmlFile> qmlFilePtr = self.ownerAs<QmlFile>())
1154 code = qmlFilePtr->code();
1155 writeOutAttributes(self, ow, component, code);
1159 ow.writeRegion(fLoc, RightBraceRegion);
1180 scriptCode, ScriptExpression::ExpressionType::BindingExpression)),
1196 : m_bindingType(o.m_bindingType),
1203 m_value = std::make_unique<BindingValue>(*o.m_value);
1212 m_bindingType = o.m_bindingType;
1213 m_annotations = o.m_annotations;
1214 m_comments = o.m_comments;
1215 m_bindingIdentifiers = o.m_bindingIdentifiers;
1218 m_value = std::make_unique<BindingValue>(*o.m_value);
1220 *m_value = *o.m_value;
1230 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::name), m_name);
1231 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isSignalHandler),
1234 cont = cont && visitor(
PathEls::Field(Fields::value), []() {
return DomItem(); });
1236 cont = cont && visitor(
PathEls::Field(Fields::value), [
this, &self]() {
1237 return m_value->value(self);
1239 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::bindingType),
1240 int(m_bindingType));
1241 cont = cont && self.invokeVisitorOnField(visitor, Fields::comments, m_comments);
1242 cont = cont && self.invokeVisitorOnLazyField(visitor, Fields::preCode, [
this]() {
1243 return this->preCode();
1245 cont = cont && self.invokeVisitorOnLazyField(visitor, Fields::postCode, [
this]() {
1246 return this->postCode();
1248 if (m_bindingIdentifiers) {
1249 cont = cont && visitor(
PathEls::Field(Fields::bindingIdentifiers), [
this, &self]() {
1250 return self.subScriptElementWrapperItem(m_bindingIdentifiers);
1253 cont = cont && self.invokeVisitorOnField(visitor, Fields::annotations, m_annotations);
1261 return m_value->value(self);
1267 return BindingValueKind::Empty;
1268 return m_value->kind;
1273 if (valueKind() == BindingValueKind::Object)
1274 return &(m_value->object);
1280 if (valueKind() == BindingValueKind::Object)
1281 return &(m_value->object);
1287 if (valueKind() == BindingValueKind::Array)
1288 return &(m_value->array);
1294 if (valueKind() == BindingValueKind::Array)
1295 return &(m_value->array);
1301 if (valueKind() == BindingValueKind::ScriptExpression)
1302 return m_value->scriptExpression;
1308 if (valueKind() == BindingValueKind::ScriptExpression)
1309 return m_value->scriptExpression;
1315 m_value = std::move(value);
1320 return appendUpdatableElementInQList(selfPathFromOwner.withField(Fields::annotations),
1321 m_annotations, annotation, aPtr);
1326 Path base = newPath.withField(Fields::annotations);
1328 m_value->updatePathFromOwner(newPath.withField(Fields::value));
1329 updatePathFromOwnerQList(m_annotations, newPath.withField(Fields::annotations));
1337 lw.writeRegion(fLoc, IdentifierRegion, name());
1338 lw.writeRegion(fLoc, ColonTokenRegion).ensureSpace();
1347 qCWarning(writeOutLog()) <<
"On Binding requires an QmlObject Value, not "
1357 switch (valueKind()) {
1358 case BindingValueKind::Empty:
1359 qCWarning(writeOutLog()) <<
"Writing of empty binding " << name();
1360 lw.writeRegion(fLoc, LeftBraceRegion);
1361 lw.writeRegion(fLoc, RightBraceRegion);
1363 case BindingValueKind::Array:
1364 if (
const List *vPtr = v.as<
List>()) {
1370 case BindingValueKind::Object:
1371 case BindingValueKind::ScriptExpression:
1379 bool cont = Component::iterateDirectSubpaths(self, visitor);
1380 cont = cont && self.invokeVisitorOnField(visitor, Fields::exports, m_exports);
1381 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::metaRevisions),
1383 if (!fileName().isEmpty()) {
1384 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::fileName),
1387 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::interfaceNames),
1389 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::hasCustomParser),
1391 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::elementTypeName),
1393 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::extensionTypeName),
1394 m_extensionTypeName);
1395 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::accessSemantics),
1396 int(m_accessSemantics));
1404 res.exportSourcePath = source;
1405 res.typePath = typePath;
1406 int slashIdx = exp.indexOf(QLatin1Char(
'/'));
1407 int spaceIdx = exp.indexOf(QLatin1Char(
' '));
1409 spaceIdx = exp.size();
1411 res.version = Version::fromString(exp.mid(spaceIdx + 1));
1412 if (!res.version.isValid())
1414 .error(tr(
"Expected string literal to contain 'Package/Name major.minor' "
1415 "or 'Name major.minor' not '%1'.")
1419 res.uri = exp.left(slashIdx).toString();
1420 res.typeName = exp.mid(slashIdx + 1, spaceIdx - (slashIdx + 1)).toString();
1427 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::name), name);
1428 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::access),
int(
access));
1429 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::typeName), typeName);
1430 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isReadonly),
isReadonly);
1431 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isList),
isList);
1432 cont = cont && self.invokeVisitorOnField(visitor, Fields::comments, comments);
1433 cont = cont && self.invokeVisitorOnField(visitor, Fields::annotations, annotations);
1440 return appendUpdatableElementInQList(selfPathFromOwner.withField(Fields::annotations), annotations,
1446 Path base = newPath.withField(Fields::annotations);
1447 updatePathFromOwnerQList(annotations, newPath.withField(Fields::annotations));
1452 bool cont = CommentableDomElement::iterateDirectSubpaths(self, visitor);
1453 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::name), name());
1454 cont = cont && self.invokeVisitorOnField(visitor, Fields::values, m_values);
1455 cont = cont && self.invokeVisitorOnField(visitor, Fields::annotations, m_annotations);
1462 updatePathFromOwnerQList(m_annotations, newPath.withField(Fields::annotations));
1467 m_annotations = annotations;
1472 return appendUpdatableElementInQList(
pathFromOwner().withField(Fields::annotations), m_annotations,
1479 ow.writeRegion(fLoc, EnumKeywordRegion)
1481 .writeRegion(fLoc, IdentifierRegion, name())
1483 .writeRegion(fLoc, LeftBraceRegion);
1484 const auto values = self.field(Fields::values).values();
1485 for (
const auto &value : values)
1487 ow.ensureNewline().writeRegion(fLoc, RightBraceRegion);
1495 RefCacheEntry cached = (env ? RefCacheEntry::forPath(env, selfPath) : RefCacheEntry());
1496 if (cached.cached == RefCacheEntry::Cached::All)
1497 return cached.canonicalPaths;
1499 QSet<Path> knownPaths;
1500 QList<Path> toDo(m_importSourcePaths.rbegin(), m_importSourcePaths.rend());
1501 while (!toDo.isEmpty()) {
1502 Path pNow = toDo.takeLast();
1503 if (knownPaths.contains(pNow))
1505 knownPaths.insert(pNow);
1508 for (
const DomItem &autoExp : sourceBase.field(Fields::autoExports).values()) {
1509 if (
const ModuleAutoExport *autoExpPtr = autoExp.as<ModuleAutoExport>()) {
1511 if (autoExpPtr->inheritVersion) {
1512 Version v = autoExpPtr->import.version;
1513 DomItem sourceVersion = sourceBase.field(Fields::version);
1514 if (
const Version *sourceVersionPtr = sourceVersion.as<Version>()) {
1515 if (v.majorVersion < 0)
1516 v.majorVersion = sourceVersionPtr->majorVersion;
1517 if (v.minorVersion < 0)
1518 v.minorVersion = sourceVersionPtr->minorVersion;
1520 qWarning() <<
"autoExport with inherited version " << autoExp
1521 <<
" but missing version in source" << pNow;
1523 Import toImport(autoExpPtr->import.uri, v);
1524 newSource = toImport.importedPath();
1526 newSource = autoExpPtr->import.importedPath();
1528 if (newSource && !knownPaths.contains(newSource))
1529 toDo.append(newSource);
1531 qWarning() <<
"expected ModuleAutoExport not " << autoExp.internalKindStr()
1532 <<
"looking up autoExports of" << sourceBase;
1545 cont = cont && self.invokeVisitorOnReferences(visitor, Fields::importSources, m_importSourcePaths);
1546 cont = cont && visitor(
PathEls::Field(Fields::allSources), [
this, &self]() ->
DomItem {
1548 self.pathFromOwner().withField(Fields::allSources), allSources(self),
1549 [](
const DomItem &list,
const PathEls::PathComponent &p,
1550 const Path &el) {
return list.subDataItem(p, el.toString()); }));
1552 cont = cont && self.invokeVisitorOnField(visitor, Fields::qualifiedImports, m_subImports);
1553 cont = cont && visitor(
PathEls::Field(Fields::imported), [
this, &self]() ->
DomItem {
1556 [
this, &self](
const DomItem &map,
const QString &key) {
1557 return map.subListItem(List::fromQList<DomItem>(
1558 map.pathFromOwner().withKey(key),
1559 importedItemsWithName(self, key),
1560 [](
const DomItem &,
const PathEls::PathComponent &,
1561 const DomItem &el) {
return el; }));
1563 [
this, &self](
const DomItem &) {
return this->importedNames(self); },
1564 QLatin1String(
"List<Export>"))
);
1572 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::propertyDefs),
1574 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::bindings), bindings);
1582 new (&object) QmlObject(o);
1588 new (&scriptExpression) std::shared_ptr<ScriptExpression>(o);
1593 new (&array) QList<QmlObject>(l);
1604 case BindingValueKind::Empty:
1606 case BindingValueKind::Object:
1607 new (&object) QmlObject(o.object);
1609 case BindingValueKind::ScriptExpression:
1610 new (&scriptExpression) std::shared_ptr<ScriptExpression>(o.scriptExpression);
1612 case BindingValueKind::Array:
1613 new (&array) QList<QmlObject>(o.array);
1622 case BindingValueKind::Empty:
1624 case BindingValueKind::Object:
1625 new (&object) QmlObject(o.object);
1627 case BindingValueKind::ScriptExpression:
1628 new (&scriptExpression) std::shared_ptr<ScriptExpression>(o.scriptExpression);
1630 case BindingValueKind::Array:
1631 new (&array) QList<QmlObject>(o.array);
1639 case BindingValueKind::Empty:
1641 case BindingValueKind::Object:
1642 return binding.copy(&object);
1643 case BindingValueKind::ScriptExpression:
1644 return binding.subOwnerItem(
PathEls::Field(Fields::value), scriptExpression);
1645 case BindingValueKind::Array:
1646 return binding
.subListItem(List::fromQListRef<QmlObject>(
1647 binding.pathFromOwner().withField(u"value"), array,
1648 [](
const DomItem &self,
const PathEls::PathComponent &,
const QmlObject &obj) {
1649 return self.copy(&obj);
1658 case BindingValueKind::Empty:
1660 case BindingValueKind::Object:
1661 object.updatePathFromOwner(newPath);
1663 case BindingValueKind::ScriptExpression:
1665 case BindingValueKind::Array:
1666 updatePathFromOwnerQList(array, newPath);
1674 case BindingValueKind::Empty:
1676 case BindingValueKind::Object:
1677 object.~QmlObject();
1679 case BindingValueKind::ScriptExpression:
1680 scriptExpression.~shared_ptr();
1682 case BindingValueKind::Array:
1683 array.~QList<QmlObject>();
1686 kind = BindingValueKind::Empty;
1691 ExpressionType expressionType,
const SourceLocation &localOffset)
1700 if (m_expressionType == ExpressionType::BindingExpression)
1701 if (AST::ExpressionStatement *exp = AST::cast<AST::ExpressionStatement *>(m_ast))
1702 m_ast = exp->expression;
1703 Q_ASSERT(m_astComments);
1708 QMutexLocker l(mutex());
1709 m_expressionType = e.m_expressionType;
1710 m_engine = e.m_engine;
1712 if (m_codeStr.isEmpty()) {
1715 m_codeStr = e.m_codeStr;
1718 m_localOffset = e.m_localOffset;
1719 m_astComments = e.m_astComments;
1724 bool cont = OwningItem::iterateDirectSubpaths(self, visitor);
1725 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::code), code());
1726 cont = cont && self.invokeVisitorOnLazyField(
1727 visitor, Fields::localOffset,
1728 [
this]() {
return sourceLocationToQCborValue(localOffset()); },
1730 cont = cont && self.invokeVisitorOnLazyField(visitor, Fields::astRelocatableDump, [
this]() {
1731 return astRelocatableDump();
1733 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::expressionType),
1734 int(expressionType()));
1736 cont = cont && visitor(
PathEls::Field(Fields::scriptElement), [
this, &self]() {
1737 return self.subScriptElementWrapperItem(m_element);
1754 if (!VisitAll::uiKinds().contains(n->kind)) {
1755 qsizetype start = n->firstSourceLocation().begin();
1756 qsizetype end = n->lastSourceLocation().end();
1757 if (!firstNodeInRange && minStart <= start && end <= maxEnd && start < end)
1758 firstNodeInRange = n;
1760 return !firstNodeInRange;
1767 AST::Node::accept(n, &visitor);
1768 return visitor.firstNodeInRange;
1774 if (code.isEmpty()) {
1778 m_code = QStringView(m_codeStr);
1780 m_localOffset = SourceLocation();
1781 m_localOffset.length = m_code.size();
1783 m_engine = std::make_shared<QQmlJS::Engine>();
1784 m_astComments = std::make_shared<AstComments>(m_engine);
1787 if (AST::Program *programPtr = AST::cast<AST::Program *>(m_ast)) {
1788 m_ast = programPtr->statements;
1791 if (
auto *sList = AST::cast<AST::FormalParameterList *>(m_ast)) {
1792 m_ast = sList->element;
1794 if (m_expressionType != ExpressionType::FunctionBody) {
1795 if (AST::StatementList *sList = AST::cast<AST::StatementList *>(m_ast)) {
1797 m_ast = sList->statement;
1800 if (m_expressionType == ExpressionType::BindingExpression)
1801 if (AST::ExpressionStatement *exp = AST::cast<AST::ExpressionStatement *>(m_ast))
1802 m_ast = exp->expression;
1810 QQmlJS::Lexer lexer(m_engine.get());
1812 const bool qmlMode = m_expressionType != ExpressionType::ESMCode
1813 && m_expressionType != ExpressionType::JSCode;
1814 lexer.setCode(m_codeStr, 1, qmlMode);
1815 QQmlJS::Parser parser(m_engine.get());
1816 const bool parserSucceeded = m_expressionType == ExpressionType::ESMCode ? parser.parseModule()
1817 : parser.parseScript();
1818 if (!parserSucceeded) {
1821 const auto messages = parser.diagnosticMessages();
1822 for (
const DiagnosticMessage &msg : messages) {
1823 ErrorMessage err = domParsingErrors().errorMessage(msg);
1824 err.location.offset -= m_localOffset.offset;
1825 err.location.startLine -= m_localOffset.startLine;
1826 if (err.location.startLine == 1)
1827 err.location.startColumn -= m_localOffset.startColumn;
1828 addErrorLocal(std::move(err));
1830 return parser.rootNode();
1835 astNodeDumper(s, ast(), options, 1, 0, [
this](SourceLocation astL) {
1836 SourceLocation l =
this->locationToLocal(astL);
1837 return this->code().mid(l.offset, l.length);
1843 return dumperToString([
this](
const Sink &s) {
1844 this->astDumper(s, AstDumperOption::NoLocations | AstDumperOption::SloppyCompare);
1850 reformatAst(lw,
this);
1855 SourceLocation l =
this->locationToLocal(astL);
1856 return this->code().mid(l.offset, l.length);
1861 if (
const FileLocations::Tree tree = FileLocations::treeOf(self)) {
1862 return FileLocations::region(tree, MainRegion);
1864 return SourceLocation();
1869 return typeName.contains(QChar(u'<'));
1877 lw.writeRegion(fLoc, DefaultKeywordRegion).ensureSpace();
1879 lw.writeRegion(fLoc, VirtualKeywordRegion).ensureSpace();
1881 lw.writeRegion(fLoc, OverrideKeywordRegion).ensureSpace();
1883 lw.writeRegion(fLoc, FinalKeywordRegion).ensureSpace();
1885 lw.writeRegion(fLoc, RequiredKeywordRegion).ensureSpace();
1887 lw.writeRegion(fLoc, ReadonlyKeywordRegion).ensureSpace();
1888 if (!typeName.isEmpty()) {
1889 lw.writeRegion(fLoc, PropertyKeywordRegion).ensureSpace();
1890 lw.writeRegion(fLoc, TypeIdentifierRegion, typeName).ensureSpace();
1892 lw.writeRegion(fLoc, IdentifierRegion, name);
1897 bool cont = AttributeInfo::iterateDirectSubpaths(self, visitor);
1898 cont = cont && self.invokeVisitorOnField(visitor, Fields::parameters, parameters);
1899 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::methodType),
1901 if (!typeName.isEmpty())
1902 cont = cont && self.invokeVisitorOnReference(visitor, Fields::type,
typePath(self
));
1903 if (methodType == MethodType::Method) {
1904 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isConstructor),
1908 cont = cont && visitor(
PathEls::Field(Fields::returnType), [
this, &self]() {
1909 return self.subOwnerItem(PathEls::Field(Fields::returnType), returnType);
1912 cont = cont && visitor(
PathEls::Field(Fields::body), [
this, &self]() {
1913 return self.subOwnerItem(PathEls::Field(Fields::body), body);
1920 if (parameters.isEmpty() && methodType == MethodType::Signal)
1924 ow.writeRegion(fLoc, LeftParenthesisRegion);
1926 for (
const DomItem &arg : self.field(Fields::parameters).values()) {
1930 const auto fLocArg = FileLocations::treeOf(arg);
1931 ow.writeRegion(fLocArg, CommaTokenRegion).ensureSpace();
1935 ow.writeRegion(fLoc, RightParenthesisRegion);
1940 if (typeName.isEmpty())
1944 ow.writeRegion(fLoc, ColonTokenRegion);
1946 ow.writeRegion(fLoc, TypeIdentifierRegion, typeName);
1952 ow.ensureSpace().writeRegion(fLoc, LeftBraceRegion);
1953 if (
DomItem b = self.field(Fields::body)) {
1957 ow.ensureNewline().writeRegion(fLoc, RightBraceRegion);
1963 if (methodType == MethodType::Signal) {
1964 ow.writeRegion(fLoc, SignalKeywordRegion).ensureSpace();
1966 ow.writeRegion(fLoc, FunctionKeywordRegion).ensureSpace();
1968 ow.writeRegion(fLoc, IdentifierRegion, name);
1969 writeOutArguments(self, ow);
1970 if (methodType == MethodType::Signal) {
1974 writeOutReturnType(self, ow);
1975 writeOutBody(self, ow);
1981 QTextStream res(&resultStr);
1982 LineWriter lw([&res](QStringView s) { res << s; }, QLatin1String(
"*testStream*"));
1984 ow.indentNextlines =
true;
1985 ow.skipComments =
true;
1987 writeOutArguments(self, ow);
1988 writeOutReturnType(self, ow);
1992 return resultStr.simplified();
1998 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::name), name);
1999 if (!typeName.isEmpty()) {
2000 cont = cont && self.invokeVisitorOnReference(visitor, Fields::type,
2001 Paths::lookupTypePath(typeName));
2002 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::typeName), typeName);
2004 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isPointer),
isPointer);
2005 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isReadonly),
isReadonly);
2006 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::isList),
isList);
2007 cont = cont && self.invokeVisitorOnField(visitor, Fields::defaultValue, defaultValue);
2008 cont = cont && self.invokeVisitorOnField(visitor, Fields::value, value);
2010 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::preCode),
2012 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::postCode), u") {}"_s);
2014 if (!annotations.isEmpty())
2015 cont = cont && self.invokeVisitorOnField(visitor, Fields::annotations, annotations);
2016 cont = cont && self.invokeVisitorOnField(visitor, Fields::comments, comments);
2027 if (!name.isEmpty()) {
2030 ow.writeRegion(fLoc, EllipsisTokenRegion);
2031 ow.writeRegion(fLoc, IdentifierRegion, name);
2032 if (!typeName.isEmpty())
2033 ow.writeRegion(fLoc, ColonTokenRegion)
2035 .writeRegion(fLoc, TypeIdentifierRegion, typeName);
2037 ow.ensureSpace().writeRegion(fLoc, EqualTokenRegion).ensureSpace();
2038 self.subOwnerItem(
PathEls::Field(Fields::defaultValue), defaultValue).writeOut(ow);
2042 self.subOwnerItem(
PathEls::Field(Fields::value), value).writeOut(ow);
2051 if (!typeName.isEmpty())
2052 ow.writeRegion(fLoc, TypeIdentifierRegion, typeName).ensureSpace();
2053 ow.writeRegion(fLoc, IdentifierRegion, name);
2061 ow.writeRegion(fLoc, PragmaKeywordRegion)
2063 .writeRegion(fLoc, IdentifierRegion, name);
2065 bool isFirst =
true;
2066 for (
const auto &value : values) {
2069 ow.writeRegion(fLoc, ColonTokenRegion).ensureSpace();
2070 ow.writeRegion(fLoc, PragmaValuesRegion, value);
2074 ow.writeRegion(fLoc, CommaTokenRegion).ensureSpace();
2075 ow.writeRegion(fLoc, PragmaValuesRegion, value);
2083 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::name), name());
2084 cont = cont && self.invokeVisitorOnValue(visitor,
PathEls::Field(Fields::value),
value());
2085 cont = cont && self.invokeVisitorOnField(visitor, Fields::comments, m_comments);
2094 ow.writeRegion(fLoc, CommaTokenRegion);
2096 ow.writeRegion(fLoc, IdentifierRegion, name());
2098 QString v = QString::number(value(),
'f', 0);
2099 if (abs(value() - v.toDouble()) > 1.e-10)
2100 v = QString::number(value());
2102 .writeRegion(fLoc, EqualTokenRegion)
2104 .writeRegion(fLoc, EnumValueRegion, v);
2110 if (str.startsWith(u'"'))
2111 return fromDirectoryString(str.mid(1, str.size() - 2)
2112 .replace(u"\\\""_s, u"\""_s)
2113 .replace(u"\\\\"_s, u"\\"_s));
2115 return fromUriString(str);
2120 QRegularExpression moduleUriRe(QLatin1String(R"(\A\w+(?:\.\w+)*\Z)"));
2127 if (url.isValid() && url.scheme().size() > 1)
2129 if (!str.isEmpty()) {
2130 QFileInfo path(str);
2162 if (m_kind == Kind::ModuleUri)
2163 return std::get<QString>(m_value);
2174 const QUrl &url = std::get<QUrl>(m_value);
2175 if (url.scheme().compare(u"file", Qt::CaseInsensitive) == 0)
2179 case Kind::RelativePath:
2180 case Kind::AbsolutePath:
2181 return std::get<QString>(m_value);
2193 const QUrl &url = std::get<QUrl>(m_value);
2194 if (url.scheme().compare(u"file", Qt::CaseInsensitive) == 0)
2199 if (!basePath.isEmpty())
2200 return QDir(basePath).filePath(std::get<QString>(m_value));
2203 case Kind::AbsolutePath:
2204 return std::get<QString>(m_value);
2211 if (m_kind == Kind::DirectoryUrl)
2212 return std::get<QUrl>(m_value);
2222 case Kind::DirectoryUrl:
2223 return std::get<QUrl>(m_value).toString();
2224 case Kind::RelativePath:
2225 case Kind::AbsolutePath:
2226 return std::get<QString>(m_value);
2236 case Kind::ModuleUri:
2237 return std::get<QString>(m_value);
2238 case Kind::DirectoryUrl:
2239 case Kind::RelativePath:
2240 case Kind::AbsolutePath:
2241 return u"\""_s + directoryString().replace(u'\\', u"\\\\"_s).replace(u'"', u"\\\""_s)
2262#include "moc_qqmldomelements_p.cpp"
void updatePathFromOwner(const Path &newPath)
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor visitor) const
Path addAnnotation(const Path &selfPathFromOwner, const QmlObject &annotation, QmlObject **aPtr=nullptr)
BindingValue(const QList< QmlObject > &l)
void updatePathFromOwner(const Path &newPath)
BindingValue(const std::shared_ptr< ScriptExpression > &o)
BindingValue(const BindingValue &o)
DomItem value(const DomItem &binding) const
BindingValue & operator=(const BindingValue &o)
BindingValue(const QmlObject &o)
void setValue(std::unique_ptr< BindingValue > &&value)
std::shared_ptr< ScriptExpression > scriptExpressionValue() const
Binding & operator=(const Binding &)
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor) const
Path addAnnotation(const Path &selfPathFromOwner, const QmlObject &a, QmlObject **aPtr=nullptr)
void updatePathFromOwner(const Path &newPath)
QmlObject const * objectValue() const
DomItem valueItem(const DomItem &self) const
QList< QmlObject > * arrayValue()
Binding(const Binding &o)
QList< QmlObject > const * arrayValue() const
void writeOut(const DomItem &self, OutWriter &lw) const
Binding(const QString &m_name, const QString &scriptCode, BindingType bindingType=BindingType::Normal)
Binding(const QString &m_name, std::unique_ptr< BindingValue > &&value, BindingType bindingType=BindingType::Normal)
bool isSignalHandler() const
void writeOutValue(const DomItem &self, OutWriter &lw) const
Binding(const QString &m_name=QString())
Binding(const QString &m_name, const QmlObject &value, BindingType bindingType=BindingType::Normal)
QmlObject * objectValue()
std::shared_ptr< ScriptExpression > scriptExpressionValue()
BindingValueKind valueKind() const
Binding(const QString &m_name, const std::shared_ptr< ScriptExpression > &value, BindingType bindingType=BindingType::Normal)
void updatePathFromOwner(const Path &newPath) override
Component(const Path &pathFromOwner=Path())
Path addObject(const QmlObject &object, QmlObject **oPtr=nullptr)
Path attachedTypePath(const DomItem &) const
DomItem field(const DomItem &self, QStringView name) const override
bool iterateDirectSubpaths(const DomItem &, DirectVisitor) const override
Component(const QString &name)
virtual void updatePathFromOwner(const Path &newPath)
Path pathFromOwner() const override
A value type that references any element of the Dom.
void writeOutPost(OutWriter &lw) const
DomItem path(const Path &p, const ErrorHandler &h=&defaultErrorHandler) const
DomItem subListItem(const List &list) const
void writeOutPre(OutWriter &lw) const
DomItem environment() const
Path canonicalPath() const
DomItem containingObject() const
DomItem subMapItem(const Map &map) const
DomItem subReferenceItem(const PathEls::PathComponent &c, const Path &referencedObject) const
InternalKind internalKind() const
DomItem owner() const
The owner of an element, for an qmlObject this is the containing qml file.
void writeOut(OutWriter &lw) const
Path pathFromOwner() const
DomItem subScriptElementWrapperItem(const ScriptElementVariant &obj) const
static ErrorGroup domErrorGroup
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor visitor) const override
void setAnnotations(const QList< QmlObject > &annotations)
void writeOut(const DomItem &self, OutWriter &lw) const override
void updatePathFromOwner(const Path &newP) override
Path addAnnotation(const QmlObject &child, QmlObject **cPtr=nullptr)
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor visitor) const
void writeOut(const DomItem &self, OutWriter &lw) const
Represents a set of tags grouping a set of related error messages.
ErrorMessage warning(const Dumper &message) const
ErrorMessage error(const Dumper &message) const
AST::Node * firstNodeInRange
bool preVisit(AST::Node *n) override
FirstNodeVisitor(qsizetype minStart, qsizetype maxEnd)
Path addAnnotation(const Path &selfPathFromOwner, const QmlObject &ann, QmlObject **aPtr=nullptr)
Id(const QString &idName=QString(), const Path &referredObject=Path())
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor) const
void updatePathFromOwner(const Path &pathFromOwner)
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor visitor) const
QList< Path > allSources(const DomItem &self) const
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor) const
void writeOut(const DomItem &self, OutWriter &ow) const
void eof(bool ensureNewline=true)
void writeOut(const DomItem &self, OutWriter &ow, bool compact) const
std::function< index_type(const DomItem &)> Length
std::function< QSet< QString >(const DomItem &)> Keys
Path typePath(const DomItem &) const
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor visitor) const
void writeOut(const DomItem &self, OutWriter &ow) const
QString signature(const DomItem &self) const
void writeOut(const DomItem &self, OutWriter &ow) const
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor visitor) const
void writeOutSignal(const DomItem &self, OutWriter &ow) const
TypeAnnotationStyle typeAnnotationStyle
MutableDomItem path(const Path &p)
A DomItem that owns other DomItems and is managed through a shared pointer.
void addErrorLocal(ErrorMessage &&msg)
OwningItem(int derivedFrom=0)
static Path fromRoot(PathRoot r)
static ErrorGroups myErrors()
Path operator[](int i) const
void writeOut(const DomItem &self, OutWriter &ow) const
bool isParametricType() const
void writeOut(const DomItem &self, OutWriter &lw) const
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor visitor) const
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor) const override
QList< DomItem > subComponents(const DomItem &self) const
void updatePathFromOwner(const Path &newPath) override
void writeOut(const DomItem &self, OutWriter &) const override
QList< QString > subComponentsNames(const DomItem &self) const
MutableDomItem addBinding(MutableDomItem &self, const Binding &binding, AddOption option)
void writeOutSortedEnumerations(const QList< DomItem > &descs, OutWriter &ow) const
void writeOut(const DomItem &self, OutWriter &ow, const QString &onTarget) const
QList< std::pair< SourceLocation, DomItem > > orderOfAttributes(const DomItem &self, const DomItem &component) const
bool iterateSubOwners(const DomItem &self, function_ref< bool(const DomItem &owner)> visitor) const
void updatePathFromOwner(const Path &newPath) override
Path addBinding(const Binding &binding, AddOption option, Binding **bPtr=nullptr)
static constexpr DomType kindValue
LocallyResolvedAlias resolveAlias(const DomItem &self, std::shared_ptr< ScriptExpression > accessSequence) const
void writeOutId(const DomItem &self, OutWriter &ow) const
LocallyResolvedAlias resolveAlias(const DomItem &self, const QStringList &accessSequence) const
void writeOutSortedPropertyDefinition(const DomItem &self, OutWriter &ow, QSet< QString > &mergedDefBinding, const QStringList &keys) const
Path addPropertyDef(const PropertyDefinition &propertyDef, AddOption option, PropertyDefinition **pDef=nullptr)
MutableDomItem addPropertyDef(MutableDomItem &self, const PropertyDefinition &propertyDef, AddOption option)
DomItem field(const DomItem &self, QStringView name) const override
QString localDefaultPropertyName() const
QString defaultPropertyName(const DomItem &self) const
QmlObject(const Path &pathFromOwner=Path())
void writeOutSortedAttributes(const DomItem &self, OutWriter &ow, const DomItem &component) const
MutableDomItem addMethod(MutableDomItem &self, const MethodInfo &functionDef, AddOption option)
void writeOutAttributes(const DomItem &self, OutWriter &ow, const DomItem &component, const QString &code) const
Path nextScopePath() const
void writeOut(const DomItem &self, OutWriter &lw) const override
QList< QString > fields() const
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor) const override
Path addMethod(const MethodInfo &functionDef, AddOption option, MethodInfo **mPtr=nullptr)
bool iterateBaseDirectSubpaths(const DomItem &self, DirectVisitor) const
QString absoluteLocalPath(const QString &basePath=QString()) const
QUrl directoryUrl() const
QString directoryString() const
QString localPath() const
QString moduleUri() const
bool iterateDirectSubpaths(const DomItem &, DirectVisitor) const override
static bool addForPath(const DomItem &el, const Path &canonicalPath, const RefCacheEntry &entry, AddOption addOption=AddOption::KeepExisting)
Use this to contain any script element.
QStringView loc2Str(const SourceLocation &) const
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor visitor) const override
ScriptExpression(const ScriptExpression &e)
void writeOut(const DomItem &self, OutWriter &lw) const override
SourceLocation globalLocation(const DomItem &self) const
ScriptExpression(QStringView code, const std::shared_ptr< QQmlJS::Engine > &engine, AST::Node *ast, const std::shared_ptr< AstComments > &comments, ExpressionType expressionType, const SourceLocation &localOffset=SourceLocation())
QString astRelocatableDump() const
void setScriptElement(const ScriptElementVariant &p)
void astDumper(const Sink &s, AstDumperOptions options) const
bool iterateDirectSubpaths(const DomItem &self, DirectVisitor) const
Version(qint32 majorVersion=Undefined, qint32 minorVersion=Undefined)
QString stringValue() const
A vistor that visits all the AST:Node.
Provides entities to maintain mappings between elements and their location in a file.
Tree treeOf(const DomItem &)
Tree find(const Tree &self, const Path &p)
std::shared_ptr< Node > Tree
Path moduleIndexPath(const QString &uri, int majorVersion, const ErrorHandler &errorHandler=nullptr)
Path moduleScopePath(const QString &uri, const QString &version, const ErrorHandler &errorHandler=nullptr)
Path moduleScopePath(const QString &uri, Version version, const ErrorHandler &errorHandler=nullptr)
static ErrorGroups domParsingErrors()
static QStringList dotExpressionToList(const std::shared_ptr< ScriptExpression > &expr)
static QStringList keepKeysOrder(const QList< std::pair< SourceLocation, DomItem > > &attribs, const QStringList &keys)
static QList< DomItem > keepDomItemsOrder(const QList< std::pair< SourceLocation, DomItem > > &attribs, const QStringView &field, const DomItem &refItem)
std::function< void(const ErrorMessage &)> ErrorHandler
AST::Node * firstNodeInRange(AST::Node *n, qsizetype minStart=0, qsizetype maxEnd=std::numeric_limits< qint32 >::max())
Combined button and popup list for selecting options.
#define NewErrorGroup(name)