rust-analyzer/editors/code/src/extension.ts

410 lines
13 KiB
TypeScript
Raw Normal View History

2018-08-10 12:07:43 +00:00
'use strict';
import * as vscode from 'vscode';
2018-08-10 21:55:32 +00:00
import * as lc from 'vscode-languageclient'
2018-08-10 12:07:43 +00:00
2018-08-10 21:55:32 +00:00
let client: lc.LanguageClient;
2018-08-10 12:07:43 +00:00
let uris = {
2018-09-16 09:54:24 +00:00
syntaxTree: vscode.Uri.parse('ra-lsp://syntaxtree')
2018-08-10 12:07:43 +00:00
}
let highlightingOn = true;
2018-08-10 12:07:43 +00:00
export function activate(context: vscode.ExtensionContext) {
let config = vscode.workspace.getConfiguration('ra-lsp');
if (config.has('highlightingOn')) {
highlightingOn = config.get('highlightingOn') as boolean;
}
2018-08-10 18:13:39 +00:00
let textDocumentContentProvider = new TextDocumentContentProvider()
2018-08-27 19:52:43 +00:00
let dispose = (disposable: vscode.Disposable) => {
2018-08-10 12:07:43 +00:00
context.subscriptions.push(disposable);
}
2018-08-27 19:52:43 +00:00
let registerCommand = (name: string, f: any) => {
2018-08-10 12:07:43 +00:00
dispose(vscode.commands.registerCommand(name, f))
}
2018-09-16 09:54:24 +00:00
registerCommand('ra-lsp.syntaxTree', () => openDoc(uris.syntaxTree))
registerCommand('ra-lsp.extendSelection', async () => {
2018-08-10 18:13:39 +00:00
let editor = vscode.window.activeTextEditor
if (editor == null || editor.document.languageId != "rust") return
let request: ExtendSelectionParams = {
textDocument: { uri: editor.document.uri.toString() },
selections: editor.selections.map((s) => {
2018-08-11 13:10:34 +00:00
return client.code2ProtocolConverter.asRange(s)
2018-08-10 18:13:39 +00:00
})
}
let response = await client.sendRequest<ExtendSelectionResult>("m/extendSelection", request)
editor.selections = response.selections.map((range) => {
2018-08-11 13:10:34 +00:00
let r = client.protocol2CodeConverter.asRange(range)
return new vscode.Selection(r.start, r.end)
2018-08-10 18:13:39 +00:00
})
})
2018-09-16 09:54:24 +00:00
registerCommand('ra-lsp.matchingBrace', async () => {
2018-08-15 21:23:22 +00:00
let editor = vscode.window.activeTextEditor
if (editor == null || editor.document.languageId != "rust") return
let request: FindMatchingBraceParams = {
textDocument: { uri: editor.document.uri.toString() },
offsets: editor.selections.map((s) => {
return client.code2ProtocolConverter.asPosition(s.active)
2018-08-27 19:52:43 +00:00
})
2018-08-15 21:23:22 +00:00
}
let response = await client.sendRequest<lc.Position[]>("m/findMatchingBrace", request)
editor.selections = editor.selections.map((sel, idx) => {
let active = client.protocol2CodeConverter.asPosition(response[idx])
let anchor = sel.isEmpty ? active : sel.anchor
return new vscode.Selection(anchor, active)
})
2018-08-23 21:13:16 +00:00
editor.revealRange(editor.selection)
2018-08-15 21:23:22 +00:00
})
2018-09-16 09:54:24 +00:00
registerCommand('ra-lsp.joinLines', async () => {
2018-08-23 19:14:51 +00:00
let editor = vscode.window.activeTextEditor
if (editor == null || editor.document.languageId != "rust") return
let request: JoinLinesParams = {
textDocument: { uri: editor.document.uri.toString() },
range: client.code2ProtocolConverter.asRange(editor.selection),
}
2018-08-29 15:03:14 +00:00
let change = await client.sendRequest<SourceChange>("m/joinLines", request)
await applySourceChange(change)
2018-08-23 19:14:51 +00:00
})
2018-09-16 09:54:24 +00:00
registerCommand('ra-lsp.parentModule', async () => {
2018-08-22 07:18:58 +00:00
let editor = vscode.window.activeTextEditor
if (editor == null || editor.document.languageId != "rust") return
let request: lc.TextDocumentIdentifier = {
uri: editor.document.uri.toString()
}
2018-08-27 19:52:43 +00:00
let response = await client.sendRequest<lc.Location[]>("m/parentModule", request)
let loc = response[0]
2018-08-22 07:18:58 +00:00
if (loc == null) return
let uri = client.protocol2CodeConverter.asUri(loc.uri)
let range = client.protocol2CodeConverter.asRange(loc.range)
let doc = await vscode.workspace.openTextDocument(uri)
let e = await vscode.window.showTextDocument(doc)
2018-09-16 13:25:24 +00:00
e.selection = new vscode.Selection(range.start, range.start)
2018-08-22 07:18:58 +00:00
e.revealRange(range, vscode.TextEditorRevealType.InCenter)
})
2018-08-27 17:58:38 +00:00
2018-08-27 19:52:43 +00:00
let prevRunnable: RunnableQuickPick | undefined = undefined
2018-09-16 09:54:24 +00:00
registerCommand('ra-lsp.run', async () => {
2018-08-27 19:52:43 +00:00
let editor = vscode.window.activeTextEditor
if (editor == null || editor.document.languageId != "rust") return
let textDocument: lc.TextDocumentIdentifier = {
uri: editor.document.uri.toString()
}
let params: RunnablesParams = {
textDocument,
position: client.code2ProtocolConverter.asPosition(editor.selection.active)
}
let runnables = await client.sendRequest<Runnable[]>('m/runnables', params)
let items: RunnableQuickPick[] = []
if (prevRunnable) {
items.push(prevRunnable)
}
for (let r of runnables) {
2018-08-27 20:12:45 +00:00
if (prevRunnable && JSON.stringify(prevRunnable.runnable) == JSON.stringify(r)) {
continue
}
2018-08-27 19:52:43 +00:00
items.push(new RunnableQuickPick(r))
}
let item = await vscode.window.showQuickPick(items)
if (item) {
2018-08-27 20:12:45 +00:00
item.detail = "rerun"
2018-08-27 19:52:43 +00:00
prevRunnable = item
let task = createTask(item.runnable)
return await vscode.tasks.executeTask(task)
}
2018-08-24 10:41:25 +00:00
})
2018-09-16 09:54:24 +00:00
registerCommand('ra-lsp.applySourceChange', applySourceChange)
2018-08-10 18:13:39 +00:00
2018-08-10 12:07:43 +00:00
dispose(vscode.workspace.registerTextDocumentContentProvider(
2018-09-16 09:54:24 +00:00
'ra-lsp',
2018-08-10 18:13:39 +00:00
textDocumentContentProvider
2018-08-10 12:07:43 +00:00
))
startServer()
2018-08-10 18:13:39 +00:00
vscode.workspace.onDidChangeTextDocument((event: vscode.TextDocumentChangeEvent) => {
let doc = event.document
if (doc.languageId != "rust") return
2018-08-17 16:54:08 +00:00
afterLs(() => {
2018-08-10 18:13:39 +00:00
textDocumentContentProvider.eventEmitter.fire(uris.syntaxTree)
2018-08-17 16:54:08 +00:00
})
2018-08-10 18:13:39 +00:00
}, null, context.subscriptions)
2018-08-27 21:20:59 +00:00
vscode.window.onDidChangeActiveTextEditor(async (editor) => {
if (!editor || editor.document.languageId != 'rust') return
let params: lc.TextDocumentIdentifier = {
uri: editor.document.uri.toString()
}
let decorations = await client.sendRequest<Decoration[]>("m/decorationsRequest", params)
setHighlights(editor, decorations)
})
2018-08-10 12:07:43 +00:00
}
2018-08-17 16:54:08 +00:00
// We need to order this after LS updates, but there's no API for that.
// Hence, good old setTimeout.
2018-08-27 19:52:43 +00:00
function afterLs(f: () => any) {
2018-08-17 16:54:08 +00:00
setTimeout(f, 10)
}
2018-08-10 12:07:43 +00:00
export function deactivate(): Thenable<void> {
if (!client) {
2018-08-27 19:52:43 +00:00
return Promise.resolve();
2018-08-10 12:07:43 +00:00
}
return client.stop();
}
function startServer() {
2018-08-10 21:55:32 +00:00
let run: lc.Executable = {
2018-09-16 09:54:24 +00:00
command: "ra_lsp_server",
2018-08-10 14:49:45 +00:00
options: { cwd: "." }
2018-08-10 12:07:43 +00:00
}
2018-08-10 21:55:32 +00:00
let serverOptions: lc.ServerOptions = {
2018-08-10 12:07:43 +00:00
run,
debug: run
};
2018-08-10 21:55:32 +00:00
let clientOptions: lc.LanguageClientOptions = {
2018-08-10 12:07:43 +00:00
documentSelector: [{ scheme: 'file', language: 'rust' }],
};
2018-08-10 21:55:32 +00:00
client = new lc.LanguageClient(
2018-09-16 09:54:24 +00:00
'ra-lsp',
'rust-analyzer languge server',
2018-08-10 12:07:43 +00:00
serverOptions,
clientOptions,
);
2018-08-10 21:55:32 +00:00
client.onReady().then(() => {
client.onNotification(
2018-08-27 19:52:43 +00:00
"m/publishDecorations",
2018-08-10 21:55:32 +00:00
(params: PublishDecorationsParams) => {
let editor = vscode.window.visibleTextEditors.find(
(editor) => editor.document.uri.toString() == params.uri
)
if (editor == null) return;
setHighlights(
editor,
params.decorations,
)
}
)
})
2018-08-10 12:07:43 +00:00
client.start();
}
async function openDoc(uri: vscode.Uri) {
let document = await vscode.workspace.openTextDocument(uri)
return vscode.window.showTextDocument(document, vscode.ViewColumn.Two, true)
}
class TextDocumentContentProvider implements vscode.TextDocumentContentProvider {
public eventEmitter = new vscode.EventEmitter<vscode.Uri>()
public syntaxTree: string = "Not available"
public provideTextDocumentContent(uri: vscode.Uri): vscode.ProviderResult<string> {
let editor = vscode.window.activeTextEditor;
if (editor == null) return ""
2018-08-10 18:13:39 +00:00
let request: SyntaxTreeParams = {
textDocument: { uri: editor.document.uri.toString() }
};
return client.sendRequest<SyntaxTreeResult>("m/syntaxTree", request);
2018-08-10 12:07:43 +00:00
}
get onDidChange(): vscode.Event<vscode.Uri> {
return this.eventEmitter.event
}
}
2018-08-10 18:13:39 +00:00
2018-08-10 21:55:32 +00:00
2018-08-27 19:52:43 +00:00
const decorations: { [index: string]: vscode.TextEditorDecorationType } = (() => {
const decor = (obj: any) => vscode.window.createTextEditorDecorationType({ color: obj })
2018-08-10 21:55:32 +00:00
return {
background: decor("#3F3F3F"),
error: vscode.window.createTextEditorDecorationType({
borderColor: "red",
borderStyle: "none none dashed none",
}),
comment: decor("#7F9F7F"),
string: decor("#CC9393"),
keyword: decor("#F0DFAF"),
function: decor("#93E0E3"),
parameter: decor("#94BFF3"),
builtin: decor("#DD6718"),
text: decor("#DCDCCC"),
attribute: decor("#BFEBBF"),
literal: decor("#DFAF8F"),
}
})()
function setHighlights(
editor: vscode.TextEditor,
highlights: Array<Decoration>
2018-08-10 21:55:32 +00:00
) {
if (!highlightingOn) {
return;
}
2018-08-27 19:52:43 +00:00
let byTag: Map<string, vscode.Range[]> = new Map()
2018-08-10 21:55:32 +00:00
for (let tag in decorations) {
2018-08-27 19:52:43 +00:00
byTag.set(tag, [])
2018-08-10 21:55:32 +00:00
}
for (let d of highlights) {
2018-08-27 19:52:43 +00:00
if (!byTag.get(d.tag)) {
2018-08-10 21:55:32 +00:00
console.log(`unknown tag ${d.tag}`)
continue
}
2018-08-27 19:52:43 +00:00
byTag.get(d.tag)!.push(
2018-08-11 13:10:34 +00:00
client.protocol2CodeConverter.asRange(d.range)
)
2018-08-10 21:55:32 +00:00
}
2018-08-27 19:52:43 +00:00
for (let tag of byTag.keys()) {
let dec: vscode.TextEditorDecorationType = decorations[tag]
let ranges = byTag.get(tag)!
2018-08-10 21:55:32 +00:00
editor.setDecorations(dec, ranges)
}
}
2018-08-10 18:13:39 +00:00
interface SyntaxTreeParams {
2018-08-10 21:55:32 +00:00
textDocument: lc.TextDocumentIdentifier;
2018-08-10 18:13:39 +00:00
}
type SyntaxTreeResult = string
interface ExtendSelectionParams {
2018-08-10 21:55:32 +00:00
textDocument: lc.TextDocumentIdentifier;
selections: lc.Range[];
2018-08-10 18:13:39 +00:00
}
interface ExtendSelectionResult {
2018-08-10 21:55:32 +00:00
selections: lc.Range[];
}
2018-08-15 21:23:22 +00:00
interface FindMatchingBraceParams {
textDocument: lc.TextDocumentIdentifier;
offsets: lc.Position[];
}
2018-08-23 19:14:51 +00:00
interface JoinLinesParams {
textDocument: lc.TextDocumentIdentifier;
range: lc.Range;
}
2018-08-10 21:55:32 +00:00
interface PublishDecorationsParams {
uri: string,
decorations: Decoration[],
}
2018-08-27 19:52:43 +00:00
interface RunnablesParams {
textDocument: lc.TextDocumentIdentifier,
position?: lc.Position,
2018-08-10 18:13:39 +00:00
}
2018-08-24 10:41:25 +00:00
2018-08-27 19:52:43 +00:00
interface Runnable {
range: lc.Range;
label: string;
2018-08-24 10:41:25 +00:00
bin: string;
args: string[];
2018-08-27 19:52:43 +00:00
env: { [index: string]: string },
2018-08-24 10:41:25 +00:00
}
2018-08-27 19:52:43 +00:00
class RunnableQuickPick implements vscode.QuickPickItem {
label: string;
description?: string | undefined;
detail?: string | undefined;
picked?: boolean | undefined;
constructor(public runnable: Runnable) {
this.label = runnable.label
}
}
interface Decoration {
range: lc.Range,
tag: string,
}
2018-08-24 10:41:25 +00:00
interface CargoTaskDefinition extends vscode.TaskDefinition {
type: 'cargo';
label: string;
command: string;
args: Array<string>;
env?: { [key: string]: string };
}
2018-08-27 19:52:43 +00:00
function createTask(spec: Runnable): vscode.Task {
2018-08-24 10:41:25 +00:00
const TASK_SOURCE = 'Rust';
let definition: CargoTaskDefinition = {
type: 'cargo',
label: 'cargo',
command: spec.bin,
args: spec.args,
env: spec.env
}
let execCmd = `${definition.command} ${definition.args.join(' ')}`;
let execOption: vscode.ShellExecutionOptions = {
cwd: '.',
env: definition.env,
};
2018-08-27 20:12:45 +00:00
let exec = new vscode.ShellExecution(`clear; ${execCmd}`, execOption);
2018-08-24 10:41:25 +00:00
2018-08-27 19:52:43 +00:00
let f = vscode.workspace.workspaceFolders![0]
2018-08-24 10:41:25 +00:00
let t = new vscode.Task(definition, f, definition.label, TASK_SOURCE, exec, ['$rustc']);
return t;
}
2018-08-28 15:22:52 +00:00
2018-08-29 15:03:14 +00:00
interface FileSystemEdit {
2018-08-28 15:22:52 +00:00
type: string;
uri?: string;
src?: string;
dst?: string;
}
2018-08-29 15:03:14 +00:00
interface SourceChange {
label: string,
sourceFileEdits: lc.TextDocumentEdit[],
fileSystemEdits: FileSystemEdit[],
cursorPosition?: lc.TextDocumentPositionParams,
}
async function applySourceChange(change: SourceChange) {
console.log(`applySOurceChange ${JSON.stringify(change)}`)
let wsEdit = new vscode.WorkspaceEdit()
for (let sourceEdit of change.sourceFileEdits) {
let uri = client.protocol2CodeConverter.asUri(sourceEdit.textDocument.uri)
let edits = client.protocol2CodeConverter.asTextEdits(sourceEdit.edits)
wsEdit.set(uri, edits)
}
let created;
let moved;
for (let fsEdit of change.fileSystemEdits) {
if (fsEdit.type == "createFile") {
let uri = vscode.Uri.parse(fsEdit.uri!)
wsEdit.createFile(uri)
created = uri
} else if (fsEdit.type == "moveFile") {
let src = vscode.Uri.parse(fsEdit.src!)
let dst = vscode.Uri.parse(fsEdit.dst!)
wsEdit.renameFile(src, dst)
moved = dst
} else {
console.error(`unknown op: ${JSON.stringify(fsEdit)}`)
}
}
let toOpen = created || moved
let toReveal = change.cursorPosition
await vscode.workspace.applyEdit(wsEdit)
if (toOpen) {
let doc = await vscode.workspace.openTextDocument(toOpen)
await vscode.window.showTextDocument(doc)
} else if (toReveal) {
let uri = client.protocol2CodeConverter.asUri(toReveal.textDocument.uri)
let position = client.protocol2CodeConverter.asPosition(toReveal.position)
let editor = vscode.window.activeTextEditor;
2018-08-30 09:51:46 +00:00
if (!editor || editor.document.uri.toString() != uri.toString()) return
2018-08-29 15:03:14 +00:00
if (!editor.selection.isEmpty) return
editor!.selection = new vscode.Selection(position, position)
}
}