feat(lsp): buffer renames in order to send a single, atomic message
This commit is contained in:
parent
1fd4455adb
commit
1c37de6c29
3 changed files with 84 additions and 49 deletions
|
@ -4323,13 +4323,18 @@ pub const Editor = struct {
|
|||
self.need_render();
|
||||
}
|
||||
|
||||
pub fn rename_symbol_item(self: *Self, sel: Selection, new_text: []const u8) Result {
|
||||
pub fn rename_symbol_item(
|
||||
self: *Self,
|
||||
sel: Selection,
|
||||
new_text: []const u8,
|
||||
root_prev: *?Buffer.Root,
|
||||
final: bool,
|
||||
) !void {
|
||||
self.get_primary().selection = sel;
|
||||
const buf = try self.buf_for_update();
|
||||
const r1 = try self.delete_selection(buf.root, self.get_primary(), self.allocator);
|
||||
const r2 = try self.insert(r1, self.get_primary(), new_text, self.allocator);
|
||||
try self.update_buf(r2);
|
||||
self.need_render();
|
||||
if (root_prev.* == null) root_prev.* = (try self.buf_for_update()).root;
|
||||
const root = try self.delete_selection(root_prev.*.?, self.get_primary(), self.allocator);
|
||||
root_prev.* = try self.insert(root, self.get_primary(), new_text, self.allocator);
|
||||
if (final) try self.update_buf(root_prev.*.?);
|
||||
}
|
||||
|
||||
pub fn select(self: *Self, ctx: Context) Result {
|
||||
|
|
|
@ -555,26 +555,34 @@ const cmds = struct {
|
|||
pub const add_diagnostic_meta = .{ .arguments = &.{ .string, .string, .string, .string, .integer, .integer, .integer, .integer, .integer } };
|
||||
|
||||
pub fn rename_symbol_item(self: *Self, ctx: Ctx) Result {
|
||||
var file_uri: []const u8 = undefined;
|
||||
var sel: ed.Selection = .{};
|
||||
var new_text: []const u8 = undefined;
|
||||
if (!try ctx.args.match(.{
|
||||
tp.extract(&file_uri),
|
||||
tp.extract(&sel.begin.row),
|
||||
tp.extract(&sel.begin.col),
|
||||
tp.extract(&sel.end.row),
|
||||
tp.extract(&sel.end.col),
|
||||
tp.extract(&new_text),
|
||||
})) return error.InvalidRenameSymbolArgument;
|
||||
file_uri = project_manager.normalize_file_path(file_uri);
|
||||
if (self.get_active_editor()) |editor| {
|
||||
// TODO match correctly. endsWith() isn't correct because path is a
|
||||
// short, relative path while file_uri is an absolute path starting with 'file://'
|
||||
const match = if (editor.file_path) |path| std.mem.endsWith(u8, file_uri, path) else false;
|
||||
if (match) {
|
||||
try editor.rename_symbol_item(sel, new_text);
|
||||
} else {
|
||||
// TODO perform renames in other files
|
||||
// because the incoming message is an array of Renames, we manuallly
|
||||
// parse instead of using ctx.args.match() which doesn't seem to return
|
||||
// the parsed length needed to correctly advance iter.
|
||||
var iter = ctx.args.buf;
|
||||
var len = try cbor.decodeArrayHeader(&iter);
|
||||
var mroot: ?@import("Buffer").Root = null;
|
||||
while (len != 0) {
|
||||
var file_uri: []const u8 = undefined;
|
||||
var sel: ed.Selection = .{};
|
||||
var new_text: []const u8 = undefined;
|
||||
len -= 1;
|
||||
std.debug.assert(try cbor.decodeArrayHeader(&iter) == 6);
|
||||
if (!try cbor.matchString(&iter, &file_uri)) return error.MissingArgument;
|
||||
if (!try cbor.matchInt(usize, &iter, &sel.begin.row)) return error.MissingArgument;
|
||||
if (!try cbor.matchInt(usize, &iter, &sel.begin.col)) return error.MissingArgument;
|
||||
if (!try cbor.matchInt(usize, &iter, &sel.end.row)) return error.MissingArgument;
|
||||
if (!try cbor.matchInt(usize, &iter, &sel.end.col)) return error.MissingArgument;
|
||||
if (!try cbor.matchString(&iter, &new_text)) return error.MissingArgument;
|
||||
file_uri = project_manager.normalize_file_path(file_uri);
|
||||
if (self.get_active_editor()) |editor| {
|
||||
// TODO match file_uri correctly. endsWith() isn't correct because 'path' is a
|
||||
// short, relative path while 'file_uri' is an absolute path starting with 'file://'
|
||||
const match = if (editor.file_path) |path| std.mem.endsWith(u8, file_uri, path) else false;
|
||||
if (match) {
|
||||
try editor.rename_symbol_item(sel, new_text, &mroot, len == 0);
|
||||
} else {
|
||||
log.logger("LSP").print("TODO perform renames in other files\n", .{});
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue