langium
Version:
A language engineering tool for the Language Server Protocol
119 lines (104 loc) • 5.01 kB
text/typescript
/******************************************************************************
* 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 { Hover, HoverParams } from 'vscode-languageserver';
import type { CancellationToken } from '../utils/cancellation.js';
import type { GrammarConfig } from '../languages/grammar-config.js';
import type { References } from '../references/references.js';
import type { LangiumServices } from './lsp-services.js';
import type { AstNode } from '../syntax-tree.js';
import type { MaybePromise } from '../utils/promise-utils.js';
import type { LangiumDocument } from '../workspace/documents.js';
import type { DocumentationProvider } from '../documentation/documentation-provider.js';
import { findCommentNode, findDeclarationNodeAtOffset } from '../utils/cst-utils.js';
import { isKeyword } from '../languages/generated/ast.js';
import { isJSDoc, parseJSDoc } from '../documentation/jsdoc.js';
import { isAstNodeWithComment } from '../serializer/json-serializer.js';
/**
* Language-specific service for handling hover requests.
*/
export interface HoverProvider {
/**
* Handle a hover request.
*
* @throws `OperationCancelled` if cancellation is detected during execution
* @throws `ResponseError` if an error is detected that should be sent as response to the client
*/
getHoverContent(document: LangiumDocument, params: HoverParams, cancelToken?: CancellationToken): MaybePromise<Hover | undefined>;
}
export abstract class AstNodeHoverProvider implements HoverProvider {
protected readonly references: References;
protected readonly grammarConfig: GrammarConfig;
protected readonly languageId: string;
constructor(services: LangiumServices) {
this.references = services.references.References;
this.grammarConfig = services.parser.GrammarConfig;
this.languageId = services.LanguageMetaData.languageId;
}
async getHoverContent(document: LangiumDocument, params: HoverParams): Promise<Hover | undefined> {
const rootNode = document.parseResult?.value?.$cstNode;
if (rootNode) {
const offset = document.textDocument.offsetAt(params.position);
const cstNode = findDeclarationNodeAtOffset(rootNode, offset, this.grammarConfig.nameRegexp);
if (cstNode && cstNode.offset + cstNode.length > offset) {
const contents: string[] = [];
const targetNodes = this.references.findDeclarations(cstNode);
for (const targetNode of targetNodes) {
const content = await this.getAstNodeHoverContent(targetNode);
if (typeof content === 'string') {
contents.push(content);
}
}
if (contents.length > 0) {
return {
contents: {
kind: 'markdown',
language: this.languageId,
value: contents.join(' ')
}
};
}
// Add support for documentation on keywords
if (isKeyword(cstNode.grammarSource)) {
return this.getKeywordHoverContent(cstNode.grammarSource);
}
}
}
return undefined;
}
protected abstract getAstNodeHoverContent(node: AstNode): MaybePromise<string | undefined>;
protected getKeywordHoverContent(node: AstNode): MaybePromise<Hover | undefined> {
let comment = isAstNodeWithComment(node) ? node.$comment : undefined;
if (!comment) {
comment = findCommentNode(node.$cstNode, ['ML_COMMENT'])?.text;
}
if (comment && isJSDoc(comment)) {
const content = parseJSDoc(comment).toMarkdown();
if (content) {
return {
contents: {
kind: 'markdown',
value: content
}
};
}
}
return undefined;
}
}
export class MultilineCommentHoverProvider extends AstNodeHoverProvider {
protected readonly documentationProvider: DocumentationProvider;
constructor(services: LangiumServices) {
super(services);
this.documentationProvider = services.documentation.DocumentationProvider;
}
protected getAstNodeHoverContent(node: AstNode): MaybePromise<string | undefined> {
const content = this.documentationProvider.getDocumentation(node);
if (content) {
return content;
}
return undefined;
}
}