aboutsummaryrefslogtreecommitdiff
path: root/editors/code/src/commands/index.ts
blob: 0937b495c26c711ddf8b344deef4d4ecb641b91d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
import * as vscode from 'vscode';
import * as lc from 'vscode-languageclient';
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';
export * from './join_lines';
export * from './on_enter';
export * from './parent_module';
export * from './syntax_tree';
export * from './expand_macro';
export * from './runnables';
export * from './ssr';
export * from './server_version';

export function collectGarbage(ctx: Ctx): Cmd {
    return async () => ctx.client.sendRequest(ra.collectGarbage, null);
}

export function showReferences(ctx: Ctx): Cmd {
    return (uri: string, position: lc.Position, locations: lc.Location[]) => {
        const client = ctx.client;
        if (client) {
            vscode.commands.executeCommand(
                'editor.action.showReferences',
                vscode.Uri.parse(uri),
                client.protocol2CodeConverter.asPosition(position),
                locations.map(client.protocol2CodeConverter.asLocation),
            );
        }
    };
}

export function applySourceChange(ctx: Ctx): Cmd {
    return async (change: ra.SourceChange) => {
        await sourceChange.applySourceChange(ctx, change);
    };
}

export function selectAndApplySourceChange(ctx: Ctx): Cmd {
    return async (changes: ra.SourceChange[]) => {
        if (changes.length === 1) {
            await sourceChange.applySourceChange(ctx, changes[0]);
        } else if (changes.length > 0) {
            const selectedChange = await vscode.window.showQuickPick(changes);
            if (!selectedChange) return;
            await sourceChange.applySourceChange(ctx, selectedChange);
        }
    };
}

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) {
                const isSnippet = indel.newText.indexOf('$0') !== -1 || indel.newText.indexOf('${') !== -1;
                if (isSnippet) {
                    editWithSnippet = indel;
                } 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);
        }
    };
}