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-27 19:52:43 +00:00
|
|
|
import { DH_UNABLE_TO_CHECK_GENERATOR } from 'constants';
|
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 = {
|
|
|
|
syntaxTree: vscode.Uri.parse('libsyntax-rust://syntaxtree')
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function activate(context: vscode.ExtensionContext) {
|
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))
|
|
|
|
}
|
|
|
|
|
|
|
|
registerCommand('libsyntax-rust.syntaxTree', () => openDoc(uris.syntaxTree))
|
2018-08-10 18:13:39 +00:00
|
|
|
registerCommand('libsyntax-rust.extendSelection', async () => {
|
|
|
|
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-08-15 21:23:22 +00:00
|
|
|
registerCommand('libsyntax-rust.matchingBrace', async () => {
|
|
|
|
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-08-23 19:14:51 +00:00
|
|
|
registerCommand('libsyntax-rust.joinLines', async () => {
|
|
|
|
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),
|
|
|
|
}
|
|
|
|
let response = await client.sendRequest<lc.TextEdit[]>("m/joinLines", request)
|
|
|
|
let edits = client.protocol2CodeConverter.asTextEdits(response)
|
|
|
|
let wsEdit = new vscode.WorkspaceEdit()
|
|
|
|
wsEdit.set(editor.document.uri, edits)
|
|
|
|
return vscode.workspace.applyEdit(wsEdit)
|
|
|
|
})
|
2018-08-22 07:18:58 +00:00
|
|
|
registerCommand('libsyntax-rust.parentModule', async () => {
|
|
|
|
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)
|
|
|
|
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
|
|
|
|
registerCommand('libsyntax-rust.run', async () => {
|
|
|
|
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-08-28 15:22:52 +00:00
|
|
|
registerCommand('libsyntax-rust.fsEdit', async (ops: FsOp[]) => {
|
2018-08-27 17:58:38 +00:00
|
|
|
let edit = new vscode.WorkspaceEdit()
|
2018-08-28 15:22:52 +00:00
|
|
|
let created;
|
|
|
|
let moved;
|
|
|
|
for (let op of ops) {
|
|
|
|
if (op.type == "createFile") {
|
|
|
|
let uri = vscode.Uri.parse(op.uri!)
|
|
|
|
edit.createFile(uri)
|
|
|
|
created = uri
|
|
|
|
} else if (op.type == "moveFile") {
|
|
|
|
let src = vscode.Uri.parse(op.src!)
|
|
|
|
let dst = vscode.Uri.parse(op.dst!)
|
|
|
|
edit.renameFile(src, dst)
|
|
|
|
moved = dst
|
|
|
|
} else {
|
|
|
|
console.error(`unknown op: ${JSON.stringify(op)}`)
|
|
|
|
}
|
|
|
|
}
|
2018-08-27 17:58:38 +00:00
|
|
|
await vscode.workspace.applyEdit(edit)
|
2018-08-28 15:22:52 +00:00
|
|
|
let doc = await vscode.workspace.openTextDocument((created || moved)!)
|
2018-08-27 17:58:38 +00:00
|
|
|
await vscode.window.showTextDocument(doc)
|
|
|
|
})
|
2018-08-10 18:13:39 +00:00
|
|
|
|
2018-08-10 12:07:43 +00:00
|
|
|
dispose(vscode.workspace.registerTextDocumentContentProvider(
|
|
|
|
'libsyntax-rust',
|
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-08-13 12:35:53 +00:00
|
|
|
// command: "cargo",
|
|
|
|
// args: ["run", "--package", "m"],
|
|
|
|
command: "m",
|
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-08-10 12:07:43 +00:00
|
|
|
'm',
|
|
|
|
'm languge server',
|
|
|
|
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-16 10:46:31 +00:00
|
|
|
client.onRequest(
|
|
|
|
new lc.RequestType<lc.Position, void, any, any>("m/moveCursor"),
|
|
|
|
(params: lc.Position, token: lc.CancellationToken) => {
|
|
|
|
let editor = vscode.window.activeTextEditor;
|
2018-08-27 19:52:43 +00:00
|
|
|
if (!editor) return
|
2018-08-16 10:46:31 +00:00
|
|
|
if (!editor.selection.isEmpty) return
|
|
|
|
let position = client.protocol2CodeConverter.asPosition(params)
|
2018-08-17 16:54:08 +00:00
|
|
|
afterLs(() => {
|
2018-08-27 19:52:43 +00:00
|
|
|
editor!.selection = new vscode.Selection(position, position)
|
2018-08-17 16:54:08 +00:00
|
|
|
})
|
2018-08-16 10:46:31 +00:00
|
|
|
}
|
|
|
|
)
|
2018-08-10 21:55:32 +00:00
|
|
|
})
|
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,
|
|
|
|
highlihgs: Array<Decoration>
|
|
|
|
) {
|
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 highlihgs) {
|
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
|
|
|
|
|
|
|
interface FsOp {
|
|
|
|
type: string;
|
|
|
|
uri?: string;
|
|
|
|
src?: string;
|
|
|
|
dst?: string;
|
|
|
|
}
|