/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ import { LanguageServiceDefaults } from './monaco.contribution'; import type { CSSWorker } from './cssWorker'; import * as lsTypes from 'vscode-languageserver-types'; import { languages, editor, IMarkdownString, Uri, Position, CancellationToken } from '../fillers/monaco-editor-core'; import { DiagnosticsAdapter, fromPosition, toRange, toTextEdit, fromRange, CompletionAdapter } from '../common/lspLanguageFeatures'; export interface WorkerAccessor { (first: Uri, ...more: Uri[]): Promise; } export class CSSDiagnosticsAdapter extends DiagnosticsAdapter { constructor(languageId: string, worker: WorkerAccessor, defaults: LanguageServiceDefaults) { super(languageId, worker, defaults.onDidChange); } } export class CSSCompletionAdapter extends CompletionAdapter { constructor(worker: WorkerAccessor) { super(worker, ['/', '-', ':']); } } function isMarkupContent(thing: any): thing is lsTypes.MarkupContent { return ( thing && typeof thing === 'object' && typeof (thing).kind === 'string' ); } function toMarkdownString(entry: lsTypes.MarkupContent | lsTypes.MarkedString): IMarkdownString { if (typeof entry === 'string') { return { value: entry }; } if (isMarkupContent(entry)) { if (entry.kind === 'plaintext') { return { value: entry.value.replace(/[\\`*_{}[\]()#+\-.!]/g, '\\$&') }; } return { value: entry.value }; } return { value: '```' + entry.language + '\n' + entry.value + '\n```\n' }; } function toMarkedStringArray( contents: lsTypes.MarkupContent | lsTypes.MarkedString | lsTypes.MarkedString[] ): IMarkdownString[] | undefined { if (!contents) { return void 0; } if (Array.isArray(contents)) { return contents.map(toMarkdownString); } return [toMarkdownString(contents)]; } // --- hover ------ export class HoverAdapter implements languages.HoverProvider { constructor(private _worker: WorkerAccessor) {} provideHover( model: editor.IReadOnlyModel, position: Position, token: CancellationToken ): Promise { let resource = model.uri; return this._worker(resource) .then((worker) => { return worker.doHover(resource.toString(), fromPosition(position)); }) .then((info) => { if (!info) { return; } return { range: toRange(info.range), contents: toMarkedStringArray(info.contents) }; }); } } // --- document highlights ------ function toDocumentHighlightKind( kind: lsTypes.DocumentHighlightKind ): languages.DocumentHighlightKind { switch (kind) { case lsTypes.DocumentHighlightKind.Read: return languages.DocumentHighlightKind.Read; case lsTypes.DocumentHighlightKind.Write: return languages.DocumentHighlightKind.Write; case lsTypes.DocumentHighlightKind.Text: return languages.DocumentHighlightKind.Text; } return languages.DocumentHighlightKind.Text; } export class DocumentHighlightAdapter implements languages.DocumentHighlightProvider { constructor(private _worker: WorkerAccessor) {} public provideDocumentHighlights( model: editor.IReadOnlyModel, position: Position, token: CancellationToken ): Promise { const resource = model.uri; return this._worker(resource) .then((worker) => worker.findDocumentHighlights(resource.toString(), fromPosition(position))) .then((entries) => { if (!entries) { return; } return entries.map((entry) => { return { range: toRange(entry.range), kind: toDocumentHighlightKind(entry.kind) }; }); }); } } // --- definition ------ function toLocation(location: lsTypes.Location): languages.Location { return { uri: Uri.parse(location.uri), range: toRange(location.range) }; } export class DefinitionAdapter { constructor(private _worker: WorkerAccessor) {} public provideDefinition( model: editor.IReadOnlyModel, position: Position, token: CancellationToken ): Promise { const resource = model.uri; return this._worker(resource) .then((worker) => { return worker.findDefinition(resource.toString(), fromPosition(position)); }) .then((definition) => { if (!definition) { return; } return [toLocation(definition)]; }); } } // --- references ------ export class ReferenceAdapter implements languages.ReferenceProvider { constructor(private _worker: WorkerAccessor) {} provideReferences( model: editor.IReadOnlyModel, position: Position, context: languages.ReferenceContext, token: CancellationToken ): Promise { const resource = model.uri; return this._worker(resource) .then((worker) => { return worker.findReferences(resource.toString(), fromPosition(position)); }) .then((entries) => { if (!entries) { return; } return entries.map(toLocation); }); } } // --- rename ------ function toWorkspaceEdit(edit: lsTypes.WorkspaceEdit): languages.WorkspaceEdit { if (!edit || !edit.changes) { return void 0; } let resourceEdits: languages.WorkspaceTextEdit[] = []; for (let uri in edit.changes) { const _uri = Uri.parse(uri); for (let e of edit.changes[uri]) { resourceEdits.push({ resource: _uri, edit: { range: toRange(e.range), text: e.newText } }); } } return { edits: resourceEdits }; } export class RenameAdapter implements languages.RenameProvider { constructor(private _worker: WorkerAccessor) {} provideRenameEdits( model: editor.IReadOnlyModel, position: Position, newName: string, token: CancellationToken ): Promise { const resource = model.uri; return this._worker(resource) .then((worker) => { return worker.doRename(resource.toString(), fromPosition(position), newName); }) .then((edit) => { return toWorkspaceEdit(edit); }); } } // --- document symbols ------ function toSymbolKind(kind: lsTypes.SymbolKind): languages.SymbolKind { let mKind = languages.SymbolKind; switch (kind) { case lsTypes.SymbolKind.File: return mKind.Array; case lsTypes.SymbolKind.Module: return mKind.Module; case lsTypes.SymbolKind.Namespace: return mKind.Namespace; case lsTypes.SymbolKind.Package: return mKind.Package; case lsTypes.SymbolKind.Class: return mKind.Class; case lsTypes.SymbolKind.Method: return mKind.Method; case lsTypes.SymbolKind.Property: return mKind.Property; case lsTypes.SymbolKind.Field: return mKind.Field; case lsTypes.SymbolKind.Constructor: return mKind.Constructor; case lsTypes.SymbolKind.Enum: return mKind.Enum; case lsTypes.SymbolKind.Interface: return mKind.Interface; case lsTypes.SymbolKind.Function: return mKind.Function; case lsTypes.SymbolKind.Variable: return mKind.Variable; case lsTypes.SymbolKind.Constant: return mKind.Constant; case lsTypes.SymbolKind.String: return mKind.String; case lsTypes.SymbolKind.Number: return mKind.Number; case lsTypes.SymbolKind.Boolean: return mKind.Boolean; case lsTypes.SymbolKind.Array: return mKind.Array; } return mKind.Function; } export class DocumentSymbolAdapter implements languages.DocumentSymbolProvider { constructor(private _worker: WorkerAccessor) {} public provideDocumentSymbols( model: editor.IReadOnlyModel, token: CancellationToken ): Promise { const resource = model.uri; return this._worker(resource) .then((worker) => worker.findDocumentSymbols(resource.toString())) .then((items) => { if (!items) { return; } return items.map((item) => ({ name: item.name, detail: '', containerName: item.containerName, kind: toSymbolKind(item.kind), range: toRange(item.location.range), selectionRange: toRange(item.location.range), tags: [] })); }); } } export class DocumentColorAdapter implements languages.DocumentColorProvider { constructor(private _worker: WorkerAccessor) {} public provideDocumentColors( model: editor.IReadOnlyModel, token: CancellationToken ): Promise { const resource = model.uri; return this._worker(resource) .then((worker) => worker.findDocumentColors(resource.toString())) .then((infos) => { if (!infos) { return; } return infos.map((item) => ({ color: item.color, range: toRange(item.range) })); }); } public provideColorPresentations( model: editor.IReadOnlyModel, info: languages.IColorInformation, token: CancellationToken ): Promise { const resource = model.uri; return this._worker(resource) .then((worker) => worker.getColorPresentations(resource.toString(), info.color, fromRange(info.range)) ) .then((presentations) => { if (!presentations) { return; } return presentations.map((presentation) => { let item: languages.IColorPresentation = { label: presentation.label }; if (presentation.textEdit) { item.textEdit = toTextEdit(presentation.textEdit); } if (presentation.additionalTextEdits) { item.additionalTextEdits = presentation.additionalTextEdits.map(toTextEdit); } return item; }); }); } } export class FoldingRangeAdapter implements languages.FoldingRangeProvider { constructor(private _worker: WorkerAccessor) {} public provideFoldingRanges( model: editor.IReadOnlyModel, context: languages.FoldingContext, token: CancellationToken ): Promise { const resource = model.uri; return this._worker(resource) .then((worker) => worker.getFoldingRanges(resource.toString(), context)) .then((ranges) => { if (!ranges) { return; } return ranges.map((range) => { const result: languages.FoldingRange = { start: range.startLine + 1, end: range.endLine + 1 }; if (typeof range.kind !== 'undefined') { result.kind = toFoldingRangeKind(range.kind); } return result; }); }); } } function toFoldingRangeKind( kind: lsTypes.FoldingRangeKind ): languages.FoldingRangeKind | undefined { switch (kind) { case lsTypes.FoldingRangeKind.Comment: return languages.FoldingRangeKind.Comment; case lsTypes.FoldingRangeKind.Imports: return languages.FoldingRangeKind.Imports; case lsTypes.FoldingRangeKind.Region: return languages.FoldingRangeKind.Region; } return void 0; } export class SelectionRangeAdapter implements languages.SelectionRangeProvider { constructor(private _worker: WorkerAccessor) {} public provideSelectionRanges( model: editor.IReadOnlyModel, positions: Position[], token: CancellationToken ): Promise { const resource = model.uri; return this._worker(resource) .then((worker) => worker.getSelectionRanges( resource.toString(), positions.map(fromPosition) ) ) .then((selectionRanges) => { if (!selectionRanges) { return; } return selectionRanges.map((selectionRange: lsTypes.SelectionRange | undefined) => { const result: languages.SelectionRange[] = []; while (selectionRange) { result.push({ range: toRange(selectionRange.range) }); selectionRange = selectionRange.parent; } return result; }); }); } }