From 89386915a1460bbd7db56723e19afb3afb89bcc4 Mon Sep 17 00:00:00 2001 From: Tyler James Leonhardt Date: Mon, 5 Aug 2019 12:24:55 -0700 Subject: [PATCH 1/2] everything needed so far (#2118) --- src/features/DebugSession.ts | 2 +- src/features/DocumentFormatter.ts | 356 ------------------------------ src/main.ts | 2 - src/session.ts | 12 +- 4 files changed, 7 insertions(+), 365 deletions(-) delete mode 100644 src/features/DocumentFormatter.ts diff --git a/src/features/DebugSession.ts b/src/features/DebugSession.ts index 4c092f2f6e..f38dd885c4 100644 --- a/src/features/DebugSession.ts +++ b/src/features/DebugSession.ts @@ -570,7 +570,7 @@ export class PickRunspaceFeature implements IFeature { } private pickRunspace(processId): Thenable { - return this.languageClient.sendRequest(GetRunspaceRequestType, processId).then((response) => { + return this.languageClient.sendRequest(GetRunspaceRequestType, { processId }).then((response) => { const items: IRunspaceItem[] = []; for (const runspace of response) { diff --git a/src/features/DocumentFormatter.ts b/src/features/DocumentFormatter.ts deleted file mode 100644 index cb801fa5dd..0000000000 --- a/src/features/DocumentFormatter.ts +++ /dev/null @@ -1,356 +0,0 @@ -/*--------------------------------------------------------- - * Copyright (C) Microsoft Corporation. All rights reserved. - *--------------------------------------------------------*/ - -import * as path from "path"; -import vscode = require("vscode"); -import { - CancellationToken, - DocumentFormattingEditProvider, - DocumentRangeFormattingEditProvider, - FormattingOptions, - OnTypeFormattingEditProvider, - Position, - Range, - TextDocument, - TextEdit, - TextEditor, - TextLine, -} from "vscode"; -import { - DocumentFormattingRequest, - DocumentRangeFormattingParams, - DocumentRangeFormattingRequest, - DocumentSelector, - LanguageClient, - RequestType, -} from "vscode-languageclient"; -import { TextDocumentIdentifier } from "vscode-languageserver-types"; -import Window = vscode.window; -import * as AnimatedStatusBar from "../controls/animatedStatusBar"; -import { IFeature } from "../feature"; -import { Logger } from "../logging"; -import * as Settings from "../settings"; -import * as Utils from "../utils"; - -export const ScriptRegionRequestType = new RequestType("powerShell/getScriptRegion"); - -interface IScriptRegionRequestParams { - fileUri: string; - character: string; - line: number; - column: number; -} - -interface IScriptRegionRequestResult { - scriptRegion: IScriptRegion; -} - -interface IScriptRegion { - file: string; - text: string; - startLineNumber: number; - startColumnNumber: number; - startOffset: number; - endLineNumber: number; - endColumnNumber: number; - endOffset: number; -} - -function toRange(scriptRegion: IScriptRegion): vscode.Range { - return new vscode.Range( - scriptRegion.startLineNumber - 1, - scriptRegion.startColumnNumber - 1, - scriptRegion.endLineNumber - 1, - scriptRegion.endColumnNumber - 1); -} - -function toOneBasedPosition(position: Position): Position { - return position.translate({ lineDelta: 1, characterDelta: 1 }); -} - -class DocumentLocker { - // tslint:disable-next-line:ban-types - private lockedDocuments: Object; - - constructor() { - this.lockedDocuments = new Object(); - } - - public isLocked(document: TextDocument): boolean { - return this.isLockedInternal(this.getKey(document)); - } - - public lock(document: TextDocument, unlockWhenDone?: Thenable): void { - this.lockInternal(this.getKey(document), unlockWhenDone); - } - - public unlock(document: TextDocument): void { - this.unlockInternal(this.getKey(document)); - } - - public unlockAll(): void { - Object.keys(this.lockedDocuments).slice().forEach((documentKey) => this.unlockInternal(documentKey)); - } - - private getKey(document: TextDocument): string { - return document.uri.toString(); - } - - private lockInternal(documentKey: string, unlockWhenDone?: Thenable): void { - if (!this.isLockedInternal(documentKey)) { - this.lockedDocuments[documentKey] = true; - } - - if (unlockWhenDone !== undefined) { - unlockWhenDone.then(() => this.unlockInternal(documentKey)); - } - } - - private unlockInternal(documentKey: string): void { - if (this.isLockedInternal(documentKey)) { - delete this.lockedDocuments[documentKey]; - } - } - - private isLockedInternal(documentKey: string): boolean { - return this.lockedDocuments.hasOwnProperty(documentKey); - } -} - -class PSDocumentFormattingEditProvider implements - DocumentFormattingEditProvider, - DocumentRangeFormattingEditProvider, - OnTypeFormattingEditProvider { - - private static documentLocker = new DocumentLocker(); - private static statusBarTracker = new Object(); - - private static showStatusBar(document: TextDocument, hideWhenDone: Thenable): void { - const statusBar = - AnimatedStatusBar.showAnimatedStatusBarMessage("Formatting PowerShell document", hideWhenDone); - this.statusBarTracker[document.uri.toString()] = statusBar; - hideWhenDone.then(() => { - this.disposeStatusBar(document.uri.toString()); - }); - } - - private static disposeStatusBar(documentUri: string) { - if (this.statusBarTracker.hasOwnProperty(documentUri)) { - this.statusBarTracker[documentUri].dispose(); - delete this.statusBarTracker[documentUri]; - } - } - - private static disposeAllStatusBars() { - Object.keys(this.statusBarTracker).slice().forEach((key) => this.disposeStatusBar(key)); - } - - private languageClient: LanguageClient; - - private get emptyPromise(): Promise { - return Promise.resolve(TextEdit[0]); - } - - constructor(private logger: Logger) { - } - - public setLanguageClient(languageClient: LanguageClient): void { - this.languageClient = languageClient; - - // setLanguageClient is called while restarting a session, - // so this makes sure we clean up the document locker and - // any residual status bars - PSDocumentFormattingEditProvider.documentLocker.unlockAll(); - PSDocumentFormattingEditProvider.disposeAllStatusBars(); - } - - public provideDocumentFormattingEdits( - document: TextDocument, - options: FormattingOptions, - token: CancellationToken): TextEdit[] | Thenable { - - this.logger.writeVerbose(`Formatting entire document - ${document.uri}...`); - return this.sendDocumentFormatRequest(document, null, options, token); - } - - public provideDocumentRangeFormattingEdits( - document: TextDocument, - range: Range, - options: FormattingOptions, - token: CancellationToken): TextEdit[] | Thenable { - - this.logger.writeVerbose(`Formatting document range ${JSON.stringify(range)} - ${document.uri}...`); - return this.sendDocumentFormatRequest(document, range, options, token); - } - - public provideOnTypeFormattingEdits( - document: TextDocument, - position: Position, - ch: string, - options: FormattingOptions, - token: CancellationToken): TextEdit[] | Thenable { - - this.logger.writeVerbose(`Formatting on type at position ${JSON.stringify(position)} - ${document.uri}...`); - - return this.getScriptRegion(document, position, ch).then((scriptRegion) => { - if (scriptRegion === null) { - this.logger.writeVerbose("No formattable range returned."); - return this.emptyPromise; - } - - return this.sendDocumentFormatRequest( - document, - toRange(scriptRegion), - options, - token); - }, - (err) => { - this.logger.writeVerbose(`Error while requesting script region for formatting: ${err}`); - }); - } - - private sendDocumentFormatRequest( - document: TextDocument, - range: Range, - options: FormattingOptions, - token: CancellationToken): TextEdit[] | Thenable { - - const editor: TextEditor = this.getEditor(document); - if (editor === undefined) { - return this.emptyPromise; - } - - // Check if the document is already being formatted. - // If so, then ignore the formatting request. - if (this.isDocumentLocked(document)) { - return this.emptyPromise; - } - - // somehow range object gets serialized to an array of Position objects, - // so we need to use the object literal syntax to initialize it. - let rangeParam = null; - if (range != null) { - rangeParam = { - start: { - line: range.start.line, - character: range.start.character, - }, - end: { - line: range.end.line, - character: range.end.character, - }, - }; - } - - const requestParams: DocumentRangeFormattingParams = { - textDocument: TextDocumentIdentifier.create(document.uri.toString()), - range: rangeParam, - options: this.getEditorSettings(editor), - }; - - const formattingStartTime = new Date().valueOf(); - function getFormattingDuration() { - return ((new Date().valueOf()) - formattingStartTime) / 1000; - } - - const textEdits = this.languageClient.sendRequest( - DocumentRangeFormattingRequest.type, - requestParams); - this.lockDocument(document, textEdits); - PSDocumentFormattingEditProvider.showStatusBar(document, textEdits); - - return this.logAndReturnTextEdits(textEdits, getFormattingDuration); - } - - // There is something about having this code in the calling method that causes a TS compile error. - // It causes the following error: - // Type 'import("C:/Users/Keith/GitHub/rkeithhill/vscode-powershell/node_modules/vscode-languageserver-typ...' - // is not assignable to type ''vscode'.TextEdit'. Property 'newEol' is missing in type 'TextEdit'. - private logAndReturnTextEdits( - textEdits, - getFormattingDuration: () => number): vscode.TextEdit[] | Thenable { - - return textEdits.then((edits) => { - this.logger.writeVerbose(`Document formatting finished in ${getFormattingDuration()}s`); - return edits; - }, (err) => { - this.logger.writeVerbose(`Document formatting failed in ${getFormattingDuration()}: ${err}`); - }); - } - - private getScriptRegion(document: TextDocument, position: Position, ch: string): Thenable { - const oneBasedPosition = toOneBasedPosition(position); - return this.languageClient.sendRequest( - ScriptRegionRequestType, - { - fileUri: document.uri.toString(), - character: ch, - line: oneBasedPosition.line, - column: oneBasedPosition.character, - }).then((result: IScriptRegionRequestResult) => { - if (result === null) { - return null; - } - - return result.scriptRegion; - }); - } - - private getEditor(document: TextDocument): TextEditor { - return Window.visibleTextEditors.find((e, n, obj) => e.document === document); - } - - private isDocumentLocked(document: TextDocument): boolean { - return PSDocumentFormattingEditProvider.documentLocker.isLocked(document); - } - - private lockDocument(document: TextDocument, unlockWhenDone: Thenable): void { - PSDocumentFormattingEditProvider.documentLocker.lock(document, unlockWhenDone); - } - - private getEditorSettings(editor: TextEditor): { insertSpaces: boolean, tabSize: number } { - // Writing the editor options allows string or strong types going in, but always - // resolves to an appropriate value on read. - return { - insertSpaces: editor.options.insertSpaces as boolean, - tabSize: editor.options.tabSize as number, - }; - } -} - -export class DocumentFormatterFeature implements IFeature { - private firstTriggerCharacter: string = "}"; - private moreTriggerCharacters: string[] = ["\n"]; - private formattingEditProvider: vscode.Disposable; - private rangeFormattingEditProvider: vscode.Disposable; - private onTypeFormattingEditProvider: vscode.Disposable; - private languageClient: LanguageClient; - private documentFormattingEditProvider: PSDocumentFormattingEditProvider; - - constructor(private logger: Logger, documentSelector: DocumentSelector) { - this.documentFormattingEditProvider = new PSDocumentFormattingEditProvider(logger); - this.formattingEditProvider = vscode.languages.registerDocumentFormattingEditProvider( - documentSelector, - this.documentFormattingEditProvider); - this.rangeFormattingEditProvider = vscode.languages.registerDocumentRangeFormattingEditProvider( - documentSelector, - this.documentFormattingEditProvider); - this.onTypeFormattingEditProvider = vscode.languages.registerOnTypeFormattingEditProvider( - documentSelector, - this.documentFormattingEditProvider, - this.firstTriggerCharacter, - ...this.moreTriggerCharacters); - } - - public dispose(): any { - this.formattingEditProvider.dispose(); - this.rangeFormattingEditProvider.dispose(); - this.onTypeFormattingEditProvider.dispose(); - } - - public setLanguageClient(languageclient: LanguageClient): void { - this.languageClient = languageclient; - this.documentFormattingEditProvider.setLanguageClient(languageclient); - } -} diff --git a/src/main.ts b/src/main.ts index dd6148df42..b88886f9ca 100644 --- a/src/main.ts +++ b/src/main.ts @@ -16,7 +16,6 @@ import { DebugSessionFeature } from "./features/DebugSession"; import { PickPSHostProcessFeature } from "./features/DebugSession"; import { PickRunspaceFeature } from "./features/DebugSession"; import { SpecifyScriptArgsFeature } from "./features/DebugSession"; -import { DocumentFormatterFeature } from "./features/DocumentFormatter"; import { ExamplesFeature } from "./features/Examples"; import { ExpandAliasFeature } from "./features/ExpandAlias"; import { ExtensionCommandsFeature } from "./features/ExtensionCommands"; @@ -155,7 +154,6 @@ export function activate(context: vscode.ExtensionContext): void { new SelectPSSARulesFeature(logger), new CodeActionsFeature(logger), new NewFileOrProjectFeature(), - new DocumentFormatterFeature(logger, documentSelector), new RemoteFilesFeature(), new DebugSessionFeature(context, sessionManager), new PickPSHostProcessFeature(), diff --git a/src/session.ts b/src/session.ts index 5a0598f287..78e0482a1c 100644 --- a/src/session.ts +++ b/src/session.ts @@ -352,15 +352,15 @@ export class SessionManager implements Middleware { codeLensToFix.command.arguments = [ vscode.Uri.parse(oldArgs[0]), - new vscode.Position(oldArgs[1].line, oldArgs[1].character), + new vscode.Position(oldArgs[1].Line, oldArgs[1].Character), oldArgs[2].map((position) => { return new vscode.Location( - vscode.Uri.parse(position.uri), + vscode.Uri.parse(position.Uri), new vscode.Range( - position.range.start.line, - position.range.start.character, - position.range.end.line, - position.range.end.character)); + position.Range.Start.Line, + position.Range.Start.Character, + position.Range.End.Line, + position.Range.End.Character)); }), ]; } From edf0bcfeab96ae2622c21ab9435a3365960c3f60 Mon Sep 17 00:00:00 2001 From: Tyler James Leonhardt Date: Tue, 27 Aug 2019 07:22:07 -0700 Subject: [PATCH 2/2] Needed changes due to lack of support for strings over the wire. (#2150) * everything needed so far * needed changes due to lack of support for strings over the wire --- src/features/ExpandAlias.ts | 6 +++--- src/features/GetCommands.ts | 6 +++--- src/features/ShowHelp.ts | 10 +++++----- 3 files changed, 11 insertions(+), 11 deletions(-) diff --git a/src/features/ExpandAlias.ts b/src/features/ExpandAlias.ts index dd1872ff90..ec1568e098 100644 --- a/src/features/ExpandAlias.ts +++ b/src/features/ExpandAlias.ts @@ -8,7 +8,7 @@ import { LanguageClient, NotificationType, RequestType } from "vscode-languagecl import { IFeature } from "../feature"; import { Logger } from "../logging"; -export const ExpandAliasRequestType = new RequestType("powerShell/expandAlias"); +export const ExpandAliasRequestType = new RequestType("powerShell/expandAlias"); export class ExpandAliasFeature implements IFeature { private command: vscode.Disposable; @@ -39,9 +39,9 @@ export class ExpandAliasFeature implements IFeature { range = new vscode.Range(sls.line, sls.character, sle.line, sle.character); } - this.languageClient.sendRequest(ExpandAliasRequestType, text).then((result) => { + this.languageClient.sendRequest(ExpandAliasRequestType, { text }).then((result) => { editor.edit((editBuilder) => { - editBuilder.replace(range, result); + editBuilder.replace(range, result.text); }); }); }); diff --git a/src/features/GetCommands.ts b/src/features/GetCommands.ts index b109c74e73..12fff1d453 100644 --- a/src/features/GetCommands.ts +++ b/src/features/GetCommands.ts @@ -2,7 +2,7 @@ * Copyright (C) Microsoft Corporation. All rights reserved. *--------------------------------------------------------*/ import * as vscode from "vscode"; -import { LanguageClient, RequestType } from "vscode-languageclient"; +import { LanguageClient, RequestType0 } from "vscode-languageclient"; import { IFeature } from "../feature"; import { Logger } from "../logging"; @@ -18,7 +18,7 @@ interface ICommand { * RequestType sent over to PSES. * Expects: ICommand to be returned */ -export const GetCommandRequestType = new RequestType("powerShell/getCommand"); +export const GetCommandRequestType = new RequestType0("powerShell/getCommand"); /** * A PowerShell Command listing feature. Implements a treeview control. @@ -63,7 +63,7 @@ export class GetCommandsFeature implements IFeature { this.log.writeVerbose(`<${GetCommandsFeature.name}>: Unable to send getCommand request`); return; } - this.languageClient.sendRequest(GetCommandRequestType, "").then((result) => { + this.languageClient.sendRequest(GetCommandRequestType).then((result) => { const SidebarConfig = vscode.workspace.getConfiguration("powershell.sideBar"); const excludeFilter = (SidebarConfig.CommandExplorerExcludeFilter).map((filter) => filter.toLowerCase()); result = result.filter((command) => (excludeFilter.indexOf(command.moduleName.toLowerCase()) === -1)); diff --git a/src/features/ShowHelp.ts b/src/features/ShowHelp.ts index 3322a43f66..fd185eb05a 100644 --- a/src/features/ShowHelp.ts +++ b/src/features/ShowHelp.ts @@ -3,12 +3,12 @@ *--------------------------------------------------------*/ import vscode = require("vscode"); -import { LanguageClient, NotificationType, RequestType } from "vscode-languageclient"; +import { LanguageClient, NotificationType } from "vscode-languageclient"; import { IFeature } from "../feature"; import { Logger } from "../logging"; -export const ShowHelpRequestType = - new RequestType("powerShell/showHelp"); +export const ShowHelpNotificationType = + new NotificationType("powerShell/showHelp"); export class ShowHelpFeature implements IFeature { private command: vscode.Disposable; @@ -31,9 +31,9 @@ export class ShowHelpFeature implements IFeature { const cwr = doc.getWordRangeAtPosition(selection.active); const text = doc.getText(cwr); - this.languageClient.sendRequest(ShowHelpRequestType, text); + this.languageClient.sendNotification(ShowHelpNotificationType, { text }); } else { - this.languageClient.sendRequest(ShowHelpRequestType, item.Name); + this.languageClient.sendNotification(ShowHelpNotificationType, { text: item.Name } ); } }); }