/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ 'use strict'; import * as modes from 'vs/editor/common/modes'; import * as types from './extHostTypes'; import * as search from 'vs/workbench/parts/search/common/search'; import { ITextEditorOptions } from 'vs/platform/editor/common/editor'; import { EditorViewColumn } from 'vs/workbench/api/shared/editor'; import { IDecorationOptions } from 'vs/editor/common/editorCommon'; import { EndOfLineSequence } from 'vs/editor/common/model'; import * as vscode from 'vscode'; import { URI } from 'vs/base/common/uri'; import { ProgressLocation as MainProgressLocation } from 'vs/workbench/services/progress/common/progress'; import { SaveReason } from 'vs/workbench/services/textfile/common/textfiles'; import { IPosition } from 'vs/editor/common/core/position'; import { IRange } from 'vs/editor/common/core/range'; import { ISelection } from 'vs/editor/common/core/selection'; import * as htmlContent from 'vs/base/common/htmlContent'; import * as languageSelector from 'vs/editor/common/modes/languageSelector'; import { WorkspaceEditDto, ResourceTextEditDto, ResourceFileEditDto } from 'vs/workbench/api/node/extHost.protocol'; import { MarkerSeverity, IRelatedInformation, IMarkerData, MarkerTag } from 'vs/platform/markers/common/markers'; import { ACTIVE_GROUP, SIDE_GROUP } from 'vs/workbench/services/editor/common/editorService'; import { ExtHostDocumentsAndEditors } from 'vs/workbench/api/node/extHostDocumentsAndEditors'; import { isString, isNumber } from 'vs/base/common/types'; export interface PositionLike { line: number; character: number; } export interface RangeLike { start: PositionLike; end: PositionLike; } export interface SelectionLike extends RangeLike { anchor: PositionLike; active: PositionLike; } export namespace Selection { export function to(selection: ISelection): types.Selection { let { selectionStartLineNumber, selectionStartColumn, positionLineNumber, positionColumn } = selection; let start = new types.Position(selectionStartLineNumber - 1, selectionStartColumn - 1); let end = new types.Position(positionLineNumber - 1, positionColumn - 1); return new types.Selection(start, end); } export function from(selection: SelectionLike): ISelection { let { anchor, active } = selection; return { selectionStartLineNumber: anchor.line + 1, selectionStartColumn: anchor.character + 1, positionLineNumber: active.line + 1, positionColumn: active.character + 1 }; } } export namespace Range { export function from(range: RangeLike): IRange { if (!range) { return undefined; } let { start, end } = range; return { startLineNumber: start.line + 1, startColumn: start.character + 1, endLineNumber: end.line + 1, endColumn: end.character + 1 }; } export function to(range: IRange): types.Range { if (!range) { return undefined; } let { startLineNumber, startColumn, endLineNumber, endColumn } = range; return new types.Range(startLineNumber - 1, startColumn - 1, endLineNumber - 1, endColumn - 1); } } export namespace Position { export function to(position: IPosition): types.Position { return new types.Position(position.lineNumber - 1, position.column - 1); } export function from(position: types.Position): IPosition { return { lineNumber: position.line + 1, column: position.character + 1 }; } } export namespace DiagnosticTag { export function from(value: vscode.DiagnosticTag): MarkerTag { switch (value) { case types.DiagnosticTag.Unnecessary: return MarkerTag.Unnecessary; } return undefined; } } export namespace Diagnostic { export function from(value: vscode.Diagnostic): IMarkerData { return { ...Range.from(value.range), message: value.message, source: value.source, code: isString(value.code) || isNumber(value.code) ? String(value.code) : void 0, severity: DiagnosticSeverity.from(value.severity), relatedInformation: value.relatedInformation && value.relatedInformation.map(DiagnosticRelatedInformation.from), tags: Array.isArray(value.tags) ? value.tags.map(DiagnosticTag.from) : undefined, }; } } export namespace DiagnosticRelatedInformation { export function from(value: types.DiagnosticRelatedInformation): IRelatedInformation { return { ...Range.from(value.location.range), message: value.message, resource: value.location.uri }; } export function to(value: IRelatedInformation): types.DiagnosticRelatedInformation { return new types.DiagnosticRelatedInformation(new types.Location(value.resource, Range.to(value)), value.message); } } export namespace DiagnosticSeverity { export function from(value: number): MarkerSeverity { switch (value) { case types.DiagnosticSeverity.Error: return MarkerSeverity.Error; case types.DiagnosticSeverity.Warning: return MarkerSeverity.Warning; case types.DiagnosticSeverity.Information: return MarkerSeverity.Info; case types.DiagnosticSeverity.Hint: return MarkerSeverity.Hint; } return MarkerSeverity.Error; } export function to(value: MarkerSeverity): types.DiagnosticSeverity { switch (value) { case MarkerSeverity.Info: return types.DiagnosticSeverity.Information; case MarkerSeverity.Warning: return types.DiagnosticSeverity.Warning; case MarkerSeverity.Error: return types.DiagnosticSeverity.Error; case MarkerSeverity.Hint: return types.DiagnosticSeverity.Hint; } return types.DiagnosticSeverity.Error; } } export namespace ViewColumn { export function from(column?: vscode.ViewColumn): EditorViewColumn { if (typeof column === 'number' && column >= types.ViewColumn.One) { return column - 1; // adjust zero index (ViewColumn.ONE => 0) } if (column === types.ViewColumn.Beside) { return SIDE_GROUP; } return ACTIVE_GROUP; // default is always the active group } export function to(position?: EditorViewColumn): vscode.ViewColumn { if (typeof position === 'number' && position >= 0) { return position + 1; // adjust to index (ViewColumn.ONE => 1) } return undefined; } } function isDecorationOptions(something: any): something is vscode.DecorationOptions { return (typeof something.range !== 'undefined'); } export function isDecorationOptionsArr(something: vscode.Range[] | vscode.DecorationOptions[]): something is vscode.DecorationOptions[] { if (something.length === 0) { return true; } return isDecorationOptions(something[0]) ? true : false; } export namespace MarkdownString { export function fromMany(markup: (vscode.MarkdownString | vscode.MarkedString)[]): htmlContent.IMarkdownString[] { return markup.map(MarkdownString.from); } interface Codeblock { language: string; value: string; } function isCodeblock(thing: any): thing is Codeblock { return thing && typeof thing === 'object' && typeof (thing).language === 'string' && typeof (thing).value === 'string'; } export function from(markup: vscode.MarkdownString | vscode.MarkedString): htmlContent.IMarkdownString { if (isCodeblock(markup)) { const { language, value } = markup; return { value: '```' + language + '\n' + value + '\n```\n' }; } else if (htmlContent.isMarkdownString(markup)) { return markup; } else if (typeof markup === 'string') { return { value: markup }; } else { return { value: '' }; } } export function to(value: htmlContent.IMarkdownString): vscode.MarkdownString { const ret = new htmlContent.MarkdownString(value.value); ret.isTrusted = value.isTrusted; return ret; } export function fromStrict(value: string | types.MarkdownString): undefined | string | htmlContent.IMarkdownString { if (!value) { return undefined; } return typeof value === 'string' ? value : MarkdownString.from(value); } } export function fromRangeOrRangeWithMessage(ranges: vscode.Range[] | vscode.DecorationOptions[]): IDecorationOptions[] { if (isDecorationOptionsArr(ranges)) { return ranges.map(r => { return { range: Range.from(r.range), hoverMessage: Array.isArray(r.hoverMessage) ? MarkdownString.fromMany(r.hoverMessage) : r.hoverMessage && MarkdownString.from(r.hoverMessage), renderOptions: /* URI vs Uri */r.renderOptions }; }); } else { return ranges.map((r): IDecorationOptions => { return { range: Range.from(r) }; }); } } export const TextEdit = { from(edit: vscode.TextEdit): modes.TextEdit { return { text: edit.newText, eol: EndOfLine.from(edit.newEol), range: Range.from(edit.range) }; }, to(edit: modes.TextEdit): types.TextEdit { let result = new types.TextEdit(Range.to(edit.range), edit.text); result.newEol = EndOfLine.to(edit.eol); return result; } }; export namespace WorkspaceEdit { export function from(value: vscode.WorkspaceEdit, documents?: ExtHostDocumentsAndEditors): WorkspaceEditDto { const result: WorkspaceEditDto = { edits: [] }; for (const entry of (value as types.WorkspaceEdit)._allEntries()) { const [uri, uriOrEdits] = entry; if (Array.isArray(uriOrEdits)) { // text edits let doc = documents ? documents.getDocument(uri.toString()) : undefined; result.edits.push({ resource: uri, modelVersionId: doc && doc.version, edits: uriOrEdits.map(TextEdit.from) }); } else { // resource edits result.edits.push({ oldUri: uri, newUri: uriOrEdits, options: entry[2] }); } } return result; } export function to(value: WorkspaceEditDto) { const result = new types.WorkspaceEdit(); for (const edit of value.edits) { if (Array.isArray((edit).edits)) { result.set( URI.revive((edit).resource), (edit).edits.map(TextEdit.to) ); } else { result.renameFile( URI.revive((edit).oldUri), URI.revive((edit).newUri), (edit).options ); } } return result; } } export namespace SymbolKind { const _fromMapping: { [kind: number]: modes.SymbolKind } = Object.create(null); _fromMapping[types.SymbolKind.File] = modes.SymbolKind.File; _fromMapping[types.SymbolKind.Module] = modes.SymbolKind.Module; _fromMapping[types.SymbolKind.Namespace] = modes.SymbolKind.Namespace; _fromMapping[types.SymbolKind.Package] = modes.SymbolKind.Package; _fromMapping[types.SymbolKind.Class] = modes.SymbolKind.Class; _fromMapping[types.SymbolKind.Method] = modes.SymbolKind.Method; _fromMapping[types.SymbolKind.Property] = modes.SymbolKind.Property; _fromMapping[types.SymbolKind.Field] = modes.SymbolKind.Field; _fromMapping[types.SymbolKind.Constructor] = modes.SymbolKind.Constructor; _fromMapping[types.SymbolKind.Enum] = modes.SymbolKind.Enum; _fromMapping[types.SymbolKind.Interface] = modes.SymbolKind.Interface; _fromMapping[types.SymbolKind.Function] = modes.SymbolKind.Function; _fromMapping[types.SymbolKind.Variable] = modes.SymbolKind.Variable; _fromMapping[types.SymbolKind.Constant] = modes.SymbolKind.Constant; _fromMapping[types.SymbolKind.String] = modes.SymbolKind.String; _fromMapping[types.SymbolKind.Number] = modes.SymbolKind.Number; _fromMapping[types.SymbolKind.Boolean] = modes.SymbolKind.Boolean; _fromMapping[types.SymbolKind.Array] = modes.SymbolKind.Array; _fromMapping[types.SymbolKind.Object] = modes.SymbolKind.Object; _fromMapping[types.SymbolKind.Key] = modes.SymbolKind.Key; _fromMapping[types.SymbolKind.Null] = modes.SymbolKind.Null; _fromMapping[types.SymbolKind.EnumMember] = modes.SymbolKind.EnumMember; _fromMapping[types.SymbolKind.Struct] = modes.SymbolKind.Struct; _fromMapping[types.SymbolKind.Event] = modes.SymbolKind.Event; _fromMapping[types.SymbolKind.Operator] = modes.SymbolKind.Operator; _fromMapping[types.SymbolKind.TypeParameter] = modes.SymbolKind.TypeParameter; export function from(kind: vscode.SymbolKind): modes.SymbolKind { return _fromMapping[kind] || modes.SymbolKind.Property; } export function to(kind: modes.SymbolKind): vscode.SymbolKind { for (let k in _fromMapping) { if (_fromMapping[k] === kind) { return Number(k); } } return types.SymbolKind.Property; } } export namespace WorkspaceSymbol { export function from(info: vscode.SymbolInformation): search.IWorkspaceSymbol { return { name: info.name, kind: SymbolKind.from(info.kind), containerName: info.containerName, location: location.from(info.location) }; } export function to(info: search.IWorkspaceSymbol): types.SymbolInformation { return new types.SymbolInformation( info.name, SymbolKind.to(info.kind), info.containerName, location.to(info.location) ); } } export namespace DocumentSymbol { export function from(info: vscode.DocumentSymbol): modes.DocumentSymbol { let result: modes.DocumentSymbol = { name: info.name, detail: info.detail, range: Range.from(info.range), selectionRange: Range.from(info.selectionRange), kind: SymbolKind.from(info.kind) }; if (info.children) { result.children = info.children.map(from); } return result; } export function to(info: modes.DocumentSymbol): vscode.DocumentSymbol { let result = new types.DocumentSymbol( info.name, info.detail, SymbolKind.to(info.kind), Range.to(info.range), Range.to(info.selectionRange), ); if (info.children) { result.children = info.children.map(to) as any; } return result; } } export const location = { from(value: vscode.Location): modes.Location { return { range: value.range && Range.from(value.range), uri: value.uri }; }, to(value: modes.Location): types.Location { return new types.Location(value.uri, Range.to(value.range)); } }; export namespace DefinitionLink { export function from(value: vscode.Location | vscode.DefinitionLink): modes.DefinitionLink { const definitionLink = value; const location = value; return { origin: definitionLink.originSelectionRange ? Range.from(definitionLink.originSelectionRange) : undefined, uri: definitionLink.targetUri ? definitionLink.targetUri : location.uri, range: Range.from(definitionLink.targetRange ? definitionLink.targetRange : location.range), selectionRange: definitionLink.targetSelectionRange ? Range.from(definitionLink.targetSelectionRange) : undefined, }; } } export namespace Hover { export function from(hover: vscode.Hover): modes.Hover { return { range: Range.from(hover.range), contents: MarkdownString.fromMany(hover.contents) }; } export function to(info: modes.Hover): types.Hover { return new types.Hover(info.contents.map(MarkdownString.to), Range.to(info.range)); } } export namespace DocumentHighlight { export function from(documentHighlight: vscode.DocumentHighlight): modes.DocumentHighlight { return { range: Range.from(documentHighlight.range), kind: documentHighlight.kind }; } export function to(occurrence: modes.DocumentHighlight): types.DocumentHighlight { return new types.DocumentHighlight(Range.to(occurrence.range), occurrence.kind); } } export namespace CompletionTriggerKind { export function from(kind: modes.CompletionTriggerKind) { switch (kind) { case modes.CompletionTriggerKind.TriggerCharacter: return types.CompletionTriggerKind.TriggerCharacter; case modes.CompletionTriggerKind.TriggerForIncompleteCompletions: return types.CompletionTriggerKind.TriggerForIncompleteCompletions; case modes.CompletionTriggerKind.Invoke: default: return types.CompletionTriggerKind.Invoke; } } } export namespace CompletionContext { export function from(context: modes.CompletionContext): types.CompletionContext { return { triggerKind: CompletionTriggerKind.from(context.triggerKind), triggerCharacter: context.triggerCharacter }; } } export const CompletionItemKind = { from(kind: types.CompletionItemKind): modes.CompletionKind { switch (kind) { case types.CompletionItemKind.Method: return modes.CompletionKind.Method; case types.CompletionItemKind.Function: return modes.CompletionKind.Function; case types.CompletionItemKind.Constructor: return modes.CompletionKind.Constructor; case types.CompletionItemKind.Field: return modes.CompletionKind.Field; case types.CompletionItemKind.Variable: return modes.CompletionKind.Variable; case types.CompletionItemKind.Class: return modes.CompletionKind.Class; case types.CompletionItemKind.Interface: return modes.CompletionKind.Interface; case types.CompletionItemKind.Struct: return modes.CompletionKind.Struct; case types.CompletionItemKind.Module: return modes.CompletionKind.Module; case types.CompletionItemKind.Property: return modes.CompletionKind.Property; case types.CompletionItemKind.Unit: return modes.CompletionKind.Unit; case types.CompletionItemKind.Value: return modes.CompletionKind.Value; case types.CompletionItemKind.Constant: return modes.CompletionKind.Constant; case types.CompletionItemKind.Enum: return modes.CompletionKind.Enum; case types.CompletionItemKind.EnumMember: return modes.CompletionKind.EnumMember; case types.CompletionItemKind.Keyword: return modes.CompletionKind.Keyword; case types.CompletionItemKind.Snippet: return modes.CompletionKind.Snippet; case types.CompletionItemKind.Text: return modes.CompletionKind.Text; case types.CompletionItemKind.Color: return modes.CompletionKind.Color; case types.CompletionItemKind.File: return modes.CompletionKind.File; case types.CompletionItemKind.Reference: return modes.CompletionKind.Reference; case types.CompletionItemKind.Folder: return modes.CompletionKind.Folder; case types.CompletionItemKind.Event: return modes.CompletionKind.Event; case types.CompletionItemKind.Operator: return modes.CompletionKind.Operator; case types.CompletionItemKind.TypeParameter: return modes.CompletionKind.TypeParameter; } return modes.CompletionKind.Property; }, to(kind: modes.CompletionKind): types.CompletionItemKind { switch (kind) { case modes.CompletionKind.Method: return types.CompletionItemKind.Method; case modes.CompletionKind.Function: return types.CompletionItemKind.Function; case modes.CompletionKind.Constructor: return types.CompletionItemKind.Constructor; case modes.CompletionKind.Field: return types.CompletionItemKind.Field; case modes.CompletionKind.Variable: return types.CompletionItemKind.Variable; case modes.CompletionKind.Class: return types.CompletionItemKind.Class; case modes.CompletionKind.Interface: return types.CompletionItemKind.Interface; case modes.CompletionKind.Struct: return types.CompletionItemKind.Struct; case modes.CompletionKind.Module: return types.CompletionItemKind.Module; case modes.CompletionKind.Property: return types.CompletionItemKind.Property; case modes.CompletionKind.Unit: return types.CompletionItemKind.Unit; case modes.CompletionKind.Value: return types.CompletionItemKind.Value; case modes.CompletionKind.Constant: return types.CompletionItemKind.Constant; case modes.CompletionKind.Enum: return types.CompletionItemKind.Enum; case modes.CompletionKind.EnumMember: return types.CompletionItemKind.EnumMember; case modes.CompletionKind.Keyword: return types.CompletionItemKind.Keyword; case modes.CompletionKind.Snippet: return types.CompletionItemKind.Snippet; case modes.CompletionKind.Text: return types.CompletionItemKind.Text; case modes.CompletionKind.Color: return types.CompletionItemKind.Color; case modes.CompletionKind.File: return types.CompletionItemKind.File; case modes.CompletionKind.Reference: return types.CompletionItemKind.Reference; case modes.CompletionKind.Folder: return types.CompletionItemKind.Folder; case modes.CompletionKind.Event: return types.CompletionItemKind.Event; case modes.CompletionKind.Operator: return types.CompletionItemKind.Operator; case modes.CompletionKind.TypeParameter: return types.CompletionItemKind.TypeParameter; } return types.CompletionItemKind.Property; } }; export namespace Suggest { export function to(suggestion: modes.CompletionItem): types.CompletionItem { const result = new types.CompletionItem(suggestion.label); result.insertText = suggestion.insertText; result.kind = CompletionItemKind.to(suggestion.kind); result.detail = suggestion.detail; result.documentation = htmlContent.isMarkdownString(suggestion.documentation) ? MarkdownString.to(suggestion.documentation) : suggestion.documentation; result.sortText = suggestion.sortText; result.filterText = suggestion.filterText; result.preselect = suggestion.preselect; result.commitCharacters = suggestion.commitCharacters; result.range = Range.to(suggestion.range); // 'inserText'-logic if (suggestion.insertTextIsSnippet) { result.insertText = new types.SnippetString(suggestion.insertText); } else { result.insertText = suggestion.insertText; result.textEdit = new types.TextEdit(result.range, result.insertText); } // TODO additionalEdits, command return result; } } export namespace ParameterInformation { export function from(info: types.ParameterInformation): modes.ParameterInformation { return { label: info.label, documentation: MarkdownString.fromStrict(info.documentation) }; } export function to(info: modes.ParameterInformation): types.ParameterInformation { return { label: info.label, documentation: htmlContent.isMarkdownString(info.documentation) ? MarkdownString.to(info.documentation) : info.documentation }; } } export namespace SignatureInformation { export function from(info: types.SignatureInformation): modes.SignatureInformation { return { label: info.label, documentation: MarkdownString.fromStrict(info.documentation), parameters: info.parameters && info.parameters.map(ParameterInformation.from) }; } export function to(info: modes.SignatureInformation): types.SignatureInformation { return { label: info.label, documentation: htmlContent.isMarkdownString(info.documentation) ? MarkdownString.to(info.documentation) : info.documentation, parameters: info.parameters && info.parameters.map(ParameterInformation.to) }; } } export namespace SignatureHelp { export function from(help: types.SignatureHelp): modes.SignatureHelp { return { activeSignature: help.activeSignature, activeParameter: help.activeParameter, signatures: help.signatures && help.signatures.map(SignatureInformation.from) }; } export function to(help: modes.SignatureHelp): types.SignatureHelp { return { activeSignature: help.activeSignature, activeParameter: help.activeParameter, signatures: help.signatures && help.signatures.map(SignatureInformation.to) }; } } export namespace DocumentLink { export function from(link: vscode.DocumentLink): modes.ILink { return { range: Range.from(link.range), url: link.target && link.target.toString() }; } export function to(link: modes.ILink): vscode.DocumentLink { return new types.DocumentLink(Range.to(link.range), link.url && URI.parse(link.url)); } } export namespace ColorPresentation { export function to(colorPresentation: modes.IColorPresentation): types.ColorPresentation { let cp = new types.ColorPresentation(colorPresentation.label); if (colorPresentation.textEdit) { cp.textEdit = TextEdit.to(colorPresentation.textEdit); } if (colorPresentation.additionalTextEdits) { cp.additionalTextEdits = colorPresentation.additionalTextEdits.map(value => TextEdit.to(value)); } return cp; } export function from(colorPresentation: vscode.ColorPresentation): modes.IColorPresentation { return { label: colorPresentation.label, textEdit: colorPresentation.textEdit ? TextEdit.from(colorPresentation.textEdit) : undefined, additionalTextEdits: colorPresentation.additionalTextEdits ? colorPresentation.additionalTextEdits.map(value => TextEdit.from(value)) : undefined }; } } export namespace Color { export function to(c: [number, number, number, number]): types.Color { return new types.Color(c[0], c[1], c[2], c[3]); } export function from(color: types.Color): [number, number, number, number] { return [color.red, color.green, color.blue, color.alpha]; } } export namespace TextDocumentSaveReason { export function to(reason: SaveReason): vscode.TextDocumentSaveReason { switch (reason) { case SaveReason.AUTO: return types.TextDocumentSaveReason.AfterDelay; case SaveReason.EXPLICIT: return types.TextDocumentSaveReason.Manual; case SaveReason.FOCUS_CHANGE: case SaveReason.WINDOW_CHANGE: return types.TextDocumentSaveReason.FocusOut; } } } export namespace EndOfLine { export function from(eol: vscode.EndOfLine): EndOfLineSequence { if (eol === types.EndOfLine.CRLF) { return EndOfLineSequence.CRLF; } else if (eol === types.EndOfLine.LF) { return EndOfLineSequence.LF; } return undefined; } export function to(eol: EndOfLineSequence): vscode.EndOfLine { if (eol === EndOfLineSequence.CRLF) { return types.EndOfLine.CRLF; } else if (eol === EndOfLineSequence.LF) { return types.EndOfLine.LF; } return undefined; } } export namespace ProgressLocation { export function from(loc: vscode.ProgressLocation): MainProgressLocation { switch (loc) { case types.ProgressLocation.SourceControl: return MainProgressLocation.Scm; case types.ProgressLocation.Window: return MainProgressLocation.Window; case types.ProgressLocation.Notification: return MainProgressLocation.Notification; } return undefined; } } export namespace FoldingRange { export function from(r: vscode.FoldingRange): modes.FoldingRange { let range: modes.FoldingRange = { start: r.start + 1, end: r.end + 1 }; if (r.kind) { range.kind = FoldingRangeKind.from(r.kind); } return range; } } export namespace FoldingRangeKind { export function from(kind: vscode.FoldingRangeKind | undefined): modes.FoldingRangeKind | undefined { if (kind) { switch (kind) { case types.FoldingRangeKind.Comment: return modes.FoldingRangeKind.Comment; case types.FoldingRangeKind.Imports: return modes.FoldingRangeKind.Imports; case types.FoldingRangeKind.Region: return modes.FoldingRangeKind.Region; } } return void 0; } } export namespace TextEditorOptions { export function from(options?: vscode.TextDocumentShowOptions): ITextEditorOptions { if (options) { return { pinned: typeof options.preview === 'boolean' ? !options.preview : undefined, preserveFocus: options.preserveFocus, selection: typeof options.selection === 'object' ? Range.from(options.selection) : undefined } as ITextEditorOptions; } return undefined; } } export namespace GlobPattern { export function from(pattern: vscode.GlobPattern): string | types.RelativePattern { if (pattern instanceof types.RelativePattern) { return pattern; } if (typeof pattern === 'string') { return pattern; } if (isRelativePattern(pattern)) { return new types.RelativePattern(pattern.base, pattern.pattern); } return pattern; // preserve `undefined` and `null` } function isRelativePattern(obj: any): obj is vscode.RelativePattern { const rp = obj as vscode.RelativePattern; return rp && typeof rp.base === 'string' && typeof rp.pattern === 'string'; } } export namespace LanguageSelector { export function from(selector: vscode.DocumentSelector): languageSelector.LanguageSelector { if (!selector) { return undefined; } else if (Array.isArray(selector)) { return selector.map(from); } else if (typeof selector === 'string') { return selector; } else { return { language: selector.language, scheme: selector.scheme, pattern: GlobPattern.from(selector.pattern), exclusive: selector.exclusive }; } } }