// Copyright (C) 2021 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include "qdocindexfiles.h" #include "access.h" #include "atom.h" #include "classnode.h" #include "collectionnode.h" #include "config.h" #include "enumnode.h" #include "examplenode.h" #include "externalpagenode.h" #include "functionnode.h" #include "generator.h" #include "headernode.h" #include "location.h" #include "utilities.h" #include "propertynode.h" #include "qdocdatabase.h" #include "qmlpropertynode.h" #include "typedefnode.h" #include "variablenode.h" #include #include QT_BEGIN_NAMESPACE enum QDocAttr { QDocAttrNone, QDocAttrExample, QDocAttrFile, QDocAttrImage, QDocAttrDocument, QDocAttrExternalPage, QDocAttrAttribution }; static Node *root_ = nullptr; static IndexSectionWriter *post_ = nullptr; /*! \class QDocIndexFiles This class handles qdoc index files. */ QDocIndexFiles *QDocIndexFiles::s_qdocIndexFiles = nullptr; /*! Constructs the singleton QDocIndexFiles. */ QDocIndexFiles::QDocIndexFiles() : m_gen(nullptr) { m_qdb = QDocDatabase::qdocDB(); m_storeLocationInfo = Config::instance().get(CONFIG_LOCATIONINFO).asBool(); } /*! Destroys the singleton QDocIndexFiles. */ QDocIndexFiles::~QDocIndexFiles() { m_qdb = nullptr; m_gen = nullptr; } /*! Creates the singleton. Allows only one instance of the class to be created. Returns a pointer to the singleton. */ QDocIndexFiles *QDocIndexFiles::qdocIndexFiles() { if (s_qdocIndexFiles == nullptr) s_qdocIndexFiles = new QDocIndexFiles; return s_qdocIndexFiles; } /*! Destroys the singleton. */ void QDocIndexFiles::destroyQDocIndexFiles() { if (s_qdocIndexFiles != nullptr) { delete s_qdocIndexFiles; s_qdocIndexFiles = nullptr; } } /*! Reads and parses the list of index files in \a indexFiles. */ void QDocIndexFiles::readIndexes(const QStringList &indexFiles) { for (const QString &file : indexFiles) { qCDebug(lcQdoc) << "Loading index file: " << file; readIndexFile(file); } } /*! Reads and parses the index file at \a path. */ void QDocIndexFiles::readIndexFile(const QString &path) { QFile file(path); if (!file.open(QFile::ReadOnly)) { qWarning() << "Could not read index file" << path; return; } QXmlStreamReader reader(&file); reader.setNamespaceProcessing(false); if (!reader.readNextStartElement()) return; if (reader.name() != QLatin1String("INDEX")) return; QXmlStreamAttributes attrs = reader.attributes(); QString indexUrl {attrs.value(QLatin1String("url")).toString()}; // Decide how we link to nodes loaded from this index file: // If building a set that will be installed AND the URL of // the dependency is identical to ours, assume that also // the dependent html files are available under the same // directory tree. Otherwise, link using the full index URL. if (!Config::installDir.isEmpty() && indexUrl == Config::instance().get(CONFIG_URL).asString()) { // Generate a relative URL between the install dir and the index file // when the -installdir command line option is set. QDir installDir(path.section('/', 0, -3) + '/' + Generator::outputSubdir()); indexUrl = installDir.relativeFilePath(path).section('/', 0, -2); } m_project = attrs.value(QLatin1String("project")).toString(); QString indexTitle = attrs.value(QLatin1String("indexTitle")).toString(); m_basesList.clear(); m_relatedNodes.clear(); NamespaceNode *root = m_qdb->newIndexTree(m_project); if (!root) { qWarning() << "Issue parsing index tree" << path; return; } root->tree()->setIndexTitle(indexTitle); // Scan all elements in the XML file, constructing a map that contains // base classes for each class found. while (reader.readNextStartElement()) { readIndexSection(reader, root, indexUrl); } // Now that all the base classes have been found for this index, // arrange them into an inheritance hierarchy. resolveIndex(); } /*! Read a
element from the index file and create the appropriate node(s). */ void QDocIndexFiles::readIndexSection(QXmlStreamReader &reader, Node *current, const QString &indexUrl) { QXmlStreamAttributes attributes = reader.attributes(); QStringView elementName = reader.name(); QString name = attributes.value(QLatin1String("name")).toString(); QString href = attributes.value(QLatin1String("href")).toString(); Node *node; Location location; Aggregate *parent = nullptr; bool hasReadChildren = false; if (current->isAggregate()) parent = static_cast(current); if (attributes.hasAttribute(QLatin1String("related"))) { bool isIntTypeRelatedValue = false; int relatedIndex = attributes.value(QLatin1String("related")).toInt(&isIntTypeRelatedValue); if (isIntTypeRelatedValue) { if (adoptRelatedNode(parent, relatedIndex)) { reader.skipCurrentElement(); return; } } else { QList::iterator nodeIterator = std::find_if(m_relatedNodes.begin(), m_relatedNodes.end(), [&](const Node *relatedNode) { return (name == relatedNode->name() && href == relatedNode->url().section(QLatin1Char('/'), -1)); }); if (nodeIterator != m_relatedNodes.end() && parent) { parent->adoptChild(*nodeIterator); reader.skipCurrentElement(); return; } } } QString filePath; int lineNo = 0; if (attributes.hasAttribute(QLatin1String("filepath"))) { filePath = attributes.value(QLatin1String("filepath")).toString(); lineNo = attributes.value("lineno").toInt(); } if (elementName == QLatin1String("namespace")) { auto *namespaceNode = new NamespaceNode(parent, name); node = namespaceNode; if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + name.toLower() + ".html"); else if (!indexUrl.isNull()) location = Location(name.toLower() + ".html"); } else if (elementName == QLatin1String("class") || elementName == QLatin1String("struct") || elementName == QLatin1String("union")) { Node::NodeType type = Node::Class; if (elementName == QLatin1String("class")) type = Node::Class; else if (elementName == QLatin1String("struct")) type = Node::Struct; else if (elementName == QLatin1String("union")) type = Node::Union; node = new ClassNode(type, parent, name); if (attributes.hasAttribute(QLatin1String("bases"))) { QString bases = attributes.value(QLatin1String("bases")).toString(); if (!bases.isEmpty()) m_basesList.append( std::pair(static_cast(node), bases)); } if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + name.toLower() + ".html"); else if (!indexUrl.isNull()) location = Location(name.toLower() + ".html"); bool abstract = false; if (attributes.value(QLatin1String("abstract")) == QLatin1String("true")) abstract = true; node->setAbstract(abstract); } else if (elementName == QLatin1String("header")) { node = new HeaderNode(parent, name); if (attributes.hasAttribute(QLatin1String("location"))) name = attributes.value(QLatin1String("location")).toString(); if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + name); else if (!indexUrl.isNull()) location = Location(name); } else if (elementName == QLatin1String("qmlclass") || elementName == QLatin1String("qmlvaluetype") || elementName == QLatin1String("qmlbasictype")) { auto *qmlTypeNode = new QmlTypeNode(parent, name, elementName == QLatin1String("qmlclass") ? Node::QmlType : Node::QmlValueType); qmlTypeNode->setTitle(attributes.value(QLatin1String("title")).toString()); QString logicalModuleName = attributes.value(QLatin1String("qml-module-name")).toString(); if (!logicalModuleName.isEmpty()) m_qdb->addToQmlModule(logicalModuleName, qmlTypeNode); bool abstract = false; if (attributes.value(QLatin1String("abstract")) == QLatin1String("true")) abstract = true; qmlTypeNode->setAbstract(abstract); QString qmlFullBaseName = attributes.value(QLatin1String("qml-base-type")).toString(); if (!qmlFullBaseName.isEmpty()) { qmlTypeNode->setQmlBaseName(qmlFullBaseName); } if (attributes.hasAttribute(QLatin1String("location"))) name = attributes.value("location").toString(); if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + name); else if (!indexUrl.isNull()) location = Location(name); node = qmlTypeNode; } else if (elementName == QLatin1String("qmlproperty")) { QString type = attributes.value(QLatin1String("type")).toString(); bool attached = false; if (attributes.value(QLatin1String("attached")) == QLatin1String("true")) attached = true; bool readonly = false; if (attributes.value(QLatin1String("writable")) == QLatin1String("false")) readonly = true; auto *qmlPropertyNode = new QmlPropertyNode(parent, name, type, attached); qmlPropertyNode->markReadOnly(readonly); if (attributes.value(QLatin1String("required")) == QLatin1String("true")) qmlPropertyNode->setRequired(); node = qmlPropertyNode; } else if (elementName == QLatin1String("group")) { auto *collectionNode = m_qdb->addGroup(name); collectionNode->setTitle(attributes.value(QLatin1String("title")).toString()); collectionNode->setSubtitle(attributes.value(QLatin1String("subtitle")).toString()); if (attributes.value(QLatin1String("seen")) == QLatin1String("true")) collectionNode->markSeen(); node = collectionNode; } else if (elementName == QLatin1String("module")) { auto *collectionNode = m_qdb->addModule(name); collectionNode->setTitle(attributes.value(QLatin1String("title")).toString()); collectionNode->setSubtitle(attributes.value(QLatin1String("subtitle")).toString()); if (attributes.value(QLatin1String("seen")) == QLatin1String("true")) collectionNode->markSeen(); node = collectionNode; } else if (elementName == QLatin1String("qmlmodule")) { auto *collectionNode = m_qdb->addQmlModule(name); const QStringList info = QStringList() << name << QString(attributes.value(QLatin1String("qml-module-version")).toString()); collectionNode->setLogicalModuleInfo(info); collectionNode->setTitle(attributes.value(QLatin1String("title")).toString()); collectionNode->setSubtitle(attributes.value(QLatin1String("subtitle")).toString()); if (attributes.value(QLatin1String("seen")) == QLatin1String("true")) collectionNode->markSeen(); node = collectionNode; } else if (elementName == QLatin1String("page")) { QDocAttr subtype = QDocAttrNone; QString attr = attributes.value(QLatin1String("subtype")).toString(); if (attr == QLatin1String("attribution")) { subtype = QDocAttrAttribution; } else if (attr == QLatin1String("example")) { subtype = QDocAttrExample; } else if (attr == QLatin1String("file")) { subtype = QDocAttrFile; } else if (attr == QLatin1String("image")) { subtype = QDocAttrImage; } else if (attr == QLatin1String("page")) { subtype = QDocAttrDocument; } else if (attr == QLatin1String("externalpage")) { subtype = QDocAttrExternalPage; } else goto done; if (current->isExample()) { auto *exampleNode = static_cast(current); if (subtype == QDocAttrFile) { exampleNode->appendFile(name); goto done; } else if (subtype == QDocAttrImage) { exampleNode->appendImage(name); goto done; } } PageNode *pageNode = nullptr; if (subtype == QDocAttrExample) pageNode = new ExampleNode(parent, name); else if (subtype == QDocAttrExternalPage) pageNode = new ExternalPageNode(parent, name); else { pageNode = new PageNode(parent, name); if (subtype == QDocAttrAttribution) pageNode->markAttribution(); } pageNode->setTitle(attributes.value(QLatin1String("title")).toString()); if (attributes.hasAttribute(QLatin1String("location"))) name = attributes.value(QLatin1String("location")).toString(); if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + name); else if (!indexUrl.isNull()) location = Location(name); node = pageNode; } else if (elementName == QLatin1String("enum")) { auto *enumNode = new EnumNode(parent, name, attributes.hasAttribute("scoped")); if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + parent->name().toLower() + ".html"); else if (!indexUrl.isNull()) location = Location(parent->name().toLower() + ".html"); while (reader.readNextStartElement()) { QXmlStreamAttributes childAttributes = reader.attributes(); if (reader.name() == QLatin1String("value")) { EnumItem item(childAttributes.value(QLatin1String("name")).toString(), childAttributes.value(QLatin1String("value")).toString(), childAttributes.value(QLatin1String("since")).toString() ); enumNode->addItem(item); } else if (reader.name() == QLatin1String("keyword")) { insertTarget(TargetRec::Keyword, childAttributes, enumNode); } else if (reader.name() == QLatin1String("target")) { insertTarget(TargetRec::Target, childAttributes, enumNode); } reader.skipCurrentElement(); } node = enumNode; hasReadChildren = true; } else if (elementName == QLatin1String("typedef")) { if (attributes.hasAttribute("aliasedtype")) node = new TypeAliasNode(parent, name, attributes.value(QLatin1String("aliasedtype")).toString()); else node = new TypedefNode(parent, name); if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + parent->name().toLower() + ".html"); else if (!indexUrl.isNull()) location = Location(parent->name().toLower() + ".html"); } else if (elementName == QLatin1String("property")) { auto *propNode = new PropertyNode(parent, name); node = propNode; if (attributes.value(QLatin1String("bindable")) == QLatin1String("true")) propNode->setPropertyType(PropertyNode::PropertyType::BindableProperty); if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + parent->name().toLower() + ".html"); else if (!indexUrl.isNull()) location = Location(parent->name().toLower() + ".html"); } else if (elementName == QLatin1String("function")) { QString t = attributes.value(QLatin1String("meta")).toString(); bool attached = false; FunctionNode::Metaness metaness = FunctionNode::Plain; if (!t.isEmpty()) metaness = FunctionNode::getMetaness(t); if (attributes.value(QLatin1String("attached")) == QLatin1String("true")) attached = true; auto *fn = new FunctionNode(metaness, parent, name, attached); fn->setReturnType(attributes.value(QLatin1String("type")).toString()); if (fn->isCppNode()) { fn->setVirtualness(attributes.value(QLatin1String("virtual")).toString()); fn->setConst(attributes.value(QLatin1String("const")) == QLatin1String("true")); fn->setStatic(attributes.value(QLatin1String("static")) == QLatin1String("true")); fn->setFinal(attributes.value(QLatin1String("final")) == QLatin1String("true")); fn->setOverride(attributes.value(QLatin1String("override")) == QLatin1String("true")); if (attributes.value(QLatin1String("explicit")) == QLatin1String("true")) fn->markExplicit(); if (attributes.value(QLatin1String("constexpr")) == QLatin1String("true")) fn->markConstexpr(); if (attributes.value(QLatin1String("noexcept")) == QLatin1String("true")) { fn->markNoexcept(attributes.value("noexcept_expression").toString()); } qsizetype refness = attributes.value(QLatin1String("refness")).toUInt(); if (refness == 1) fn->setRef(true); else if (refness == 2) fn->setRefRef(true); /* Theoretically, this should ensure that each function node receives the same overload number and overload flag it was written with, and it should be unnecessary to call normalizeOverloads() for index nodes. */ if (attributes.value(QLatin1String("overload")) == QLatin1String("true")) fn->setOverloadNumber(attributes.value(QLatin1String("overload-number")).toUInt()); else fn->setOverloadNumber(0); } /* Note: The "signature" attribute was written to the index file, but it is not read back in. That is ok because we reconstruct the parameter list and the return type, from which the signature was built in the first place and from which it can be rebuilt. */ while (reader.readNextStartElement()) { QXmlStreamAttributes childAttributes = reader.attributes(); if (reader.name() == QLatin1String("parameter")) { // Do not use the default value for the parameter; it is not // required, and has been known to cause problems. QString type = childAttributes.value(QLatin1String("type")).toString(); QString name = childAttributes.value(QLatin1String("name")).toString(); fn->parameters().append(type, name); } else if (reader.name() == QLatin1String("keyword")) { insertTarget(TargetRec::Keyword, childAttributes, fn); } else if (reader.name() == QLatin1String("target")) { insertTarget(TargetRec::Target, childAttributes, fn); } reader.skipCurrentElement(); } node = fn; if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + parent->name().toLower() + ".html"); else if (!indexUrl.isNull()) location = Location(parent->name().toLower() + ".html"); hasReadChildren = true; } else if (elementName == QLatin1String("variable")) { node = new VariableNode(parent, name); if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + parent->name().toLower() + ".html"); else if (!indexUrl.isNull()) location = Location(parent->name().toLower() + ".html"); } else if (elementName == QLatin1String("keyword")) { insertTarget(TargetRec::Keyword, attributes, current); goto done; } else if (elementName == QLatin1String("target")) { insertTarget(TargetRec::Target, attributes, current); goto done; } else if (elementName == QLatin1String("contents")) { insertTarget(TargetRec::Contents, attributes, current); goto done; } else if (elementName == QLatin1String("proxy")) { node = new ProxyNode(parent, name); if (!indexUrl.isEmpty()) location = Location(indexUrl + QLatin1Char('/') + name.toLower() + ".html"); else if (!indexUrl.isNull()) location = Location(name.toLower() + ".html"); } else { goto done; } { if (!href.isEmpty()) { node->setUrl(href); // Include the index URL if it exists if (!node->isExternalPage() && !indexUrl.isEmpty()) node->setUrl(indexUrl + QLatin1Char('/') + href); } const QString access = attributes.value(QLatin1String("access")).toString(); if (access == "protected") node->setAccess(Access::Protected); else if ((access == "private") || (access == "internal")) node->setAccess(Access::Private); else node->setAccess(Access::Public); if (attributes.hasAttribute(QLatin1String("related"))) { node->setRelatedNonmember(true); m_relatedNodes << node; } if (attributes.hasAttribute(QLatin1String("threadsafety"))) { QString threadSafety = attributes.value(QLatin1String("threadsafety")).toString(); if (threadSafety == QLatin1String("non-reentrant")) node->setThreadSafeness(Node::NonReentrant); else if (threadSafety == QLatin1String("reentrant")) node->setThreadSafeness(Node::Reentrant); else if (threadSafety == QLatin1String("thread safe")) node->setThreadSafeness(Node::ThreadSafe); else node->setThreadSafeness(Node::UnspecifiedSafeness); } else node->setThreadSafeness(Node::UnspecifiedSafeness); QString status = attributes.value(QLatin1String("status")).toString(); // TODO: "obsolete" is kept for backward compatibility, remove in the near future if (status == QLatin1String("obsolete") || status == QLatin1String("deprecated")) node->setStatus(Node::Deprecated); else if (status == QLatin1String("preliminary")) node->setStatus(Node::Preliminary); else if (status == QLatin1String("internal")) node->setStatus(Node::Internal); else if (status == QLatin1String("ignored")) node->setStatus(Node::DontDocument); else node->setStatus(Node::Active); QString physicalModuleName = attributes.value(QLatin1String("module")).toString(); if (!physicalModuleName.isEmpty()) m_qdb->addToModule(physicalModuleName, node); QString since = attributes.value(QLatin1String("since")).toString(); if (!since.isEmpty()) { node->setSince(since); } if (attributes.hasAttribute(QLatin1String("documented"))) { if (attributes.value(QLatin1String("documented")) == QLatin1String("true")) node->setHadDoc(); } QString groupsAttr = attributes.value(QLatin1String("groups")).toString(); if (!groupsAttr.isEmpty()) { const QStringList groupNames = groupsAttr.split(QLatin1Char(',')); for (const auto &group : groupNames) { m_qdb->addToGroup(group, node); } } // Create some content for the node. QSet emptySet; Location t(filePath); if (!filePath.isEmpty()) { t.setLineNo(lineNo); node->setLocation(t); location = t; } Doc doc(location, location, QString(), emptySet, emptySet); // placeholder node->setDoc(doc); node->setIndexNodeFlag(); // Important: This node came from an index file. node->setOutputSubdirectory(m_project.toLower()); QString briefAttr = attributes.value(QLatin1String("brief")).toString(); if (!briefAttr.isEmpty()) { node->setReconstitutedBrief(briefAttr); } if (!hasReadChildren) { bool useParent = (elementName == QLatin1String("namespace") && name.isEmpty()); while (reader.readNextStartElement()) { if (useParent) readIndexSection(reader, parent, indexUrl); else readIndexSection(reader, node, indexUrl); } } } done: while (!reader.isEndElement()) { if (reader.readNext() == QXmlStreamReader::Invalid) { break; } } } void QDocIndexFiles::insertTarget(TargetRec::TargetType type, const QXmlStreamAttributes &attributes, Node *node) { int priority; switch (type) { case TargetRec::Keyword: priority = 1; break; case TargetRec::Target: priority = 2; break; case TargetRec::Contents: priority = 3; break; default: return; } QString name = attributes.value(QLatin1String("name")).toString(); QString title = attributes.value(QLatin1String("title")).toString(); m_qdb->insertTarget(name, title, type, node, priority); } /*! This function tries to resolve class inheritance immediately after the index file is read. It is not always possible to resolve a class inheritance at this point, because the base class might be in an index file that hasn't been read yet, or it might be in one of the header files that will be read for the current module. These cases will be resolved after all the index files and header and source files have been read, just prior to beginning the generate phase for the current module. I don't think this is completely correct because it always sets the access to public. */ void QDocIndexFiles::resolveIndex() { for (const auto &pair : std::as_const(m_basesList)) { const QStringList bases = pair.second.split(QLatin1Char(',')); for (const auto &base : bases) { QStringList basePath = base.split(QString("::")); Node *n = m_qdb->findClassNode(basePath); if (n) pair.first->addResolvedBaseClass(Access::Public, static_cast(n)); else pair.first->addUnresolvedBaseClass(Access::Public, basePath); } } // No longer needed. m_basesList.clear(); } static QString getAccessString(Access t) { switch (t) { case Access::Public: return QLatin1String("public"); case Access::Protected: return QLatin1String("protected"); case Access::Private: return QLatin1String("private"); default: break; } return QLatin1String("public"); } static QString getStatusString(Node::Status t) { switch (t) { case Node::Deprecated: return QLatin1String("deprecated"); case Node::Preliminary: return QLatin1String("preliminary"); case Node::Active: return QLatin1String("active"); case Node::Internal: return QLatin1String("internal"); case Node::DontDocument: return QLatin1String("ignored"); default: break; } return QLatin1String("active"); } static QString getThreadSafenessString(Node::ThreadSafeness t) { switch (t) { case Node::NonReentrant: return QLatin1String("non-reentrant"); case Node::Reentrant: return QLatin1String("reentrant"); case Node::ThreadSafe: return QLatin1String("thread safe"); case Node::UnspecifiedSafeness: default: break; } return QLatin1String("unspecified"); } /*! Returns the index of \a node in the list of related non-member nodes. */ int QDocIndexFiles::indexForNode(Node *node) { qsizetype i = m_relatedNodes.indexOf(node); if (i == -1) { i = m_relatedNodes.size(); m_relatedNodes << node; } return i; } /*! Adopts the related non-member node identified by \a index to the parent \a adoptiveParent. Returns \c true if successful. */ bool QDocIndexFiles::adoptRelatedNode(Aggregate *adoptiveParent, int index) { Node *related = m_relatedNodes.value(index); if (adoptiveParent && related) { adoptiveParent->adoptChild(related); return true; } return false; } /*! Generate the index section with the given \a writer for the \a node specified, returning true if an element was written, and returning false if an element is not written. \note Function nodes are processed in generateFunctionSection() */ bool QDocIndexFiles::generateIndexSection(QXmlStreamWriter &writer, Node *node, IndexSectionWriter *post) { if (m_gen == nullptr) m_gen = Generator::currentGenerator(); Q_ASSERT(m_gen); post_ = nullptr; /* Don't include index nodes in a new index file. */ if (node->isIndexNode()) return false; QString nodeName; QString logicalModuleName; QString logicalModuleVersion; QString qmlFullBaseName; QString baseNameAttr; QString moduleNameAttr; QString moduleVerAttr; switch (node->nodeType()) { case Node::Namespace: nodeName = "namespace"; break; case Node::Class: nodeName = "class"; break; case Node::Struct: nodeName = "struct"; break; case Node::Union: nodeName = "union"; break; case Node::HeaderFile: nodeName = "header"; break; case Node::QmlType: case Node::QmlValueType: nodeName = (node->nodeType() == Node::QmlType) ? "qmlclass" : "qmlvaluetype"; logicalModuleName = node->logicalModuleName(); baseNameAttr = "qml-base-type"; moduleNameAttr = "qml-module-name"; moduleVerAttr = "qml-module-version"; qmlFullBaseName = node->qmlFullBaseName(); break; case Node::Page: case Node::Example: case Node::ExternalPage: nodeName = "page"; break; case Node::Group: nodeName = "group"; break; case Node::Module: nodeName = "module"; break; case Node::QmlModule: nodeName = "qmlmodule"; moduleNameAttr = "qml-module-name"; moduleVerAttr = "qml-module-version"; logicalModuleName = node->logicalModuleName(); logicalModuleVersion = node->logicalModuleVersion(); break; case Node::Enum: nodeName = "enum"; break; case Node::TypeAlias: case Node::Typedef: nodeName = "typedef"; break; case Node::Property: nodeName = "property"; break; case Node::Variable: nodeName = "variable"; break; case Node::SharedComment: if (!node->isPropertyGroup()) return false; // Add an entry for property groups so that they can be linked to nodeName = "qmlproperty"; break; case Node::QmlProperty: nodeName = "qmlproperty"; break; case Node::Proxy: nodeName = "proxy"; break; case Node::Function: // Now processed in generateFunctionSection() default: return false; } QString objName = node->name(); // Special case: only the root node should have an empty name. if (objName.isEmpty() && node != m_qdb->primaryTreeRoot()) return false; writer.writeStartElement(nodeName); if (!node->isTextPageNode() && !node->isCollectionNode() && !node->isHeader()) { if (node->threadSafeness() != Node::UnspecifiedSafeness) writer.writeAttribute("threadsafety", getThreadSafenessString(node->threadSafeness())); } writer.writeAttribute("name", objName); // Write module and base type info for QML types if (!moduleNameAttr.isEmpty()) { if (!logicalModuleName.isEmpty()) writer.writeAttribute(moduleNameAttr, logicalModuleName); if (!logicalModuleVersion.isEmpty()) writer.writeAttribute(moduleVerAttr, logicalModuleVersion); } if (!baseNameAttr.isEmpty() && !qmlFullBaseName.isEmpty()) writer.writeAttribute(baseNameAttr, qmlFullBaseName); QString href; if (!node->isExternalPage()) { QString fullName = node->fullDocumentName(); if (fullName != objName) writer.writeAttribute("fullname", fullName); href = m_gen->fullDocumentLocation(node); } else href = node->name(); if (node->isQmlNode()) { Aggregate *p = node->parent(); if (p && p->isQmlType() && p->isAbstract()) href.clear(); } if (!href.isEmpty()) writer.writeAttribute("href", href); writer.writeAttribute("status", getStatusString(node->status())); if (!node->isTextPageNode() && !node->isCollectionNode() && !node->isHeader()) { writer.writeAttribute("access", getAccessString(node->access())); if (node->isAbstract()) writer.writeAttribute("abstract", "true"); } const Location &declLocation = node->declLocation(); if (!declLocation.fileName().isEmpty()) writer.writeAttribute("location", declLocation.fileName()); if (m_storeLocationInfo && !declLocation.filePath().isEmpty()) { writer.writeAttribute("filepath", declLocation.filePath()); writer.writeAttribute("lineno", QString("%1").arg(declLocation.lineNo())); } if (node->isRelatedNonmember()) writer.writeAttribute("related", QString::number(indexForNode(node))); if (!node->since().isEmpty()) writer.writeAttribute("since", node->since()); if (node->hasDoc()) writer.writeAttribute("documented", "true"); QStringList groups = m_qdb->groupNamesForNode(node); if (!groups.isEmpty()) writer.writeAttribute("groups", groups.join(QLatin1Char(','))); QString brief = node->doc().trimmedBriefText(node->name()).toString(); switch (node->nodeType()) { case Node::Class: case Node::Struct: case Node::Union: { // Classes contain information about their base classes. const auto *classNode = static_cast(node); const QList &bases = classNode->baseClasses(); QSet baseStrings; for (const auto &related : bases) { ClassNode *n = related.m_node; if (n) baseStrings.insert(n->fullName()); else if (!related.m_path.isEmpty()) baseStrings.insert(related.m_path.join(QLatin1String("::"))); } if (!baseStrings.isEmpty()) { QStringList baseStringsAsList = baseStrings.values(); baseStringsAsList.sort(); writer.writeAttribute("bases", baseStringsAsList.join(QLatin1Char(','))); } if (!node->physicalModuleName().isEmpty()) writer.writeAttribute("module", node->physicalModuleName()); if (!brief.isEmpty()) writer.writeAttribute("brief", brief); } break; case Node::HeaderFile: { const auto *headerNode = static_cast(node); if (!headerNode->physicalModuleName().isEmpty()) writer.writeAttribute("module", headerNode->physicalModuleName()); if (!brief.isEmpty()) writer.writeAttribute("brief", brief); writer.writeAttribute("title", headerNode->title()); writer.writeAttribute("fulltitle", headerNode->fullTitle()); writer.writeAttribute("subtitle", headerNode->subtitle()); } break; case Node::Namespace: { const auto *namespaceNode = static_cast(node); if (!namespaceNode->physicalModuleName().isEmpty()) writer.writeAttribute("module", namespaceNode->physicalModuleName()); if (!brief.isEmpty()) writer.writeAttribute("brief", brief); } break; case Node::QmlValueType: case Node::QmlType: { const auto *qmlTypeNode = static_cast(node); writer.writeAttribute("title", qmlTypeNode->title()); writer.writeAttribute("fulltitle", qmlTypeNode->fullTitle()); writer.writeAttribute("subtitle", qmlTypeNode->subtitle()); if (!brief.isEmpty()) writer.writeAttribute("brief", brief); } break; case Node::Page: case Node::Example: case Node::ExternalPage: { if (node->isExample()) writer.writeAttribute("subtype", "example"); else if (node->isExternalPage()) writer.writeAttribute("subtype", "externalpage"); else writer.writeAttribute("subtype", (static_cast(node)->isAttribution() ? "attribution" : "page")); const auto *pageNode = static_cast(node); writer.writeAttribute("title", pageNode->title()); writer.writeAttribute("fulltitle", pageNode->fullTitle()); writer.writeAttribute("subtitle", pageNode->subtitle()); if (!brief.isEmpty()) writer.writeAttribute("brief", brief); } break; case Node::Group: case Node::Module: case Node::QmlModule: { const auto *collectionNode = static_cast(node); writer.writeAttribute("seen", collectionNode->wasSeen() ? "true" : "false"); writer.writeAttribute("title", collectionNode->title()); if (!collectionNode->subtitle().isEmpty()) writer.writeAttribute("subtitle", collectionNode->subtitle()); if (!collectionNode->physicalModuleName().isEmpty()) writer.writeAttribute("module", collectionNode->physicalModuleName()); if (!brief.isEmpty()) writer.writeAttribute("brief", brief); } break; case Node::QmlProperty: { auto *qmlPropertyNode = static_cast(node); writer.writeAttribute("type", qmlPropertyNode->dataType()); writer.writeAttribute("attached", qmlPropertyNode->isAttached() ? "true" : "false"); writer.writeAttribute("writable", qmlPropertyNode->isWritable() ? "true" : "false"); if (qmlPropertyNode->isRequired()) writer.writeAttribute("required", "true"); if (!brief.isEmpty()) writer.writeAttribute("brief", brief); } break; case Node::Property: { const auto *propertyNode = static_cast(node); if (propertyNode->propertyType() == PropertyNode::PropertyType::BindableProperty) writer.writeAttribute("bindable", "true"); if (!brief.isEmpty()) writer.writeAttribute("brief", brief); // Property access function names for (qsizetype i{0}; i < (qsizetype)PropertyNode::FunctionRole::NumFunctionRoles; ++i) { auto role{(PropertyNode::FunctionRole)i}; for (const auto *fnNode : propertyNode->functions(role)) { writer.writeStartElement(PropertyNode::roleName(role)); writer.writeAttribute("name", fnNode->name()); writer.writeEndElement(); } } } break; case Node::Variable: { const auto *variableNode = static_cast(node); writer.writeAttribute("type", variableNode->dataType()); writer.writeAttribute("static", variableNode->isStatic() ? "true" : "false"); if (!brief.isEmpty()) writer.writeAttribute("brief", brief); } break; case Node::Enum: { const auto *enumNode = static_cast(node); if (enumNode->isScoped()) writer.writeAttribute("scoped", "true"); if (enumNode->flagsType()) writer.writeAttribute("typedef", enumNode->flagsType()->fullDocumentName()); const auto &items = enumNode->items(); for (const auto &item : items) { writer.writeStartElement("value"); writer.writeAttribute("name", item.name()); writer.writeAttribute("value", item.value()); if (!item.since().isEmpty()) writer.writeAttribute("since", item.since()); writer.writeEndElement(); // value } } break; case Node::Typedef: { const auto *typedefNode = static_cast(node); if (typedefNode->associatedEnum()) writer.writeAttribute("enum", typedefNode->associatedEnum()->fullDocumentName()); } break; case Node::TypeAlias: writer.writeAttribute("aliasedtype", static_cast(node)->aliasedType()); break; case Node::Function: // Now processed in generateFunctionSection() default: break; } /* For our pages, we canonicalize the target, keyword and content item names so that they can be used by qdoc for other sets of documentation. The reason we do this here is that we don't want to ruin externally composed indexes, containing non-qdoc-style target names when reading in indexes. targets and keywords are now allowed in any node, not just inner nodes. */ if (node->doc().hasTargets()) { bool external = false; if (node->isExternalPage()) external = true; const auto &targets = node->doc().targets(); for (const Atom *target : targets) { const QString &title = target->string(); QString name = Doc::canonicalTitle(title); writer.writeStartElement("target"); if (!external) writer.writeAttribute("name", name); else writer.writeAttribute("name", title); if (name != title) writer.writeAttribute("title", title); writer.writeEndElement(); // target } } if (node->doc().hasKeywords()) { const auto &keywords = node->doc().keywords(); for (const Atom *keyword : keywords) { const QString &title = keyword->string(); QString name = Doc::canonicalTitle(title); writer.writeStartElement("keyword"); writer.writeAttribute("name", name); if (name != title) writer.writeAttribute("title", title); writer.writeEndElement(); // keyword } } /* Some nodes have a table of contents. For these, we close the opening tag, create sub-elements for the items in the table of contents, and then add a closing tag for the element. Elements for all other nodes are closed in the opening tag. */ if (node->isPageNode() || node->isCollectionNode()) { if (node->doc().hasTableOfContents()) { for (int i = 0; i < node->doc().tableOfContents().size(); ++i) { Atom *item = node->doc().tableOfContents()[i]; int level = node->doc().tableOfContentsLevels()[i]; QString title = Text::sectionHeading(item).toString(); writer.writeStartElement("contents"); writer.writeAttribute("name", Doc::canonicalTitle(title)); writer.writeAttribute("title", title); writer.writeAttribute("level", QString::number(level)); writer.writeEndElement(); // contents } } } // WebXMLGenerator - skip the nested elements for example // files/images, as the generator produces them separately if (node->isExample() && m_gen->format() != QLatin1String("WebXML")) { const auto *exampleNode = static_cast(node); const auto &files = exampleNode->files(); for (const QString &file : files) { writer.writeStartElement("page"); writer.writeAttribute("name", file); QString href = m_gen->linkForExampleFile(file); writer.writeAttribute("href", href); writer.writeAttribute("status", "active"); writer.writeAttribute("subtype", "file"); writer.writeAttribute("title", ""); writer.writeAttribute("fulltitle", Generator::exampleFileTitle(exampleNode, file)); writer.writeAttribute("subtitle", file); writer.writeEndElement(); // page } const auto &images = exampleNode->images(); for (const QString &file : images) { writer.writeStartElement("page"); writer.writeAttribute("name", file); QString href = m_gen->linkForExampleFile(file); writer.writeAttribute("href", href); writer.writeAttribute("status", "active"); writer.writeAttribute("subtype", "image"); writer.writeAttribute("title", ""); writer.writeAttribute("fulltitle", Generator::exampleFileTitle(exampleNode, file)); writer.writeAttribute("subtitle", file); writer.writeEndElement(); // page } } // Append to the section if the callback object was set if (post) post->append(writer, node); post_ = post; return true; } /*! This function writes a element for \a fn to the index file using \a writer. */ void QDocIndexFiles::generateFunctionSection(QXmlStreamWriter &writer, FunctionNode *fn) { const QString objName = fn->name(); writer.writeStartElement("function"); writer.writeAttribute("name", objName); const QString fullName = fn->fullDocumentName(); if (fullName != objName) writer.writeAttribute("fullname", fullName); const QString href = m_gen->fullDocumentLocation(fn); if (!href.isEmpty()) writer.writeAttribute("href", href); if (fn->threadSafeness() != Node::UnspecifiedSafeness) writer.writeAttribute("threadsafety", getThreadSafenessString(fn->threadSafeness())); writer.writeAttribute("status", getStatusString(fn->status())); writer.writeAttribute("access", getAccessString(fn->access())); const Location &declLocation = fn->declLocation(); if (!declLocation.fileName().isEmpty()) writer.writeAttribute("location", declLocation.fileName()); if (m_storeLocationInfo && !declLocation.filePath().isEmpty()) { writer.writeAttribute("filepath", declLocation.filePath()); writer.writeAttribute("lineno", QString("%1").arg(declLocation.lineNo())); } if (fn->hasDoc()) writer.writeAttribute("documented", "true"); if (fn->isRelatedNonmember()) writer.writeAttribute("related", QString::number(indexForNode(fn))); if (!fn->since().isEmpty()) writer.writeAttribute("since", fn->since()); const QString brief = fn->doc().trimmedBriefText(fn->name()).toString(); writer.writeAttribute("meta", fn->metanessString()); if (fn->isCppNode()) { if (!fn->isNonvirtual()) writer.writeAttribute("virtual", fn->virtualness()); if (fn->isConst()) writer.writeAttribute("const", "true"); if (fn->isStatic()) writer.writeAttribute("static", "true"); if (fn->isFinal()) writer.writeAttribute("final", "true"); if (fn->isOverride()) writer.writeAttribute("override", "true"); if (fn->isExplicit()) writer.writeAttribute("explicit", "true"); if (fn->isConstexpr()) writer.writeAttribute("constexpr", "true"); if (auto noexcept_info = fn->getNoexcept()) { writer.writeAttribute("noexcept", "true"); if (!(*noexcept_info).isEmpty()) writer.writeAttribute("noexcept_expression", *noexcept_info); } /* This ensures that for functions that have overloads, the first function written is the one that is not an overload, and the overloads follow it immediately in the index file numbered from 1 to n. */ if (fn->isOverload() && (fn->overloadNumber() > 0)) { writer.writeAttribute("overload", "true"); writer.writeAttribute("overload-number", QString::number(fn->overloadNumber())); } if (fn->isRef()) writer.writeAttribute("refness", QString::number(1)); else if (fn->isRefRef()) writer.writeAttribute("refness", QString::number(2)); if (fn->hasAssociatedProperties()) { QStringList associatedProperties; for (const auto *node : fn->associatedProperties()) { associatedProperties << node->name(); } associatedProperties.sort(); writer.writeAttribute("associated-property", associatedProperties.join(QLatin1Char(','))); } } const auto return_type = fn->returnType(); if (!return_type.isEmpty()) writer.writeAttribute("type", return_type); if (fn->isCppNode()) { if (!brief.isEmpty()) writer.writeAttribute("brief", brief); /* Note: The "signature" attribute is written to the index file, but it is not read back in by qdoc. However, we need it for the webxml generator. */ const QString signature = appendAttributesToSignature(fn); writer.writeAttribute("signature", signature); QStringList groups = m_qdb->groupNamesForNode(fn); if (!groups.isEmpty()) writer.writeAttribute("groups", groups.join(QLatin1Char(','))); } for (int i = 0; i < fn->parameters().count(); ++i) { const Parameter ¶meter = fn->parameters().at(i); writer.writeStartElement("parameter"); writer.writeAttribute("type", parameter.type()); writer.writeAttribute("name", parameter.name()); writer.writeAttribute("default", parameter.defaultValue()); writer.writeEndElement(); // parameter } // Append to the section if the callback object was set if (post_) post_->append(writer, fn); writer.writeEndElement(); // function } /*! \internal Constructs the signature to be written to an index file for the function represented by FunctionNode \a fn. 'const' is already part of FunctionNode::signature(), which forms the basis for the signature returned by this method. The method adds, where applicable, the C++ keywords "final", "override", or "= 0", to the signature carried by the FunctionNode itself. */ QString QDocIndexFiles::appendAttributesToSignature(const FunctionNode *fn) const noexcept { QString signature = fn->signature(false, false); if (fn->isFinal()) signature += " final"; if (fn->isOverride()) signature += " override"; if (fn->isPureVirtual()) signature += " = 0"; return signature; } /*! This function outputs a element to the index file for each FunctionNode in \a aggregate using the \a writer. The \a aggregate has a function map that contains all the function nodes indexed by function name. But the map is not used as a multimap, so if the \a aggregate contains multiple functions with the same name, only one of those functions is in the function map index. The others are linked to that function using the next overload pointer. So this function generates a element for a function followed by a function element for each of its overloads. If a element represents an overload, it has an \c overload attribute set to \c true and an \c {overload-number} attribute set to the function's overload number. If the element does not represent an overload, the element has neither of these attributes. */ void QDocIndexFiles::generateFunctionSections(QXmlStreamWriter &writer, Aggregate *aggregate) { FunctionMap &functionMap = aggregate->functionMap(); if (!functionMap.isEmpty()) { for (auto it = functionMap.begin(); it != functionMap.end(); ++it) { FunctionNode *fn = it.value(); while (fn) { if (!fn->isInternal() || Config::instance().showInternal()) generateFunctionSection(writer, fn); fn = fn->nextOverload(); } } } } /*! Generate index sections for the child nodes of the given \a node using the \a writer specified. */ void QDocIndexFiles::generateIndexSections(QXmlStreamWriter &writer, Node *node, IndexSectionWriter *post) { /* Note that groups, modules, and QML modules are written after all the other nodes. */ if (node->isCollectionNode() || node->isGroup() || node->isModule() || node->isQmlModule()) return; if (node->isInternal() && !Config::instance().showInternal()) return; if (generateIndexSection(writer, node, post)) { if (node->isAggregate()) { auto *aggregate = static_cast(node); // First write the function children, then write the nonfunction children. generateFunctionSections(writer, aggregate); const auto &nonFunctionList = aggregate->nonfunctionList(); for (auto *node : nonFunctionList) generateIndexSections(writer, node, post); } if (node == root_) { /* We wait until the end of the index file to output the group, module, and QML module elements. By outputting them at the end, when we read the index file back in, all the group, module, and QML module member elements will have already been created. It is then only necessary to create the group, module, or QML module element and add each member to its member list. */ const CNMap &groups = m_qdb->groups(); if (!groups.isEmpty()) { for (auto it = groups.constBegin(); it != groups.constEnd(); ++it) { if (generateIndexSection(writer, it.value(), post)) writer.writeEndElement(); } } const CNMap &modules = m_qdb->modules(); if (!modules.isEmpty()) { for (auto it = modules.constBegin(); it != modules.constEnd(); ++it) { if (generateIndexSection(writer, it.value(), post)) writer.writeEndElement(); } } const CNMap &qmlModules = m_qdb->qmlModules(); if (!qmlModules.isEmpty()) { for (auto it = qmlModules.constBegin(); it != qmlModules.constEnd(); ++it) { if (generateIndexSection(writer, it.value(), post)) writer.writeEndElement(); } } } writer.writeEndElement(); } } /*! Writes a qdoc module index in XML to a file named \a fileName. \a url is the \c url attribute of the element. \a title is the \c title attribute of the element. \a g is a pointer to the current Generator in use, stored for later use. */ void QDocIndexFiles::generateIndex(const QString &fileName, const QString &url, const QString &title, Generator *g) { QFile file(fileName); if (!file.open(QFile::WriteOnly | QFile::Text)) return; qCDebug(lcQdoc) << "Writing index file:" << fileName; m_gen = g; m_relatedNodes.clear(); QXmlStreamWriter writer(&file); writer.setAutoFormatting(true); writer.writeStartDocument(); writer.writeDTD(""); writer.writeStartElement("INDEX"); writer.writeAttribute("url", url); writer.writeAttribute("title", title); writer.writeAttribute("version", m_qdb->version()); writer.writeAttribute("project", Config::instance().get(CONFIG_PROJECT).asString()); root_ = m_qdb->primaryTreeRoot(); if (!root_->tree()->indexTitle().isEmpty()) writer.writeAttribute("indexTitle", root_->tree()->indexTitle()); generateIndexSections(writer, root_, nullptr); writer.writeEndElement(); // INDEX writer.writeEndElement(); // QDOCINDEX writer.writeEndDocument(); file.close(); } QT_END_NAMESPACE