UNPKG

langium

Version:

A language engineering tool for the Language Server Protocol

76 lines (69 loc) 3.99 kB
/****************************************************************************** * Copyright 2021 TypeFox GmbH * This program and the accompanying materials are made available under the * terms of the MIT License, which is available in the project root. ******************************************************************************/ import type { DocumentHighlightParams } from 'vscode-languageserver'; import type { CancellationToken } from '../utils/cancellation.js'; import type { GrammarConfig } from '../languages/grammar-config.js'; import type { NameProvider } from '../references/name-provider.js'; import type { FindReferencesOptions, References } from '../references/references.js'; import type { LangiumServices } from './lsp-services.js'; import type { MaybePromise } from '../utils/promise-utils.js'; import type { ReferenceDescription } from '../workspace/ast-descriptions.js'; import type { LangiumDocument } from '../workspace/documents.js'; import { DocumentHighlight } from 'vscode-languageserver'; import { getDocument } from '../utils/ast-utils.js'; import { findDeclarationNodeAtOffset } from '../utils/cst-utils.js'; import { UriUtils } from '../utils/uri-utils.js'; /** * Language-specific service for handling document highlight requests. */ export interface DocumentHighlightProvider { /** * Handle a document highlight request. * * @param document The document in which the request was received. * @param params The parameters of the document highlight request. * @param cancelToken A cancellation token that can be used to cancel the request. * @returns The document highlights or `undefined` if no highlights are available. * @throws `OperationCancelled` if cancellation is detected during execution * @throws `ResponseError` if an error is detected that should be sent as response to the client */ getDocumentHighlight(document: LangiumDocument, params: DocumentHighlightParams, cancelToken?: CancellationToken): MaybePromise<DocumentHighlight[] | undefined>; } export class DefaultDocumentHighlightProvider implements DocumentHighlightProvider { protected readonly references: References; protected readonly nameProvider: NameProvider; protected readonly grammarConfig: GrammarConfig; constructor(services: LangiumServices) { this.references = services.references.References; this.nameProvider = services.references.NameProvider; this.grammarConfig = services.parser.GrammarConfig; } getDocumentHighlight(document: LangiumDocument, params: DocumentHighlightParams, _cancelToken?: CancellationToken): MaybePromise<DocumentHighlight[] | undefined> { const rootNode = document.parseResult.value.$cstNode; if (!rootNode) { return undefined; } const selectedNode = findDeclarationNodeAtOffset(rootNode, document.textDocument.offsetAt(params.position), this.grammarConfig.nameRegexp); if (!selectedNode) { return undefined; } const targetAstNode = this.references.findDeclarations(selectedNode); const highlights: DocumentHighlight[] = []; for (const target of targetAstNode) { const includeDeclaration = UriUtils.equals(getDocument(target).uri, document.uri); const options: FindReferencesOptions = { documentUri: document.uri, includeDeclaration: includeDeclaration }; const references = this.references.findReferences(target, options); highlights.push(...references.map(ref => this.createDocumentHighlight(ref)).toArray()); } return highlights; } /** * Override this method to determine the highlight kind of the given reference. */ protected createDocumentHighlight(reference: ReferenceDescription): DocumentHighlight { return DocumentHighlight.create(reference.segment.range); } }