UNPKG

langium

Version:

A language engineering tool for the Language Server Protocol

141 lines (123 loc) 6.01 kB
/****************************************************************************** * Copyright 2023 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 { CancellationToken, TypeHierarchyItem, TypeHierarchyPrepareParams, TypeHierarchySubtypesParams, TypeHierarchySupertypesParams } from 'vscode-languageserver'; import { SymbolKind } from 'vscode-languageserver'; import type { GrammarConfig } from '../languages/grammar-config.js'; import type { NameProvider } from '../references/name-provider.js'; import type { References } from '../references/references.js'; import type { LangiumServices } from './lsp-services.js'; import type { AstNode } from '../syntax-tree.js'; import { findDeclarationNodeAtOffset } from '../utils/cst-utils.js'; import { URI } from '../utils/uri-utils.js'; import type { LangiumDocument, LangiumDocuments } from '../workspace/documents.js'; import type { MaybePromise } from '../utils/promise-utils.js'; /** * Language-specific service for handling type hierarchy requests. */ export interface TypeHierarchyProvider { prepareTypeHierarchy(document: LangiumDocument, params: TypeHierarchyPrepareParams, cancelToken?: CancellationToken): MaybePromise<TypeHierarchyItem[] | undefined>; supertypes(params: TypeHierarchySupertypesParams, cancelToken?: CancellationToken): MaybePromise<TypeHierarchyItem[] | undefined>; subtypes(params: TypeHierarchySubtypesParams, cancelToken?: CancellationToken): MaybePromise<TypeHierarchyItem[] | undefined>; } export abstract class AbstractTypeHierarchyProvider implements TypeHierarchyProvider { protected readonly grammarConfig: GrammarConfig; protected readonly nameProvider: NameProvider; protected readonly documents: LangiumDocuments; protected readonly references: References; constructor(services: LangiumServices) { this.grammarConfig = services.parser.GrammarConfig; this.nameProvider = services.references.NameProvider; this.documents = services.shared.workspace.LangiumDocuments; this.references = services.references.References; } prepareTypeHierarchy(document: LangiumDocument, params: TypeHierarchyPrepareParams, _cancelToken?: CancellationToken): MaybePromise<TypeHierarchyItem[] | undefined> { const rootNode = document.parseResult.value; const targetNode = findDeclarationNodeAtOffset( rootNode.$cstNode, document.textDocument.offsetAt(params.position), this.grammarConfig.nameRegexp, ); if (!targetNode) { return undefined; } const declarationNodes = this.references.findDeclarationNodes(targetNode); const items: TypeHierarchyItem[] = []; for (const declarationNode of declarationNodes) { items.push(...(this.getTypeHierarchyItems(declarationNode.astNode, document) ?? [])); } return items; } protected getTypeHierarchyItems(targetNode: AstNode, document: LangiumDocument): TypeHierarchyItem[] | undefined { const nameNode = this.nameProvider.getNameNode(targetNode); const name = this.nameProvider.getName(targetNode); if (!nameNode || !targetNode.$cstNode || name === undefined) { return undefined; } return [ { kind: SymbolKind.Class, name, range: targetNode.$cstNode.range, selectionRange: nameNode.range, uri: document.uri.toString(), ...this.getTypeHierarchyItem(targetNode), }, ]; } /** * Override this method to change default properties of the type hierarchy item or add additional ones like `tags` * or `details`. * * @example * // Change the node kind to SymbolKind.Interface * return { kind: SymbolKind.Interface } * * @see NodeKindProvider */ protected getTypeHierarchyItem(_targetNode: AstNode): Partial<TypeHierarchyItem> | undefined { return undefined; } async supertypes(params: TypeHierarchySupertypesParams, _cancelToken?: CancellationToken): Promise<TypeHierarchyItem[] | undefined> { const document = await this.documents.getOrCreateDocument(URI.parse(params.item.uri)); const rootNode = document.parseResult.value; const targetNode = findDeclarationNodeAtOffset( rootNode.$cstNode, document.textDocument.offsetAt(params.item.range.start), this.grammarConfig.nameRegexp, ); if (!targetNode) { return undefined; } return this.getSupertypes(targetNode.astNode); } /** * Override this method to collect the supertypes for your language. */ protected abstract getSupertypes(node: AstNode): MaybePromise<TypeHierarchyItem[] | undefined>; async subtypes(params: TypeHierarchySubtypesParams, _cancelToken?: CancellationToken): Promise<TypeHierarchyItem[] | undefined> { const document = await this.documents.getOrCreateDocument(URI.parse(params.item.uri)); const rootNode = document.parseResult.value; const targetNode = findDeclarationNodeAtOffset( rootNode.$cstNode, document.textDocument.offsetAt(params.item.range.start), this.grammarConfig.nameRegexp, ); if (!targetNode) { return undefined; } return this.getSubtypes(targetNode.astNode); } /** * Override this method to collect the subtypes for your language. */ protected abstract getSubtypes(node: AstNode): MaybePromise<TypeHierarchyItem[] | undefined>; }