Skip to content

Commit

Permalink
Implement client-side of SnippetTextEdit
Browse files Browse the repository at this point in the history
  • Loading branch information
matklad committed May 17, 2020
1 parent 331a0c6 commit 97bd6d1
Show file tree
Hide file tree
Showing 3 changed files with 80 additions and 1 deletion.
46 changes: 45 additions & 1 deletion editors/code/src/client.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,39 @@ export function createClient(serverPath: string, cwd: string): lc.LanguageClient
const res = await next(document, token);
if (res === undefined) throw new Error('busy');
return res;
},
async provideCodeActions(document: vscode.TextDocument, range: vscode.Range, context: vscode.CodeActionContext, token: vscode.CancellationToken, _next: lc.ProvideCodeActionsSignature) {
const params: lc.CodeActionParams = {
textDocument: client.code2ProtocolConverter.asTextDocumentIdentifier(document),
range: client.code2ProtocolConverter.asRange(range),
context: client.code2ProtocolConverter.asCodeActionContext(context)
};
return client.sendRequest(lc.CodeActionRequest.type, params, token).then((values) => {
if (values === null) return undefined;
const result: (vscode.CodeAction | vscode.Command)[] = [];
for (const item of values) {
if (lc.CodeAction.is(item)) {
const action = client.protocol2CodeConverter.asCodeAction(item);
if (isSnippetEdit(item)) {
action.command = {
command: "rust-analyzer.applySnippetWorkspaceEdit",
title: "",
arguments: [action.edit],
}
action.edit = undefined;
}
result.push(action);
} else {
const command = client.protocol2CodeConverter.asCommand(item);
result.push(command);
}
}
return result;
},
(_error) => undefined
);
}

} as any
};

Expand All @@ -42,7 +74,7 @@ export function createClient(serverPath: string, cwd: string): lc.LanguageClient
clientOptions,
);

// To turn on all proposed features use: res.registerProposedFeatures();
// To turn on all proposed features use: client.registerProposedFeatures();
// Here we want to enable CallHierarchyFeature and SemanticTokensFeature
// since they are available on stable.
// Note that while these features are stable in vscode their LSP protocol
Expand All @@ -63,3 +95,15 @@ class SnippetTextEditFeature implements lc.StaticFeature {
initialize(_capabilities: lc.ServerCapabilities<any>, _documentSelector: lc.DocumentSelector | undefined): void {
}
}

function isSnippetEdit(action: lc.CodeAction): boolean {
const documentChanges = action.edit?.documentChanges ?? [];
for (const edit of documentChanges) {
if (lc.TextDocumentEdit.is(edit)) {
if (edit.edits.some((indel) => (indel as any).insertTextFormat == lc.InsertTextFormat.Snippet)) {
return true;
}
}
}
return false;
}
34 changes: 34 additions & 0 deletions editors/code/src/commands/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import * as ra from '../rust-analyzer-api';

import { Ctx, Cmd } from '../ctx';
import * as sourceChange from '../source_change';
import { assert } from '../util';

export * from './analyzer_status';
export * from './matching_brace';
Expand Down Expand Up @@ -51,3 +52,36 @@ export function selectAndApplySourceChange(ctx: Ctx): Cmd {
}
};
}

export function applySnippetWorkspaceEdit(_ctx: Ctx): Cmd {
return async (edit: vscode.WorkspaceEdit) => {
assert(edit.entries().length == 1, `bad ws edit: ${JSON.stringify(edit)}`);
const [uri, edits] = edit.entries()[0];

const editor = vscode.window.visibleTextEditors.find((it) => it.document.uri.toString() == uri.toString());
if (!editor) return;

let editWithSnippet: vscode.TextEdit | undefined = undefined;
let lineDelta = 0;
await editor.edit((builder) => {
for (const indel of edits) {
if (indel.newText.indexOf('$0') != -1) {
editWithSnippet = indel as vscode.TextEdit;
} else {
if (!editWithSnippet) {
lineDelta = (indel.newText.match(/\n/g) || []).length - (indel.range.end.line - indel.range.start.line);
}
builder.replace(indel.range, indel.newText);
}
}
})
if (editWithSnippet) {
const snip = editWithSnippet as vscode.TextEdit;
const range = snip.range.with(
snip.range.start.with(snip.range.start.line + lineDelta),
snip.range.end.with(snip.range.end.line + lineDelta),
)
await editor.insertSnippet(new vscode.SnippetString(snip.newText), range);
}
};
}
1 change: 1 addition & 0 deletions editors/code/src/main.ts
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,7 @@ export async function activate(context: vscode.ExtensionContext) {
ctx.registerCommand('debugSingle', commands.debugSingle);
ctx.registerCommand('showReferences', commands.showReferences);
ctx.registerCommand('applySourceChange', commands.applySourceChange);
ctx.registerCommand('applySnippetWorkspaceEdit', commands.applySnippetWorkspaceEdit);
ctx.registerCommand('selectAndApplySourceChange', commands.selectAndApplySourceChange);

ctx.pushCleanup(activateTaskProvider(workspaceFolder));
Expand Down

0 comments on commit 97bd6d1

Please sign in to comment.