// Copyright (C) 2024 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only // Qt-Security score:significant reason:default #include #include #include #include #include QT_BEGIN_NAMESPACE Q_LOGGING_CATEGORY(semanticTokens, "qt.languageserver.semanticTokens") using namespace QQmlJS::AST; using namespace QQmlJS::Dom; using namespace QLspSpecification; namespace QmlHighlighting { static int mapToProtocolForQtCreator(QmlHighlightKind highlightKind) { switch (highlightKind) { case QmlHighlightKind::Comment: return int(SemanticTokenProtocolTypes::Comment); case QmlHighlightKind::QmlKeyword: return int(SemanticTokenProtocolTypes::Keyword); case QmlHighlightKind::QmlType: return int(SemanticTokenProtocolTypes::Type); case QmlHighlightKind::QmlImportId: case QmlHighlightKind::QmlNamespace: return int(SemanticTokenProtocolTypes::Namespace); case QmlHighlightKind::QmlLocalId: return int(SemanticTokenProtocolTypes::QmlLocalId); case QmlHighlightKind::QmlExternalId: return int(SemanticTokenProtocolTypes::QmlExternalId); case QmlHighlightKind::QmlProperty: return int(SemanticTokenProtocolTypes::Property); case QmlHighlightKind::QmlScopeObjectProperty: return int(SemanticTokenProtocolTypes::QmlScopeObjectProperty); case QmlHighlightKind::QmlRootObjectProperty: return int(SemanticTokenProtocolTypes::QmlRootObjectProperty); case QmlHighlightKind::QmlExternalObjectProperty: return int(SemanticTokenProtocolTypes::QmlExternalObjectProperty); case QmlHighlightKind::QmlMethod: return int(SemanticTokenProtocolTypes::Method); case QmlHighlightKind::QmlMethodParameter: return int(SemanticTokenProtocolTypes::Parameter); case QmlHighlightKind::QmlSignal: return int(SemanticTokenProtocolTypes::Method); case QmlHighlightKind::QmlSignalHandler: return int(SemanticTokenProtocolTypes::Property); case QmlHighlightKind::QmlEnumName: return int(SemanticTokenProtocolTypes::Enum); case QmlHighlightKind::QmlEnumMember: return int(SemanticTokenProtocolTypes::EnumMember); case QmlHighlightKind::QmlPragmaName: case QmlHighlightKind::QmlPragmaValue: return int(SemanticTokenProtocolTypes::Variable); case QmlHighlightKind::JsImport: return int(SemanticTokenProtocolTypes::Namespace); case QmlHighlightKind::JsGlobalVar: return int(SemanticTokenProtocolTypes::JsGlobalVar); case QmlHighlightKind::JsGlobalMethod: return int(SemanticTokenProtocolTypes::Method); case QmlHighlightKind::JsScopeVar: return int(SemanticTokenProtocolTypes::JsScopeVar); case QmlHighlightKind::JsLabel: return int(SemanticTokenProtocolTypes::Variable); case QmlHighlightKind::Number: return int(SemanticTokenProtocolTypes::Number); case QmlHighlightKind::String: return int(SemanticTokenProtocolTypes::String); case QmlHighlightKind::Operator: return int(SemanticTokenProtocolTypes::Operator); case QmlHighlightKind::QmlTypeModifier: return int(SemanticTokenProtocolTypes::Decorator); case QmlHighlightKind::Field: return int(SemanticTokenProtocolTypes::Field); case QmlHighlightKind::Unknown: default: return int(SemanticTokenProtocolTypes::Unknown); } } static int mapToProtocolDefault(QmlHighlightKind highlightKind) { switch (highlightKind) { case QmlHighlightKind::Comment: return int(SemanticTokenProtocolTypes::Comment); case QmlHighlightKind::QmlKeyword: return int(SemanticTokenProtocolTypes::Keyword); case QmlHighlightKind::QmlType: return int(SemanticTokenProtocolTypes::Type); case QmlHighlightKind::QmlImportId: case QmlHighlightKind::QmlNamespace: return int(SemanticTokenProtocolTypes::Namespace); case QmlHighlightKind::QmlLocalId: case QmlHighlightKind::QmlExternalId: return int(SemanticTokenProtocolTypes::Variable); case QmlHighlightKind::QmlProperty: case QmlHighlightKind::QmlScopeObjectProperty: case QmlHighlightKind::QmlRootObjectProperty: case QmlHighlightKind::QmlExternalObjectProperty: return int(SemanticTokenProtocolTypes::Property); case QmlHighlightKind::QmlMethod: return int(SemanticTokenProtocolTypes::Method); case QmlHighlightKind::QmlMethodParameter: return int(SemanticTokenProtocolTypes::Parameter); case QmlHighlightKind::QmlSignal: return int(SemanticTokenProtocolTypes::Method); case QmlHighlightKind::QmlSignalHandler: return int(SemanticTokenProtocolTypes::Method); case QmlHighlightKind::QmlEnumName: return int(SemanticTokenProtocolTypes::Enum); case QmlHighlightKind::QmlEnumMember: return int(SemanticTokenProtocolTypes::EnumMember); case QmlHighlightKind::QmlPragmaName: case QmlHighlightKind::QmlPragmaValue: return int(SemanticTokenProtocolTypes::Variable); case QmlHighlightKind::JsImport: return int(SemanticTokenProtocolTypes::Namespace); case QmlHighlightKind::JsGlobalVar: return int(SemanticTokenProtocolTypes::Variable); case QmlHighlightKind::JsGlobalMethod: return int(SemanticTokenProtocolTypes::Method); case QmlHighlightKind::JsScopeVar: return int(SemanticTokenProtocolTypes::Variable); case QmlHighlightKind::JsLabel: return int(SemanticTokenProtocolTypes::Variable); case QmlHighlightKind::Number: return int(SemanticTokenProtocolTypes::Number); case QmlHighlightKind::String: return int(SemanticTokenProtocolTypes::String); case QmlHighlightKind::Operator: return int(SemanticTokenProtocolTypes::Operator); case QmlHighlightKind::QmlTypeModifier: return int(SemanticTokenProtocolTypes::Decorator); case QmlHighlightKind::Field: return int(SemanticTokenProtocolTypes::Property); case QmlHighlightKind::Unknown: default: return int(SemanticTokenProtocolTypes::Unknown); } } /*! \internal \brief Further resolves the type of a JavaScriptIdentifier A global object can be in the object form or in the function form. For example, Date can be used as a constructor function (like new Date()) or as a object (like Date.now()). */ static std::optional resolveJsGlobalObjectKind(const DomItem &item, const QString &name) { // Some objects are not constructable, they are always objects. static QSet noConstructorObjects = { u"Math"_s, u"JSON"_s, u"Atomics"_s, u"Reflect"_s, u"console"_s }; // if the method name is in the list of noConstructorObjects, then it is a global object. Do not // perform further checks. if (noConstructorObjects.contains(name)) return QmlHighlightKind::JsGlobalVar; // Check if the method is called with new, then it is a constructor function if (item.directParent().internalKind() == DomType::ScriptNewMemberExpression) { return QmlHighlightKind::JsGlobalMethod; } if (DomItem containingCallExpression = item.filterUp( [](DomType k, const DomItem &) { return k == DomType::ScriptCallExpression; }, FilterUpOptions::ReturnOuter)) { // Call expression // if callee is binary expression, then the rightest part is the method name const auto callee = containingCallExpression.field(Fields::callee); if (callee.internalKind() == DomType::ScriptBinaryExpression) { const auto right = callee.field(Fields::right); if (right.internalKind() == DomType::ScriptIdentifierExpression && right.field(Fields::identifier).value().toString() == name) { return QmlHighlightKind::JsGlobalMethod; } else { return QmlHighlightKind::JsGlobalVar; } } else { return QmlHighlightKind::JsGlobalVar; } } return std::nullopt; } static int fromQmlModifierKindToLspTokenType(QmlHighlightModifiers highlightModifier) { using namespace QLspSpecification; using namespace Utils; int modifier = 0; if (highlightModifier.testFlag(QmlHighlightModifier::QmlPropertyDefinition)) addModifier(SemanticTokenModifiers::Definition, &modifier); if (highlightModifier.testFlag(QmlHighlightModifier::QmlDefaultProperty)) addModifier(SemanticTokenModifiers::DefaultLibrary, &modifier); if (highlightModifier.testFlag(QmlHighlightModifier::QmlFinalProperty)) addModifier(SemanticTokenModifiers::Static, &modifier); if (highlightModifier.testFlag(QmlHighlightModifier::QmlRequiredProperty)) addModifier(SemanticTokenModifiers::Abstract, &modifier); if (highlightModifier.testFlag(QmlHighlightModifier::QmlReadonlyProperty)) addModifier(SemanticTokenModifiers::Readonly, &modifier); return modifier; } static FieldFilter highlightingFilter() { QMultiMap fieldFilterAdd{}; QMultiMap fieldFilterRemove{ { QString(), Fields::propertyInfos.toString() }, { QString(), Fields::fileLocationsTree.toString() }, { QString(), Fields::importScope.toString() }, { QString(), Fields::defaultPropertyName.toString() }, { QString(), Fields::get.toString() }, }; return FieldFilter{ fieldFilterAdd, fieldFilterRemove }; } HighlightToken::HighlightToken(const QQmlJS::SourceLocation &loc, QmlHighlightKind kind, QmlHighlightModifiers modifiers) : loc(loc), kind(kind), modifiers(modifiers) { } HighlightingVisitor::HighlightingVisitor(const QQmlJS::Dom::DomItem &item, const std::optional &range) : m_range(range) { item.visitTree( Path(), [this](const Path &path, const DomItem &item, bool b) { return this->visitor(path, item, b); }, VisitOption::Default | VisitOption::NoPath, emptyChildrenVisitor, emptyChildrenVisitor, highlightingFilter()); } bool HighlightingVisitor::visitor(Path, const DomItem &item, bool) { if (m_range.has_value()) { const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return true; const auto regions = fLocs->info().regions; if (!Utils::rangeOverlapsWithSourceLocation(regions[MainRegion], m_range.value())) return true; } switch (item.internalKind()) { case DomType::Comment: { highlightComment(item); return true; } case DomType::Import: { highlightImport(item); return true; } case DomType::Binding: { highlightBinding(item); return true; } case DomType::Pragma: { highlightPragma(item); return true; } case DomType::EnumDecl: { highlightEnumDecl(item); return true; } case DomType::EnumItem: { highlightEnumItem(item); return true; } case DomType::QmlObject: { highlightQmlObject(item); return true; } case DomType::QmlComponent: { highlightComponent(item); return true; } case DomType::PropertyDefinition: { highlightPropertyDefinition(item); return true; } case DomType::MethodInfo: { highlightMethod(item); return true; } case DomType::ScriptLiteral: { highlightScriptLiteral(item); return true; } case DomType::ScriptCallExpression: { highlightCallExpression(item); return true; } case DomType::ScriptIdentifierExpression: { highlightIdentifier(item); return true; } default: if (item.ownerAs()) highlightScriptExpressions(item); return true; } Q_UNREACHABLE_RETURN(false); } void HighlightingVisitor::highlightComment(const DomItem &item) { const auto comment = item.as(); Q_ASSERT(comment); const auto locs = Utils::sourceLocationsFromMultiLineToken( comment->info().comment(), comment->info().sourceLocation()); for (const auto &loc : locs) addHighlight(loc, QmlHighlightKind::Comment); } void HighlightingVisitor::highlightImport(const DomItem &item) { const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return; const auto regions = fLocs->info().regions; const auto import = item.as(); Q_ASSERT(import); addHighlight(regions[ImportTokenRegion], QmlHighlightKind::QmlKeyword); if (import->uri.isModule()) addHighlight(regions[ImportUriRegion], QmlHighlightKind::QmlImportId); else addHighlight(regions[ImportUriRegion], QmlHighlightKind::String); if (regions.contains(VersionRegion)) addHighlight(regions[VersionRegion], QmlHighlightKind::Number); if (regions.contains(AsTokenRegion)) { addHighlight(regions[AsTokenRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[IdNameRegion], QmlHighlightKind::QmlNamespace); } } void HighlightingVisitor::highlightBinding(const DomItem &item) { const auto binding = item.as(); Q_ASSERT(binding); const auto fLocs = FileLocations::treeOf(item); if (!fLocs) { qCDebug(semanticTokens) << "Can't find the locations for" << item.internalKind(); return; } const auto regions = fLocs->info().regions; // If dotted name, then defer it to be handled in ScriptIdentifierExpression if (binding->name().contains("."_L1)) return; if (binding->bindingType() != BindingType::Normal) { addHighlight(regions[OnTokenRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlProperty); return; } return addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlProperty); } void HighlightingVisitor::highlightPragma(const DomItem &item) { const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return; const auto regions = fLocs->info().regions; addHighlight(regions[PragmaKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlPragmaName ); const auto pragma = item.as(); for (auto i = 0; i < pragma->values.size(); ++i) { DomItem value = item.field(Fields::values).index(i); const auto valueRegions = FileLocations::treeOf(value)->info().regions; addHighlight(valueRegions[PragmaValuesRegion], QmlHighlightKind::QmlPragmaValue); } return; } void HighlightingVisitor::highlightEnumDecl(const DomItem &item) { const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return; const auto regions = fLocs->info().regions; addHighlight(regions[EnumKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlEnumName); } void HighlightingVisitor::highlightEnumItem(const DomItem &item) { const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return; const auto regions = fLocs->info().regions; addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlEnumMember); if (regions.contains(EnumValueRegion)) addHighlight(regions[EnumValueRegion], QmlHighlightKind::Number); } void HighlightingVisitor::highlightQmlObject(const DomItem &item) { const auto qmlObject = item.as(); Q_ASSERT(qmlObject); const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return; const auto regions = fLocs->info().regions; // Handle ids here if (!qmlObject->idStr().isEmpty()) { addHighlight(regions[IdTokenRegion], QmlHighlightKind::QmlProperty); addHighlight(regions[IdNameRegion], QmlHighlightKind::QmlLocalId); } // If dotted name, then defer it to be handled in ScriptIdentifierExpression if (qmlObject->name().contains("."_L1)) return; addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlType); } void HighlightingVisitor::highlightComponent(const DomItem &item) { const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return; const auto regions = fLocs->info().regions; const auto componentKeywordIt = regions.constFind(ComponentKeywordRegion); if (componentKeywordIt == regions.constEnd()) return; // not an inline component, no need for highlighting addHighlight(*componentKeywordIt, QmlHighlightKind::QmlKeyword); addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlType); } void HighlightingVisitor::highlightPropertyDefinition(const DomItem &item) { const auto propertyDef = item.as(); Q_ASSERT(propertyDef); const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return; const auto regions = fLocs->info().regions; QmlHighlightModifiers modifier = QmlHighlightModifier::QmlPropertyDefinition; if (propertyDef->isDefaultMember) { modifier |= QmlHighlightModifier::QmlDefaultProperty; addHighlight(regions[DefaultKeywordRegion], QmlHighlightKind::QmlKeyword); } if (propertyDef->isFinal) { modifier |= QmlHighlightModifier::QmlFinalProperty; addHighlight(regions[FinalKeywordRegion], QmlHighlightKind::QmlKeyword); } if (propertyDef->isRequired) { modifier |= QmlHighlightModifier::QmlRequiredProperty; addHighlight(regions[RequiredKeywordRegion], QmlHighlightKind::QmlKeyword); } if (propertyDef->isReadonly) { modifier |= QmlHighlightModifier::QmlReadonlyProperty; addHighlight(regions[ReadonlyKeywordRegion], QmlHighlightKind::QmlKeyword); } addHighlight(regions[PropertyKeywordRegion], QmlHighlightKind::QmlKeyword); if (propertyDef->isAlias()) addHighlight(regions[TypeIdentifierRegion], QmlHighlightKind::QmlKeyword); else addHighlight(regions[TypeIdentifierRegion], QmlHighlightKind::QmlType); addHighlight(regions[TypeModifierRegion], QmlHighlightKind::QmlTypeModifier); addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlProperty, modifier); } void HighlightingVisitor::highlightMethod(const DomItem &item) { const auto method = item.as(); Q_ASSERT(method); const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return; const auto regions = fLocs->info().regions; switch (method->methodType) { case MethodInfo::Signal: { addHighlight(regions[SignalKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlMethod); break; } case MethodInfo::Method: { addHighlight(regions[FunctionKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlMethod); addHighlight(regions[TypeIdentifierRegion], QmlHighlightKind::QmlType); break; } default: Q_UNREACHABLE(); } for (auto i = 0; i < method->parameters.size(); ++i) { DomItem parameter = item.field(Fields::parameters).index(i); const auto paramRegions = FileLocations::treeOf(parameter)->info().regions; addHighlight(paramRegions[IdentifierRegion], QmlHighlightKind::QmlMethodParameter); addHighlight(paramRegions[TypeIdentifierRegion], QmlHighlightKind::QmlType); } return; } void HighlightingVisitor::highlightScriptLiteral(const DomItem &item) { const auto literal = item.as(); Q_ASSERT(literal); const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return; const auto regions = fLocs->info().regions; if (std::holds_alternative(literal->literalValue())) { const auto file = item.containingFile().as(); if (!file) return; const auto &code = file->engine()->code(); const auto offset = regions[MainRegion].offset; const auto length = regions[MainRegion].length; const QStringView literalCode = QStringView{code}.mid(offset, length); const auto &locs = Utils::sourceLocationsFromMultiLineToken( literalCode, regions[MainRegion]); for (const auto &loc : locs) addHighlight(loc, QmlHighlightKind::String); } else if (std::holds_alternative(literal->literalValue())) addHighlight(regions[MainRegion], QmlHighlightKind::Number); else if (std::holds_alternative(literal->literalValue())) addHighlight(regions[MainRegion], QmlHighlightKind::QmlKeyword); else if (std::holds_alternative(literal->literalValue())) addHighlight(regions[MainRegion], QmlHighlightKind::QmlKeyword); else qCWarning(semanticTokens) << "Invalid literal variant"; } void HighlightingVisitor::highlightIdentifier(const DomItem &item) { using namespace QLspSpecification; const auto id = item.as(); Q_ASSERT(id); const auto loc = id->mainRegionLocation(); // Many of the scriptIdentifiers expressions are already handled by // other cases. In those cases, if the location offset is already in the list // we don't need to perform expensive resolveExpressionType operation. if (m_highlights.contains(loc.offset)) return; // If the item is a field member base, we need to resolve the expression type // If the item is a field member access, we don't need to resolve the expression type // because it is already resolved in the first element. if (QQmlLSUtils::isFieldMemberAccess(item)) highlightFieldMemberAccess(item, loc); else highlightBySemanticAnalysis(item, loc); } void HighlightingVisitor::highlightCallExpression(const DomItem &item) { const auto highlight = [this](const DomItem &item) { if (item.internalKind() == DomType::ScriptIdentifierExpression) { const auto id = item.as(); Q_ASSERT(id); const auto loc = id->mainRegionLocation(); addHighlight(loc, QmlHighlightKind::QmlMethod); } }; if (item.internalKind() == DomType::ScriptCallExpression) { // If the item is a call expression, we need to highlight the callee. const auto callee = item.field(Fields::callee); if (callee.internalKind() == DomType::ScriptIdentifierExpression) { highlight(callee); return; } else if (callee.internalKind() == DomType::ScriptBinaryExpression) { // If the callee is a binary expression, we need to highlight the right part. const auto right = callee.field(Fields::right); if (right.internalKind() == DomType::ScriptIdentifierExpression) highlight(right); return; } } } void HighlightingVisitor::highlightFieldMemberAccess(const DomItem &item, QQmlJS::SourceLocation loc) { // enum fields and qualified module identifiers are not just fields. Do semantic analysis if // the identifier name is an uppercase string. const auto name = item.field(Fields::identifier).value().toString(); if (!name.isEmpty() && name.at(0).category() == QChar::Letter_Uppercase) { // maybe the identifier is an attached type or enum members, use semantic analysis to figure // out. return highlightBySemanticAnalysis(item, loc); } // Check if the name is a method const auto expression = QQmlLSUtils::resolveExpressionType(item, QQmlLSUtils::ResolveOptions::ResolveOwnerType); if (!expression) { addHighlight(loc, QmlHighlightKind::Field); return; } if (expression->type == QQmlLSUtils::MethodIdentifier || expression->type == QQmlLSUtils::LambdaMethodIdentifier) { addHighlight(loc, QmlHighlightKind::QmlMethod); return; } else { return addHighlight(loc, QmlHighlightKind::Field); } } void HighlightingVisitor::highlightBySemanticAnalysis(const DomItem &item, QQmlJS::SourceLocation loc) { const auto expression = QQmlLSUtils::resolveExpressionType( item, QQmlLSUtils::ResolveOptions::ResolveOwnerType); if (!expression) { addHighlight(loc, QmlHighlightKind::Unknown); return; } switch (expression->type) { case QQmlLSUtils::QmlComponentIdentifier: addHighlight(loc, QmlHighlightKind::QmlType); return; case QQmlLSUtils::JavaScriptIdentifier: { QmlHighlightKind tokenType = QmlHighlightKind::JsScopeVar; QmlHighlightModifiers modifier = QmlHighlightModifier::None; if (const auto scope = expression->semanticScope) { if (const auto jsIdentifier = scope->jsIdentifier(*expression->name)) { if (jsIdentifier->kind == QQmlJSScope::JavaScriptIdentifier::Parameter) tokenType = QmlHighlightKind::QmlMethodParameter; if (jsIdentifier->isConst) { modifier |= QmlHighlightModifier::QmlReadonlyProperty; } addHighlight(loc, tokenType, modifier); return; } } if (const auto name = expression->name) { if (const auto highlightKind = resolveJsGlobalObjectKind(item, *name)) return addHighlight(loc, *highlightKind); } return; } case QQmlLSUtils::PropertyIdentifier: { if (const auto scope = expression->semanticScope) { QmlHighlightKind tokenType = QmlHighlightKind::QmlProperty; if (scope == item.qmlObject().semanticScope()) { tokenType = QmlHighlightKind::QmlScopeObjectProperty; } else if (scope == item.rootQmlObject(GoTo::MostLikely).semanticScope()) { tokenType = QmlHighlightKind::QmlRootObjectProperty; } else { tokenType = QmlHighlightKind::QmlExternalObjectProperty; } const auto property = scope->property(expression->name.value()); QmlHighlightModifiers modifier = QmlHighlightModifier::None; if (!property.isWritable()) modifier |= QmlHighlightModifier::QmlReadonlyProperty; addHighlight(loc, tokenType, modifier); } return; } case QQmlLSUtils::PropertyChangedSignalIdentifier: addHighlight(loc, QmlHighlightKind::QmlSignal); return; case QQmlLSUtils::PropertyChangedHandlerIdentifier: addHighlight(loc, QmlHighlightKind::QmlSignalHandler); return; case QQmlLSUtils::SignalIdentifier: addHighlight(loc, QmlHighlightKind::QmlSignal); return; case QQmlLSUtils::SignalHandlerIdentifier: addHighlight(loc, QmlHighlightKind::QmlSignalHandler); return; case QQmlLSUtils::MethodIdentifier: addHighlight(loc, QmlHighlightKind::QmlMethod); return; case QQmlLSUtils::QmlObjectIdIdentifier: { const auto qmlfile = item.fileObject().as(); if (!qmlfile) { addHighlight(loc, QmlHighlightKind::Unknown); return; } const auto resolver = qmlfile->typeResolver(); if (!resolver) { addHighlight(loc, QmlHighlightKind::Unknown); return; } const auto &objects = resolver->objectsById(); if (expression->name.has_value()) { const auto &name = expression->name.value(); const auto boundName = objects.id(expression->semanticScope, item.qmlObject().semanticScope()); if (!boundName.isEmpty() && name == boundName) { // If the name is the same as the bound name, then it is a local id. addHighlight(loc, QmlHighlightKind::QmlLocalId); return; } else { addHighlight(loc, QmlHighlightKind::QmlExternalId); return; } } else { addHighlight(loc, QmlHighlightKind::QmlExternalId); return; } } case QQmlLSUtils::SingletonIdentifier: addHighlight(loc, QmlHighlightKind::QmlType); return; case QQmlLSUtils::EnumeratorIdentifier: addHighlight(loc, QmlHighlightKind::QmlEnumName); return; case QQmlLSUtils::EnumeratorValueIdentifier: addHighlight(loc, QmlHighlightKind::QmlEnumMember); return; case QQmlLSUtils::AttachedTypeIdentifier: case QQmlLSUtils::AttachedTypeIdentifierInBindingTarget: addHighlight(loc, QmlHighlightKind::QmlType); return; case QQmlLSUtils::GroupedPropertyIdentifier: addHighlight(loc, QmlHighlightKind::QmlProperty); return; case QQmlLSUtils::QualifiedModuleIdentifier: addHighlight(loc, QmlHighlightKind::QmlNamespace); return; default: qCWarning(semanticTokens) << QString::fromLatin1("Semantic token for %1 has not been implemented yet") .arg(int(expression->type)); } } void HighlightingVisitor::highlightScriptExpressions(const DomItem &item) { const auto fLocs = FileLocations::treeOf(item); if (!fLocs) return; const auto regions = fLocs->info().regions; switch (item.internalKind()) { case DomType::ScriptLiteral: highlightScriptLiteral(item); return; case DomType::ScriptForStatement: addHighlight(regions[ForKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[TypeIdentifierRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptVariableDeclaration: { addHighlight(regions[TypeIdentifierRegion], QmlHighlightKind::QmlKeyword); return; } case DomType::ScriptReturnStatement: addHighlight(regions[ReturnKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptCaseClause: addHighlight(regions[CaseKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptDefaultClause: addHighlight(regions[DefaultKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptSwitchStatement: addHighlight(regions[SwitchKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptWhileStatement: addHighlight(regions[WhileKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptDoWhileStatement: addHighlight(regions[DoKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[WhileKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptTryCatchStatement: addHighlight(regions[TryKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[CatchKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[FinallyKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptForEachStatement: addHighlight(regions[TypeIdentifierRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[ForKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[InOfTokenRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptThrowStatement: addHighlight(regions[ThrowKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptBreakStatement: addHighlight(regions[BreakKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptContinueStatement: addHighlight(regions[ContinueKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptIfStatement: addHighlight(regions[IfKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[ElseKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptLabelledStatement: addHighlight(regions[IdentifierRegion], QmlHighlightKind::JsLabel); return; case DomType::ScriptConditionalExpression: addHighlight(regions[QuestionMarkTokenRegion], QmlHighlightKind::Operator); addHighlight(regions[ColonTokenRegion], QmlHighlightKind::Operator); return; case DomType::ScriptUnaryExpression: case DomType::ScriptPostExpression: addHighlight(regions[OperatorTokenRegion], QmlHighlightKind::Operator); return; case DomType::ScriptType: addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlType); addHighlight(regions[TypeIdentifierRegion], QmlHighlightKind::QmlType); return; case DomType::ScriptFunctionExpression: { addHighlight(regions[FunctionKeywordRegion], QmlHighlightKind::QmlKeyword); addHighlight(regions[IdentifierRegion], QmlHighlightKind::QmlMethod); return; } case DomType::ScriptYieldExpression: addHighlight(regions[YieldKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptThisExpression: addHighlight(regions[ThisKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptSuperLiteral: addHighlight(regions[SuperKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptNewMemberExpression: case DomType::ScriptNewExpression: addHighlight(regions[NewKeywordRegion], QmlHighlightKind::QmlKeyword); return; case DomType::ScriptTemplateExpressionPart: addHighlight(regions[DollarLeftBraceTokenRegion], QmlHighlightKind::Operator); visitor(Path(), item.field(Fields::expression), false); addHighlight(regions[RightBraceRegion], QmlHighlightKind::Operator); return; case DomType::ScriptTemplateLiteral: addHighlight(regions[LeftBacktickTokenRegion], QmlHighlightKind::String); addHighlight(regions[RightBacktickTokenRegion], QmlHighlightKind::String); return; case DomType::ScriptTemplateStringPart: { // handle multiline case QString code = item.field(Fields::value).value().toString(); const auto &locs = Utils::sourceLocationsFromMultiLineToken( code, regions[MainRegion]); for (const auto &loc : locs) addHighlight(loc, QmlHighlightKind::String); return; } default: qCDebug(semanticTokens) << "Script Expressions with kind" << item.internalKind() << "not implemented"; } } void HighlightingVisitor::addHighlight(const QQmlJS::SourceLocation &loc, QmlHighlightKind highlightKind, QmlHighlightModifiers modifierKind) { return Utils::addHighlight(m_highlights, loc, highlightKind, modifierKind); } /*! \internal \brief Returns multiple source locations for a given raw comment Needed by semantic highlighting of comments. LSP clients usually don't support multiline tokens. In QML, we can have multiline tokens like string literals and comments. This method generates multiple source locations of sub-elements of token split by a newline delimiter. */ QList Utils::sourceLocationsFromMultiLineToken(QStringView stringLiteral, const QQmlJS::SourceLocation &locationInDocument) { auto lineBreakLength = qsizetype(std::char_traits::length("\n")); const auto lineLengths = [&lineBreakLength](QStringView literal) { std::vector lineLengths; qsizetype startIndex = 0; qsizetype pos = literal.indexOf(u'\n'); while (pos != -1) { // TODO: QTBUG-106813 // Since a document could be opened in normalized form // we can't use platform dependent newline handling here. // Thus, we check manually if the literal contains \r so that we split // the literal at the correct offset. if (pos - 1 > 0 && literal[pos - 1] == u'\r') { // Handle Windows line endings lineBreakLength = qsizetype(std::char_traits::length("\r\n")); // Move pos to the index of '\r' pos = pos - 1; } lineLengths.push_back(pos - startIndex); // Advance the lookup index, so it won't find the same index. startIndex = pos + lineBreakLength; pos = literal.indexOf('\n'_L1, startIndex); } // Push the last line if (startIndex < literal.length()) { lineLengths.push_back(literal.length() - startIndex); } return lineLengths; }; QList result; // First token location should start from the "stringLiteral"'s // location in the qml document. QQmlJS::SourceLocation lineLoc = locationInDocument; for (const auto lineLength : lineLengths(stringLiteral)) { lineLoc.length = lineLength; result.push_back(lineLoc); // update for the next line lineLoc.offset += lineLoc.length + lineBreakLength; ++lineLoc.startLine; lineLoc.startColumn = 1; } return result; } QList Utils::encodeSemanticTokens(const HighlightsContainer &highlights, HighlightingMode mode) { QList result; constexpr auto tokenEncodingLength = 5; result.reserve(tokenEncodingLength * highlights.size()); int prevLine = 0; int prevColumn = 0; const auto m_mapToProtocol = mode == HighlightingMode::Default ? mapToProtocolDefault : mapToProtocolForQtCreator; std::for_each(highlights.constBegin(), highlights.constEnd(), [&](const auto &token) { int length = token.loc.length; int line = token.loc.startLine - 1; // protocol is 0-based int col = token.loc.startColumn - 1; // protocol is 0-based Q_ASSERT(line >= prevLine); if (line != prevLine) prevColumn = 0; result.emplace_back(line - prevLine); result.emplace_back(col - prevColumn); result.emplace_back(length); result.emplace_back(m_mapToProtocol(token.kind)); result.emplace_back(fromQmlModifierKindToLspTokenType(token.modifiers)); prevLine = line; prevColumn = col; }); return result; } /*! \internal Computes the modifier value. Modifier is read as binary value in the protocol. The location of the bits set are interpreted as the indices of the tokenModifiers list registered by the server. Then, the client modifies the highlighting of the token. tokenModifiersList: ["declaration", definition, readonly, static ,,,] To set "definition" and "readonly", we need to send 0b00000110 */ void Utils::addModifier(SemanticTokenModifiers modifier, int *baseModifier) { if (!baseModifier) return; *baseModifier |= (1 << int(modifier)); } /*! \internal Check if the ranges overlap by ensuring that one range starts before the other ends */ bool Utils::rangeOverlapsWithSourceLocation(const QQmlJS::SourceLocation &loc, const HighlightsRange &r) { int startOffsetItem = int(loc.offset); int endOffsetItem = startOffsetItem + int(loc.length); return (startOffsetItem <= r.endOffset) && (r.startOffset <= endOffsetItem); } /* \internal Increments the resultID by one. */ void Utils::updateResultID(QByteArray &resultID) { int length = resultID.length(); for (int i = length - 1; i >= 0; --i) { if (resultID[i] == '9') { resultID[i] = '0'; } else { resultID[i] = resultID[i] + 1; return; } } resultID.prepend('1'); } /* \internal A utility method that computes the difference of two list. The first argument is the encoded token data of the file before edited. The second argument is the encoded token data after the file is edited. Returns a list of SemanticTokensEdit as expected by the protocol. */ QList Utils::computeDiff(const QList &oldData, const QList &newData) { // Find the iterators pointing the first mismatch, from the start const auto [oldStart, newStart] = std::mismatch(oldData.cbegin(), oldData.cend(), newData.cbegin(), newData.cend()); // Find the iterators pointing the first mismatch, from the end // but the iterators shouldn't pass over the start iterators found above. const auto [r1, r2] = std::mismatch(oldData.crbegin(), std::make_reverse_iterator(oldStart), newData.crbegin(), std::make_reverse_iterator(newStart)); const auto oldEnd = r1.base(); const auto newEnd = r2.base(); // no change if (oldStart == oldEnd && newStart == newEnd) return {}; SemanticTokensEdit edit; edit.start = int(std::distance(newData.cbegin(), newStart)); edit.deleteCount = int(std::distance(oldStart, oldEnd)); if (newStart >= newData.cbegin() && newEnd <= newData.cend() && newStart < newEnd) edit.data.emplace(newStart, newEnd); return { std::move(edit) }; } void Utils::addHighlight(HighlightsContainer &out, const QQmlJS::SourceLocation &loc, QmlHighlightKind highlightKind, QmlHighlightModifiers modifierKind) { if (!loc.isValid() || loc.length == 0) { qCDebug(semanticTokens) << "Invalid locations: Cannot add highlight to token"; return; } if (!out.contains(loc.offset)) out.insert(loc.offset, HighlightToken(loc, highlightKind, modifierKind)); } HighlightsContainer Utils::visitTokens(const QQmlJS::Dom::DomItem &item, const std::optional &range) { using namespace QQmlJS::Dom; HighlightingVisitor highlightDomElements(item, range); return highlightDomElements.highlights(); } QList Utils::collectTokens(const QQmlJS::Dom::DomItem &item, const std::optional &range, HighlightingMode mode) { return Utils::encodeSemanticTokens(visitTokens(item, range), mode); } } // namespace QmlHighlighting QT_END_NAMESPACE