const std = @import("std"); const Allocator = std.mem.Allocator; const log = @import("log"); const tp = @import("thespian"); const location_history = @import("location_history"); const command = @import("command"); const cmd = command.executeName; const tui = @import("../tui.zig"); const Editor = @import("../editor.zig").Editor; const CurSel = @import("../editor.zig").CurSel; const Buffer = @import("Buffer"); const Cursor = Buffer.Cursor; const Selection = Buffer.Selection; const Direction = enum { backwards, forwards }; var commands: Commands = undefined; pub fn init() !void { var v: void = {}; try commands.init(&v); } pub fn deinit() void { commands.deinit(); } const Commands = command.Collection(cmds_); const cmds_ = struct { pub const Target = void; const Ctx = command.Context; const Meta = command.Metadata; const Result = command.Result; pub fn w(_: *void, _: Ctx) Result { try cmd("save_file", .{}); } pub const w_meta: Meta = .{ .description = "w (write/save file)" }; pub fn q(_: *void, _: Ctx) Result { try cmd("quit", .{}); } pub const q_meta: Meta = .{ .description = "q (quit)" }; pub fn qa(_: *void, _: Ctx) Result { try cmd("quit", .{}); } pub const qa_meta: Meta = .{ .description = "qa (close all)" }; pub fn @"q!"(_: *void, _: Ctx) Result { try cmd("quit_without_saving", .{}); } pub const @"q!_meta": Meta = .{ .description = "q! (quit without saving)" }; pub fn @"qa!"(_: *void, _: Ctx) Result { try cmd("quit_without_saving", .{}); } pub const @"qa!_meta": Meta = .{ .description = "qa! (quit without saving)" }; pub fn wq(_: *void, _: Ctx) Result { try cmd("save_file", command.fmt(.{ "then", .{ "quit", .{} } })); } pub const wq_meta: Meta = .{ .description = "wq (write/save file and quit)" }; pub fn @"x!"(_: *void, _: Ctx) Result { try cmd("save_file", command.fmt(.{ "then", .{ "quit_without_saving", .{} } })); } pub const @"x!_meta": Meta = .{ .description = "x! (write/save file and exit, ignoring other unsaved changes)" }; pub fn x(_: *void, _: Ctx) Result { try cmd("save_file", command.fmt(.{ "then", .{ "quit", .{} } })); } pub const x_meta: Meta = .{ .description = "x (write/save file and quit)" }; pub fn wa(_: *void, _: Ctx) Result { if (tui.get_buffer_manager()) |bm| bm.save_all() catch |e| return tp.exit_error(e, @errorReturnTrace()); } pub const wa_meta: Meta = .{ .description = "wa (save all)" }; pub fn xa(_: *void, _: Ctx) Result { if (tui.get_buffer_manager()) |bm| { bm.save_all() catch |e| return tp.exit_error(e, @errorReturnTrace()); try cmd("quit", .{}); } } pub const xa_meta: Meta = .{ .description = "xa (write all and quit)" }; pub fn @"xa!"(_: *void, _: Ctx) Result { if (tui.get_buffer_manager()) |bm| { bm.save_all() catch {}; try cmd("quit_without_saving", .{}); } } pub const @"xa!_meta": Meta = .{ .description = "xa! (write all and exit, ignoring other unsaved changes)" }; pub fn wqa(_: *void, _: Ctx) Result { if (tui.get_buffer_manager()) |bm| bm.save_all() catch |e| return tp.exit_error(e, @errorReturnTrace()); try cmd("quit", .{}); } pub const wqa_meta: Meta = .{ .description = "wqa (write all and quit)" }; pub fn @"wqa!"(_: *void, _: Ctx) Result { if (tui.get_buffer_manager()) |bm| { bm.save_all() catch {}; try cmd("quit_without_saving", .{}); } } pub const @"wqa!_meta": Meta = .{ .description = "wqa! (write all and exit, ignoring unsaved changes)" }; pub fn rl(_: *void, _: Ctx) Result { try cmd("reload_file", .{}); } pub const rl_meta: Meta = .{ .description = "rl (reload current file)" }; pub fn rla(_: *void, _: Ctx) Result { if (tui.get_buffer_manager()) |bm| bm.reload_all() catch |e| return tp.exit_error(e, @errorReturnTrace()); } pub const rla_meta: Meta = .{ .description = "rla (reload all files)" }; pub fn o(_: *void, _: Ctx) Result { try cmd("open_file", .{}); } pub const o_meta: Meta = .{ .description = "o (open file)" }; pub fn @"wq!"(_: *void, _: Ctx) Result { cmd("save_file", .{}) catch {}; try cmd("quit_without_saving", .{}); } pub const @"wq!_meta": Meta = .{ .description = "wq! (write/save file and quit without saving)" }; pub fn n(_: *void, _: Ctx) Result { try cmd("create_new_file", .{}); } pub const n_meta: Meta = .{ .description = "n (Create new buffer/tab)" }; pub fn bn(_: *void, _: Ctx) Result { try cmd("next_tab", .{}); } pub const bn_meta: Meta = .{ .description = "bn (Next buffer/tab)" }; pub fn bp(_: *void, _: Ctx) Result { try cmd("previous_tab", .{}); } pub const bp_meta: Meta = .{ .description = "bp (Previous buffer/tab)" }; pub fn bc(_: *void, _: Ctx) Result { try cmd("delete_buffer", .{}); } pub const bc_meta: Meta = .{ .description = "bc (Close buffer/tab)" }; pub fn @"bc!"(_: *void, _: Ctx) Result { try cmd("close_file_without_saving", .{}); } pub const @"bc!_meta": Meta = .{ .description = "bc! (Close buffer/tab, ignoring changes)" }; pub fn @"bco!"(_: *void, _: Ctx) Result { const mv = tui.mainview() orelse return; if (tui.get_buffer_manager()) |bm| { if (mv.get_active_buffer()) |buffer| try bm.delete_others(buffer); } } pub const @"bco!_meta": Meta = .{ .description = "bco! (Close other buffers/tabs, discarding changes)" }; pub fn bco(_: *void, _: Ctx) Result { const logger = log.logger("helix-mode"); defer logger.deinit(); const mv = tui.mainview() orelse return; const bm = tui.get_buffer_manager() orelse return; if (mv.get_active_buffer()) |buffer| { const remaining = try bm.close_others(buffer); if (remaining > 0) { logger.print("{} unsaved buffer(s) remaining", .{remaining}); try cmd("next_tab", .{}); } } } pub const bco_meta: Meta = .{ .description = "bco (Close other buffers/tabs)" }; pub fn save_selection(_: *void, _: Ctx) Result { const logger = log.logger("helix-mode"); defer logger.deinit(); logger.print("saved location", .{}); const mv = tui.mainview() orelse return; const file_path = mv.get_active_file_path() orelse return; const primary = (mv.get_active_editor() orelse return).get_primary(); const sel: ?location_history.Selection = if (primary.selection) |sel| .{ .begin = .{ .row = sel.begin.row, .col = sel.begin.col }, .end = .{ .row = sel.end.row, .col = sel.end.col }, } else null; mv.location_history_.update(file_path, .{ .row = primary.cursor.row + 1, .col = primary.cursor.col + 1, }, sel); } pub const save_selection_meta: Meta = .{ .description = "Save current selection to location history" }; pub fn split_selection_on_newline(_: *void, _: Ctx) Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = try ed.buf_root(); const cursels = try ed.cursels.toOwnedSlice(ed.allocator); defer ed.allocator.free(cursels); for (cursels) |*cursel_| if (cursel_.*) |*cursel| { try add_cursors_to_cursel_line_ends_helix(ed, root, cursel); }; ed.clamp(); } pub const split_selection_on_newline_meta: Meta = .{ .description = "Add cursor to each line in selection helix" }; pub fn match_brackets(_: *void, ctx: Ctx) Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = ed.buf_root() catch return; try ed.with_cursels_const_once_arg(root, &match_bracket, ctx); ed.clamp(); } pub const match_brackets_meta: Meta = .{ .description = "Goto matching bracket" }; pub fn extend_line_below(_: *void, ctx: Ctx) Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = try ed.buf_root(); var repeat: usize = 1; _ = ctx.args.match(.{tp.extract(&repeat)}) catch false; while (repeat > 0) : (repeat -= 1) { for (ed.cursels.items) |*cursel_| if (cursel_.*) |*cursel| { const sel = cursel.enable_selection_normal(); sel.normalize(); try Editor.move_cursor_begin(root, &sel.begin, ed.metrics); try Editor.move_cursor_end(root, &sel.end, ed.metrics); try Editor.move_cursor_right(root, &sel.end, ed.metrics); cursel.cursor = sel.end; }; } ed.clamp(); } pub const extend_line_below_meta: Meta = .{ .arguments = &.{.integer}, .description = "Select current line, if already selected, extend to next line" }; pub fn move_next_word_start(_: *void, ctx: Ctx) Result { try move_to_word(ctx, Editor.move_cursor_word_right_vim, .forwards); } pub const move_next_word_start_meta: Meta = .{ .description = "Move next word start", .arguments = &.{.integer} }; pub fn extend_next_word_start(_: *void, ctx: Ctx) Result { try extend_to_word(ctx, Editor.move_cursor_word_right_vim, .forwards); } pub const extend_next_word_start_meta: Meta = .{ .description = "Extend next word start", .arguments = &.{.integer} }; pub fn move_next_long_word_start(_: *void, ctx: Ctx) Result { try move_to_word(ctx, move_cursor_long_word_right, .forwards); } pub const move_next_long_word_start_meta: Meta = .{ .description = "Move next long word start", .arguments = &.{.integer} }; pub fn extend_next_long_word_start(_: *void, ctx: Ctx) Result { try extend_to_word(ctx, move_cursor_long_word_right, .forwards); } pub const extend_next_long_word_start_meta: Meta = .{ .description = "Extend next long word start", .arguments = &.{.integer} }; pub fn move_prev_word_start(_: *void, ctx: Ctx) Result { try move_to_word(ctx, move_cursor_word_left_helix, .backwards); } pub const move_prev_word_start_meta: Meta = .{ .description = "Move previous word start", .arguments = &.{.integer} }; pub fn extend_prev_word_start(_: *void, ctx: Ctx) Result { try extend_to_word(ctx, move_cursor_word_left_helix, .backwards); } pub const extend_prev_word_start_meta: Meta = .{ .description = "Extend previous word start", .arguments = &.{.integer} }; pub fn move_prev_long_word_start(_: *void, ctx: Ctx) Result { try move_to_word(ctx, move_cursor_long_word_left, .backwards); } pub const move_prev_long_word_start_meta: Meta = .{ .description = "Move previous long word start", .arguments = &.{.integer} }; pub fn extend_prev_long_word_start(_: *void, ctx: Ctx) Result { try extend_to_word(ctx, move_cursor_long_word_left, .backwards); } pub const extend_prev_long_word_start_meta: Meta = .{ .description = "Extend previous word start", .arguments = &.{.integer} }; pub fn move_next_word_end(_: *void, ctx: Ctx) Result { try move_to_word(ctx, move_cursor_word_right_end_helix, .forwards); } pub const move_next_word_end_meta: Meta = .{ .description = "Move next word end", .arguments = &.{.integer} }; pub fn extend_next_word_end(_: *void, ctx: Ctx) Result { try extend_to_word(ctx, move_cursor_word_right_end_helix, .forwards); } pub const extend_next_word_end_meta: Meta = .{ .description = "Extend next word end", .arguments = &.{.integer} }; pub fn move_next_long_word_end(_: *void, ctx: Ctx) Result { try move_to_word(ctx, move_cursor_long_word_right_end, .forwards); } pub const move_next_long_word_end_meta: Meta = .{ .description = "Move next long word end", .arguments = &.{.integer} }; pub fn extend_next_long_word_end(_: *void, ctx: Ctx) Result { try extend_to_word(ctx, move_cursor_long_word_right_end, .forwards); } pub const extend_next_long_word_end_meta: Meta = .{ .description = "Extend next long word end", .arguments = &.{.integer} }; pub fn cut_forward_internal_inclusive(_: *void, _: Ctx) Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const b = try ed.buf_for_update(); tui.clipboard_start_group(); const root = try ed.cut_to(move_noop, b.root); try ed.update_buf(root); ed.clamp(); } pub const cut_forward_internal_inclusive_meta: Meta = .{ .description = "Cut next character to internal clipboard (inclusive)" }; pub fn select_right_helix(_: *void, ctx: Ctx) Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = try ed.buf_root(); var repeat: usize = 1; _ = ctx.args.match(.{tp.extract(&repeat)}) catch false; while (repeat > 0) : (repeat -= 1) { for (ed.cursels.items) |*cursel_| if (cursel_.*) |*cursel| { const sel = cursel.enable_selection(root, ed.metrics); // handling left to right transition const sel_begin: i32 = @intCast(sel.begin.col); const sel_end: i32 = @intCast(sel.end.col); if ((sel_begin - sel_end) == 1 and sel.begin.row == sel.end.row) { try Editor.move_cursor_right(root, &sel.end, ed.metrics); sel.begin.col -= 1; } try Editor.move_cursor_right(root, &sel.end, ed.metrics); cursel.cursor = sel.end; cursel.check_selection(root, ed.metrics); }; } ed.clamp(); } pub const select_right_helix_meta: Meta = .{ .description = "Select right", .arguments = &.{.integer} }; pub fn select_left_helix(_: *void, ctx: Ctx) Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = try ed.buf_root(); var repeat: usize = 1; _ = ctx.args.match(.{tp.extract(&repeat)}) catch false; while (repeat > 0) : (repeat -= 1) { for (ed.cursels.items) |*cursel_| if (cursel_.*) |*cursel| { if (cursel.selection == null) { cursel.selection = Selection.from_cursor(&cursel.cursor); try cursel.selection.?.begin.move_right(root, ed.metrics); } if (cursel.selection) |*sel| { try Editor.move_cursor_left(root, &sel.end, ed.metrics); cursel.cursor = sel.end; if (sel.begin.col == sel.end.col and sel.begin.row == sel.end.row) { try sel.begin.move_right(root, ed.metrics); try Editor.move_cursor_left(root, &sel.end, ed.metrics); cursel.cursor = sel.end; } } cursel.check_selection(root, ed.metrics); }; } ed.clamp(); } pub const select_left_helix_meta: Meta = .{ .description = "Select left", .arguments = &.{.integer} }; pub fn select_to_char_left_helix(_: *void, ctx: Ctx) Result { try to_char_helix(ctx, &select_cursel_to_char_left_helix); } pub const select_to_char_left_helix_meta: Meta = .{ .description = "Select to char left" }; pub fn select_till_char_left_helix(_: *void, ctx: Ctx) Result { try to_char_helix(ctx, &select_cursel_till_char_left_helix); } pub const select_till_char_left_helix_meta: Meta = .{ .description = "Select until char left" }; pub fn extend_to_char_left_helix(_: *void, ctx: Ctx) Result { try to_char_helix(ctx, &extend_cursel_to_char_left_helix); } pub const extend_to_char_left_helix_meta: Meta = .{ .description = "Extend Selection to char left" }; pub fn extend_till_char_left_helix(_: *void, ctx: Ctx) Result { try to_char_helix(ctx, &extend_cursel_till_char_left_helix); } pub const extend_till_char_left_helix_meta: Meta = .{ .description = "Extend Selection until char left" }; pub fn select_till_char_right_helix(_: *void, ctx: Ctx) Result { try to_char_helix(ctx, &select_cursel_till_char_right_helix); } pub const select_till_char_right_helix_meta: Meta = .{ .description = "Select until char right" }; pub fn select_to_char_right_helix(_: *void, ctx: Ctx) Result { try to_char_helix(ctx, &select_cursel_to_char_right_helix); } pub const select_to_char_right_helix_meta: Meta = .{ .description = "Select to char right" }; pub fn extend_till_char_right_helix(_: *void, ctx: Ctx) Result { try to_char_helix(ctx, &extend_cursel_till_char_right_helix); } pub const extend_till_char_right_helix_meta: Meta = .{ .description = "Extend Selection until char right" }; pub fn extend_to_char_right_helix(_: *void, ctx: Ctx) Result { try to_char_helix(ctx, &extend_cursel_to_char_right_helix); } pub const extend_to_char_right_helix_meta: Meta = .{ .description = "Extend Selection to char right" }; pub fn select_textobject_inner(_: *void, ctx: Ctx) Result { var action: []const u8 = ""; if (!try ctx.args.match(.{tp.extract(&action)})) return error.Stop; const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = ed.buf_root() catch return; if (std.mem.eql(u8, action, "w")) { try ed.with_cursels_const(root, select_inner_word, ed.metrics); } else if (std.mem.eql(u8, action, "W")) { try ed.with_cursels_const(root, select_inner_long_word, ed.metrics); } else { return; } ed.clamp(); } pub const select_textobject_inner_meta: Meta = .{ .description = "select inside object helix" }; pub fn select_textobject_around(_: *void, ctx: Ctx) Result { var action: []const u8 = ""; if (!try ctx.args.match(.{tp.extract(&action)})) return error.Stop; const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = ed.buf_root() catch return; if (std.mem.eql(u8, action, "w")) { try ed.with_cursels_const(root, select_around_word, ed.metrics); } else if (std.mem.eql(u8, action, "W")) { try ed.with_cursels_const(root, select_inner_long_word, ed.metrics); } else { return; } ed.clamp(); } pub const select_textobject_around_meta: Meta = .{ .description = "select around object helix" }; pub fn copy_helix(_: *void, _: Ctx) Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = ed.buf_root() catch return; tui.clipboard_start_group(); for (ed.cursels.items) |*cursel_| if (cursel_.*) |*cursel| if (cursel.selection) |sel| tui.clipboard_add_chunk(try Editor.copy_selection(root, sel, tui.clipboard_allocator(), ed.metrics)); ed.logger.print("copy: {d} selections", .{ed.cursels.items.len}); } pub const copy_helix_meta: Meta = .{ .description = "Copy selection to clipboard (helix)" }; pub fn paste_after(_: *void, ctx: Ctx) Result { try paste_helix(ctx, insert_after); } pub const paste_after_meta: Meta = .{ .description = "Paste from clipboard after selection" }; pub fn replace_selections_with_clipboard(_: *void, ctx: Ctx) Result { try paste_helix(ctx, insert_replace_selection); } pub const replace_selections_with_clipboard_meta: Meta = .{ .description = "Replace selection from clipboard" }; pub fn paste_clipboard_before(_: *void, ctx: Ctx) Result { try paste_helix(ctx, insert_before); } pub const paste_clipboard_before_meta: Meta = .{ .description = "Paste from clipboard before selection" }; pub fn replace_with_character_helix(_: *void, ctx: Ctx) Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; var root = ed.buf_root() catch return; root = try ed.with_cursels_mut_once_arg(root, replace_cursel_with_character, ed.allocator, ctx); try ed.update_buf(root); ed.clamp(); ed.need_render(); } pub const replace_with_character_helix_meta: Meta = .{ .description = "Replace with character" }; }; fn match_bracket(root: Buffer.Root, cursel: *CurSel, ctx: command.Context, metrics: Buffer.Metrics) error{Stop}!void { var symbol: []const u8 = undefined; const mode: enum { helix_sel_mode, helix_nor_mode } = if ((ctx.args.match(.{tp.extract(&symbol)}) catch false) and std.mem.eql(u8, @tagName(.helix_sel_mode), symbol)) .helix_sel_mode else .helix_nor_mode; if (mode == .helix_sel_mode) { const begin: Cursor = if (cursel.selection) |sel| sel.begin else cursel.*.cursor; if (cursel.*.selection) |*sel| { const row, const col = Editor.match_bracket(root, cursel.*.cursor, metrics) catch blk: { // Selection in hx mode requires to move to the left to begin manipulation try cursel.*.cursor.move_left(root, metrics); break :blk try Editor.match_bracket(root, cursel.*.cursor, metrics); }; cursel.*.cursor.row = row; cursel.*.cursor.col = col; sel.end = cursel.*.cursor; //Then to include the whole selection, requires to extend to the right if (sel.is_reversed()) { try sel.begin.move_right(root, metrics); } else { try cursel.*.cursor.move_right(root, metrics); try sel.end.move_right(root, metrics); } } else { cursel.*.selection = Selection.from_cursor(&begin); cursel.*.selection.?.end = cursel.*.cursor; } } else { const row, const col = try Editor.match_bracket(root, cursel.*.cursor, metrics); cursel.*.cursor.row = row; cursel.*.cursor.col = col; cursel.*.selection = null; } } fn move_to_word(ctx: command.Context, move: Editor.cursor_operator_const, direction: Direction) command.Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = try ed.buf_root(); // NOR mode moves n words selecting the last one var repeat: usize = 0; _ = ctx.args.match(.{tp.extract(&repeat)}) catch false; if (repeat > 1) ed.with_cursors_const_repeat(root, move, command.fmt(.{repeat - 1})) catch {}; for (ed.cursels.items) |*cursel_| if (cursel_.*) |*cursel| { var sel = Selection.from_cursor(&cursel.cursor); const cur = sel.begin.test_at(root, is_not_whitespace_or_eol, ed.metrics); if (direction == .backwards) { sel.begin.move_left(root, ed.metrics) catch continue; const prev = sel.begin.test_at(root, Editor.is_not_word_char, ed.metrics); sel.begin = sel.end; if (!cur or cur != prev) sel.begin.move_right(root, ed.metrics) catch continue; } else { sel.end.move_right(root, ed.metrics) catch continue; const next = sel.end.test_at(root, Editor.is_not_word_char, ed.metrics); if (!cur and cur != next) sel.begin = sel.end; } cursel.cursor = sel.end; cursel.selection = sel; }; ed.with_selections_const_repeat(root, move, command.fmt(.{1})) catch {}; ed.clamp(); } fn extend_to_word(ctx: command.Context, move: Editor.cursor_operator_const, _: Direction) command.Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = try ed.buf_root(); var repeat: usize = 1; _ = ctx.args.match(.{tp.extract(&repeat)}) catch false; for (ed.cursels.items) |*cursel_| if (cursel_.*) |*cursel| { const sel = try cursel.enable_selection(root, ed.metrics); const pivot: usize = if (sel.is_reversed()) sel.begin.col -| 1 else sel.begin.col; var i: usize = repeat; while (i > 0) : (i -= 1) { try move(root, &sel.end, ed.metrics); } sel.begin.col = if (sel.is_reversed()) pivot +| 1 else pivot; cursel.cursor = sel.end; }; ed.clamp(); } fn to_char_helix(ctx: command.Context, move: Editor.cursel_operator_mut_once_arg) command.Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; const root = ed.buf_root() catch return; try ed.with_cursels_const_once_arg(root, move, ctx); ed.clamp(); } fn select_inner_word(root: Buffer.Root, cursel: *CurSel, metrics: Buffer.Metrics) !void { if (!cursel.cursor.test_at(root, Editor.is_word_char, metrics)) return; var prev = cursel.cursor; var next = cursel.cursor; Editor.move_cursor_left_until(root, &prev, Editor.is_word_boundary_left, metrics); Editor.move_cursor_right_until(root, &next, Editor.is_word_boundary_right, metrics); try next.move_right(root, metrics); const sel = cursel.enable_selection(root, metrics); sel.begin = prev; sel.end = next; cursel.*.cursor = next; } fn select_inner_long_word(root: Buffer.Root, cursel: *CurSel, metrics: Buffer.Metrics) !void { if (cursel.cursor.test_at(root, Editor.is_whitespace, metrics)) return; var prev = cursel.cursor; var next = cursel.cursor; Editor.move_cursor_left_until(root, &prev, is_long_word_boundary_left, metrics); Editor.move_cursor_right_until(root, &next, is_long_word_boundary_right, metrics); try next.move_right(root, metrics); const sel = cursel.enable_selection(root, metrics); sel.begin = prev; sel.end = next; cursel.*.cursor = next; } fn is_tab_or_space(c: []const u8) bool { return (c[0] == ' ') or (c[0] == '\t'); } fn is_tab_or_espace_at_cursor(root: Buffer.Root, cursor: *const Cursor, metrics: Buffer.Metrics) bool { return cursor.test_at(root, is_tab_or_space, metrics); } fn is_not_tab_or_espace_at_cursor(root: Buffer.Root, cursor: *const Cursor, metrics: Buffer.Metrics) bool { return !cursor.test_at(root, is_tab_or_space, metrics); } fn select_around_word(root: Buffer.Root, cursel: *CurSel, metrics: Buffer.Metrics) !void { if (!cursel.cursor.test_at(root, Editor.is_word_char, metrics)) return; var expander = cursel.*; try select_inner_word(root, &expander, metrics); const sel_e = expander.enable_selection(root, metrics); var prev = sel_e.begin; var next = sel_e.end; if (next.test_at(root, is_tab_or_space, metrics)) { Editor.move_cursor_right_until(root, &next, is_not_tab_or_espace_at_cursor, metrics); } else { next = sel_e.end; prev.move_left(root, metrics) catch {}; if (prev.test_at(root, is_tab_or_space, metrics)) { Editor.move_cursor_left_until(root, &prev, is_not_tab_or_espace_at_cursor, metrics); prev.move_right(root, metrics) catch {}; } else { prev = sel_e.begin; } } const sel = cursel.enable_selection(root, metrics); sel.begin = prev; sel.end = next; cursel.*.cursor = next; } fn select_around_long_word(root: Buffer.Root, cursel: *CurSel, metrics: Buffer.Metrics) !void { if (!cursel.cursor.test_at(root, Editor.is_word_char, metrics)) return; var expander = cursel.*; try select_inner_long_word(root, &expander, metrics); const sel_e = try expander.enable_selection(root, metrics); var prev = sel_e.begin; var next = sel_e.end; if (next.test_at(root, is_tab_or_space, metrics)) { Editor.move_cursor_right_until(root, &next, is_not_tab_or_espace_at_cursor, metrics); } else { next = sel_e.end; prev.move_left(root, metrics) catch {}; if (prev.test_at(root, is_tab_or_space, metrics)) { Editor.move_cursor_left_until(root, &prev, is_not_tab_or_espace_at_cursor, metrics); prev.move_right(root, metrics) catch {}; } else { prev = sel_e.begin; } } const sel = try cursel.enable_selection(root, metrics); sel.begin = prev; sel.end = next; cursel.*.cursor = next; } fn select_cursel_to_char_left_helix(root: Buffer.Root, cursel: *CurSel, ctx: command.Context, metrics: Buffer.Metrics) error{Stop}!void { var moving_cursor: Cursor = cursel.*.cursor; var begin = cursel.*.cursor; move_cursor_to_char_left_beyond_eol(root, &moving_cursor, metrics, ctx) catch return; // Character found, selecting Editor.move_cursor_right(root, &begin, metrics) catch { //At end of file, it's ok }; moving_cursor.target = moving_cursor.col; const sel = cursel.enable_selection(root, metrics); sel.begin = begin; sel.end = moving_cursor; cursel.cursor = moving_cursor; } fn extend_cursel_to_char_left_helix(root: Buffer.Root, cursel: *CurSel, ctx: command.Context, metrics: Buffer.Metrics) error{Stop}!void { var moving_cursor: Cursor = cursel.*.cursor; const begin = if (cursel.*.selection) |sel| sel.end else cursel.*.cursor; move_cursor_to_char_left_beyond_eol(root, &moving_cursor, metrics, ctx) catch return; //Character found, selecting moving_cursor.target = moving_cursor.col; const sel = cursel.enable_selection(root, metrics); if (sel.empty()) sel.begin = begin; sel.end = moving_cursor; cursel.cursor = moving_cursor; } fn select_cursel_till_char_left_helix(root: Buffer.Root, cursel: *CurSel, ctx: command.Context, metrics: Buffer.Metrics) error{Stop}!void { var moving_cursor: Cursor = cursel.*.cursor; var begin = cursel.*.cursor; move_cursor_till_char_left_beyond_eol(root, &moving_cursor, metrics, ctx) catch return; // Character found, selecting Editor.move_cursor_right(root, &begin, metrics) catch { //At end of file, it's ok }; moving_cursor.target = moving_cursor.col; const sel = cursel.enable_selection(root, metrics); sel.begin = begin; sel.end = moving_cursor; cursel.cursor = moving_cursor; } fn extend_cursel_till_char_left_helix(root: Buffer.Root, cursel: *CurSel, ctx: command.Context, metrics: Buffer.Metrics) error{Stop}!void { var moving_cursor: Cursor = cursel.*.cursor; const begin = if (cursel.*.selection) |sel| sel.end else cursel.*.cursor; move_cursor_till_char_left_beyond_eol(root, &moving_cursor, metrics, ctx) catch return; //Character found, selecting moving_cursor.target = moving_cursor.col; const sel = cursel.enable_selection(root, metrics); if (sel.empty()) sel.begin = begin; sel.end = moving_cursor; cursel.cursor = moving_cursor; } fn select_cursel_till_char_right_helix(root: Buffer.Root, cursel: *CurSel, ctx: command.Context, metrics: Buffer.Metrics) error{Stop}!void { var moving_cursor: Cursor = cursel.*.cursor; const begin = cursel.*.cursor; move_cursor_to_char_right_beyond_eol(root, &moving_cursor, metrics, ctx) catch return; //Character found, selecting moving_cursor.target = moving_cursor.col; const sel = cursel.enable_selection(root, metrics); sel.begin = begin; sel.end = moving_cursor; cursel.cursor = moving_cursor; } fn extend_cursel_till_char_right_helix(root: Buffer.Root, cursel: *CurSel, ctx: command.Context, metrics: Buffer.Metrics) error{Stop}!void { var moving_cursor: Cursor = cursel.*.cursor; const begin = cursel.*.cursor; move_cursor_to_char_right_beyond_eol(root, &moving_cursor, metrics, ctx) catch return; //Character found, selecting moving_cursor.target = moving_cursor.col; const sel = cursel.enable_selection(root, metrics); if (sel.empty()) sel.begin = begin; sel.end = moving_cursor; cursel.cursor = moving_cursor; } fn select_cursel_to_char_right_helix(root: Buffer.Root, cursel: *CurSel, ctx: command.Context, metrics: Buffer.Metrics) error{Stop}!void { var moving_cursor: Cursor = cursel.*.cursor; const begin = cursel.*.cursor; move_cursor_to_char_right_beyond_eol(root, &moving_cursor, metrics, ctx) catch return; //Character found, selecting Editor.move_cursor_right(root, &moving_cursor, metrics) catch { // We might be at end of file }; moving_cursor.target = moving_cursor.col; const sel = cursel.enable_selection(root, metrics); sel.begin = begin; sel.end = moving_cursor; cursel.cursor = moving_cursor; } fn extend_cursel_to_char_right_helix(root: Buffer.Root, cursel: *CurSel, ctx: command.Context, metrics: Buffer.Metrics) error{Stop}!void { var moving_cursor: Cursor = cursel.*.cursor; const begin = cursel.*.cursor; move_cursor_to_char_right_beyond_eol(root, &moving_cursor, metrics, ctx) catch return; //Character found, selecting Editor.move_cursor_right(root, &moving_cursor, metrics) catch { // We might be at end of file }; moving_cursor.target = moving_cursor.col; const sel = cursel.enable_selection(root, metrics); if (sel.empty()) sel.begin = begin; sel.end = moving_cursor; cursel.cursor = moving_cursor; } fn move_cursor_find_egc_beyond_eol(root: Buffer.Root, cursor: *Cursor, ctx: command.Context, metrics: Buffer.Metrics, move: find_char_function) error{Stop}!void { move(root, cursor, metrics, ctx); } fn move_cursor_word_left_helix(root: Buffer.Root, cursor: *Cursor, metrics: Buffer.Metrics) error{Stop}!void { try Editor.move_cursor_left(root, cursor, metrics); // Consume " " while (Editor.is_whitespace_at_cursor(root, cursor, metrics)) { try Editor.move_cursor_left(root, cursor, metrics); } var next = cursor.*; next.move_left(root, metrics) catch return; var next_next = next; next_next.move_left(root, metrics) catch return; const cur = next.test_at(root, Editor.is_not_word_char, metrics); const nxt = next_next.test_at(root, Editor.is_not_word_char, metrics); if (cur != nxt) { try Editor.move_cursor_left(root, cursor, metrics); return; } else { try move_cursor_word_left_helix(root, cursor, metrics); } } fn replace_cursel_with_character(ed: *Editor, root: Buffer.Root, cursel: *CurSel, allocator: Allocator, ctx: command.Context) error{Stop}!Buffer.Root { var egc: []const u8 = undefined; if (!(ctx.args.match(.{tp.extract(&egc)}) catch return error.Stop)) return error.Stop; const no_selection = try select_char_if_no_selection(cursel, root, ed.metrics); var begin: Cursor = undefined; var sel_length: usize = 1; if (cursel.selection) |*sel| { sel.normalize(); begin = sel.*.begin; _ = root.get_range(sel.*, null, null, &sel_length, ed.metrics) catch return error.Stop; } const total_length = sel_length * egc.len; var sfa = std.heap.stackFallback(4096, ed.allocator); const sfa_allocator = sfa.get(); const replacement = sfa_allocator.alloc(u8, total_length) catch return error.Stop; defer sfa_allocator.free(replacement); for (0..sel_length) |i| @memcpy(replacement[i * egc.len .. (i + 1) * egc.len], egc); const root_ = insert_replace_selection(ed, root, cursel, replacement, allocator) catch return error.Stop; if (no_selection) { try cursel.cursor.move_left(root, ed.metrics); cursel.disable_selection(root, ed.metrics); } else { cursel.selection = Selection{ .begin = begin, .end = cursel.cursor }; } return root_; } fn move_noop(_: Buffer.Root, _: *Cursor, _: Buffer.Metrics) error{Stop}!void {} fn move_cursor_word_right_end_helix(root: Buffer.Root, cursor: *Cursor, metrics: Buffer.Metrics) error{Stop}!void { try Editor.move_cursor_right(root, cursor, metrics); Editor.move_cursor_right_until(root, cursor, Editor.is_word_boundary_right_vim, metrics); } fn move_cursor_to_char_left_beyond_eol(root: Buffer.Root, cursor: *Cursor, metrics: Buffer.Metrics, ctx: command.Context) error{Stop}!void { var egc: []const u8 = undefined; if (!(ctx.args.match(.{tp.extract(&egc)}) catch return error.Stop)) return error.Stop; var test_cursor = cursor.*; try test_cursor.move_left(root, metrics); while (true) { const curr_egc, _, _ = root.egc_at(test_cursor.row, test_cursor.col, metrics) catch return error.Stop; if (std.mem.eql(u8, curr_egc, egc)) { cursor.row = test_cursor.row; cursor.col = test_cursor.col; cursor.target = cursor.col; return; } test_cursor.move_left(root, metrics) catch return error.Stop; } } fn move_cursor_to_char_right_beyond_eol(root: Buffer.Root, cursor: *Cursor, metrics: Buffer.Metrics, ctx: command.Context) error{Stop}!void { var egc: []const u8 = undefined; if (!(ctx.args.match(.{tp.extract(&egc)}) catch return error.Stop)) return error.Stop; var test_cursor = cursor.*; while (true) { const curr_egc, _, _ = root.egc_at(test_cursor.row, test_cursor.col, metrics) catch return error.Stop; if (std.mem.eql(u8, curr_egc, egc)) { cursor.row = test_cursor.row; cursor.col = test_cursor.col; cursor.target = cursor.col; return; } test_cursor.move_right(root, metrics) catch return error.Stop; } } fn move_cursor_till_char_left_beyond_eol(root: Buffer.Root, cursor: *Cursor, metrics: Buffer.Metrics, ctx: command.Context) error{Stop}!void { var egc: []const u8 = undefined; if (!(ctx.args.match(.{tp.extract(&egc)}) catch return error.Stop)) return error.Stop; var test_cursor = cursor; try test_cursor.move_left(root, metrics); var prev = test_cursor.*; try prev.move_left(root, metrics); while (true) { const prev_egc, _, _ = root.egc_at(prev.row, prev.col, metrics) catch return error.Stop; if (std.mem.eql(u8, prev_egc, egc)) { cursor.row = test_cursor.row; cursor.col = test_cursor.col; cursor.target = cursor.col; return; } test_cursor.move_left(root, metrics) catch return error.Stop; prev.move_left(root, metrics) catch return error.Stop; } } fn move_cursor_till_char_right_beyond_eol(root: Buffer.Root, cursor: *Cursor, metrics: Buffer.Metrics, ctx: command.Context) error{Stop}!void { var egc: []const u8 = undefined; if (!(ctx.args.match(.{tp.extract(&egc)}) catch return error.Stop)) return error.Stop; var test_cursor = cursor; try test_cursor.move_right(root, metrics); var next = test_cursor.*; try next.move_right(root, metrics); while (true) { const next_egc, _, _ = root.egc_at(next.row, next.col, metrics) catch return error.Stop; if (std.mem.eql(u8, next_egc, egc)) { cursor.row = test_cursor.row; cursor.col = test_cursor.col; cursor.target = cursor.col; return; } test_cursor.move_right(root, metrics) catch return error.Stop; next.move_right(root, metrics) catch return error.Stop; } } fn add_cursors_to_cursel_line_ends_helix(ed: *Editor, root: Buffer.Root, cursel: *CurSel) !void { const original_cursel = cursel.*; const sel = cursel.enable_selection(root, ed.metrics); sel.normalize(); var row = sel.begin.row; const is_multiline = sel.begin.row != sel.end.row; var last_cursel: *CurSel = cursel; var first = true; const selection_goes_to_eol = sel.end.col == 0; const end_row = if (selection_goes_to_eol) sel.end.row else (sel.end.row + 1); while (row < end_row) : (row += 1) { const new_cursel = try ed.cursels.addOne(ed.allocator); new_cursel.* = CurSel{ .selection = null, .cursor = .{ .row = row, .col = 0, }, }; if (new_cursel.*) |*the_cursel| { if (!is_eol_right(root, &new_cursel.*.?.cursor, ed.metrics)) { if (first) { first = false; new_cursel.*.?.cursor.col = sel.begin.col; if (is_multiline) { new_cursel.*.?.cursor.col = try root.line_width(row, ed.metrics); new_cursel.*.?.selection = Selection.from_cursor(&new_cursel.*.?.cursor); new_cursel.*.?.selection.?.begin.col = sel.begin.col; new_cursel.*.?.selection.?.end.target = new_cursel.*.?.selection.?.end.col; } else if (original_cursel.selection) |the_sel| { new_cursel.*.?.cursor = original_cursel.cursor; new_cursel.*.?.selection = Selection.from_cursor(&new_cursel.*.?.cursor); new_cursel.*.?.selection = the_sel; } } else { try ed.select_line_at_cursor(root, &new_cursel.*.?, .exclude_eol); new_cursel.*.?.selection.?.end.target = new_cursel.*.?.selection.?.end.col; } new_cursel.*.?.cursor.target = new_cursel.*.?.cursor.col; } last_cursel = the_cursel; } } if (is_multiline and !selection_goes_to_eol) { if (last_cursel.selection) |*sel_| { last_cursel.cursor.col = sel.end.col; sel_.*.end.col = sel.end.col; sel_.*.end.target = sel.end.col; } } } fn insert_before(editor: *Editor, root: Buffer.Root, cursel: *CurSel, text: []const u8, allocator: Allocator) !Buffer.Root { var root_: Buffer.Root = root; const cursor: *Cursor = &cursel.cursor; cursel.check_selection(root, editor.metrics); if (cursel.selection) |sel_| { var sel = sel_; sel.normalize(); cursor.move_to(root, sel.begin.row, sel.begin.col, editor.metrics) catch {}; if (text[text.len - 1] == '\n') { cursor.move_begin(); } } else if (text[text.len - 1] == '\n') { cursor.move_begin(); } cursel.disable_selection_normal(); const begin = cursel.cursor; cursor.row, cursor.col, root_ = try root_.insert_chars(cursor.row, cursor.col, text, allocator, editor.metrics); cursor.target = cursor.col; cursel.selection = Selection{ .begin = begin, .end = cursor.* }; editor.nudge_insert(.{ .begin = begin, .end = cursor.* }, cursel, text.len); return root_; } fn insert_replace_selection(editor: *Editor, root: Buffer.Root, cursel: *CurSel, text: []const u8, allocator: Allocator) !Buffer.Root { // replaces the selection, if no selection, replaces the current // character and sets the selection to the replacement text var root_: Buffer.Root = root; cursel.check_selection(root, editor.metrics); if (cursel.selection == null) { // Select current character to replace it Editor.with_selection_const(root, move_noop, cursel, editor.metrics) catch {}; } root_ = editor.delete_selection(root, cursel, allocator) catch root; const cursor = &cursel.cursor; const begin = cursel.cursor; cursor.row, cursor.col, root_ = try root_.insert_chars(cursor.row, cursor.col, text, allocator, editor.metrics); cursor.target = cursor.col; cursel.selection = Selection{ .begin = begin, .end = cursor.* }; editor.nudge_insert(.{ .begin = begin, .end = cursor.* }, cursel, text.len); return root_; } fn insert_after(editor: *Editor, root: Buffer.Root, cursel: *CurSel, text: []const u8, allocator: Allocator) !Buffer.Root { var root_: Buffer.Root = root; const cursor = &cursel.cursor; cursel.check_selection(root, editor.metrics); if (text[text.len - 1] == '\n') { move_cursor_carriage_return(root, cursel.*, cursor, editor.metrics) catch {}; } else { if (cursel.selection) |sel_| { var sel = sel_; sel.normalize(); cursor.move_to(root, sel.end.row, sel.end.col, editor.metrics) catch {}; } else { cursor.move_right(root_, editor.metrics) catch {}; } } cursel.disable_selection_normal(); const begin = cursel.cursor; cursor.row, cursor.col, root_ = try root_.insert_chars(cursor.row, cursor.col, text, allocator, editor.metrics); cursor.target = cursor.col; cursel.selection = Selection{ .begin = begin, .end = cursor.* }; editor.nudge_insert(.{ .begin = begin, .end = cursor.* }, cursel, text.len); return root_; } fn is_eol_right(root: Buffer.Root, cursor: *const Cursor, metrics: Buffer.Metrics) bool { const line_width = root.line_width(cursor.row, metrics) catch return true; if (cursor.col >= line_width) return true; return false; } fn is_not_whitespace_or_eol(c: []const u8) bool { return !Editor.is_whitespace_or_eol(c); } fn is_whitespace_or_eol_at_cursor(root: Buffer.Root, cursor: *const Cursor, metrics: Buffer.Metrics) bool { return cursor.test_at(root, Editor.is_whitespace_or_eol, metrics); } fn is_non_whitespace_or_eol_at_cursor(root: Buffer.Root, cursor: *const Cursor, metrics: Buffer.Metrics) bool { return cursor.test_at(root, is_not_whitespace_or_eol, metrics); } fn is_long_word_boundary_left(root: Buffer.Root, cursor: *const Cursor, metrics: Buffer.Metrics) bool { if (cursor.test_at(root, Editor.is_whitespace, metrics)) return false; var next = cursor.*; next.move_left(root, metrics) catch return true; const next_is_whitespace = Editor.is_whitespace_at_cursor(root, &next, metrics); if (next_is_whitespace) return true; const curr_is_non_word = is_non_whitespace_or_eol_at_cursor(root, cursor, metrics); const next_is_non_word = is_non_whitespace_or_eol_at_cursor(root, &next, metrics); return curr_is_non_word != next_is_non_word; } fn move_cursor_long_word_left(root: Buffer.Root, cursor: *Cursor, metrics: Buffer.Metrics) error{Stop}!void { try Editor.move_cursor_left(root, cursor, metrics); // Consume " " while (Editor.is_whitespace_at_cursor(root, cursor, metrics)) { try Editor.move_cursor_left(root, cursor, metrics); } var next = cursor.*; next.move_left(root, metrics) catch return; var next_next = next; next_next.move_left(root, metrics) catch return; const cur = next.test_at(root, is_not_whitespace_or_eol, metrics); const nxt = next_next.test_at(root, is_not_whitespace_or_eol, metrics); if (cur != nxt) { try Editor.move_cursor_left(root, cursor, metrics); return; } else { try move_cursor_long_word_left(root, cursor, metrics); } } fn is_word_boundary_right(root: Buffer.Root, cursor: *const Cursor, metrics: Buffer.Metrics) bool { if (Editor.is_whitespace_at_cursor(root, cursor, metrics)) return false; var next = cursor.*; next.move_right(root, metrics) catch return true; const next_is_whitespace = Editor.is_whitespace_at_cursor(root, &next, metrics); if (next_is_whitespace) return true; const curr_is_non_word = is_non_whitespace_or_eol_at_cursor(root, cursor, metrics); const next_is_non_word = is_non_whitespace_or_eol_at_cursor(root, &next, metrics); return curr_is_non_word != next_is_non_word; } fn move_cursor_long_word_right(root: Buffer.Root, cursor: *Cursor, metrics: Buffer.Metrics) error{Stop}!void { try cursor.move_right(root, metrics); Editor.move_cursor_right_until(root, cursor, is_long_word_boundary_left, metrics); } fn is_long_word_boundary_right(root: Buffer.Root, cursor: *const Cursor, metrics: Buffer.Metrics) bool { if (Editor.is_whitespace_at_cursor(root, cursor, metrics)) return false; var next = cursor.*; next.move_right(root, metrics) catch return true; const next_is_whitespace = Editor.is_whitespace_at_cursor(root, &next, metrics); if (next_is_whitespace) return true; const curr_is_non_word = is_non_whitespace_or_eol_at_cursor(root, cursor, metrics); const next_is_non_word = is_non_whitespace_or_eol_at_cursor(root, &next, metrics); return curr_is_non_word != next_is_non_word; } fn move_cursor_long_word_right_end(root: Buffer.Root, cursor: *Cursor, metrics: Buffer.Metrics) error{Stop}!void { // try Editor.move_cursor_right(root, cursor, metrics); Editor.move_cursor_right_until(root, cursor, is_long_word_boundary_right, metrics); try cursor.move_right(root, metrics); } const pasting_function = @TypeOf(insert_before); const find_char_function = @TypeOf(move_cursor_to_char_left_beyond_eol); fn paste_helix(ctx: command.Context, do_paste: pasting_function) command.Result { const mv = tui.mainview() orelse return; const ed = mv.get_active_editor() orelse return; var text_: []const u8 = undefined; const clipboard: []const tui.ClipboardEntry = if (ctx.args.buf.len > 0 and try ctx.args.match(.{tp.extract(&text_)})) &[_]tui.ClipboardEntry{.{ .text = text_ }} else tui.clipboard_get_group(0); if (clipboard.len == 0) { ed.logger.print("paste: nothing to paste", .{}); return; } const b = try ed.buf_for_update(); var root = b.root; // Chunks from clipboard are paired to selections // If more selections than chunks in the clipboard, the exceding selections // use the last chunk in the clipboard var bytes: usize = 0; for (ed.cursels.items, 0..) |*cursel_, idx| if (cursel_.*) |*cursel| { if (idx < clipboard.len) { root = try do_paste(ed, root, cursel, clipboard[idx].text, b.allocator); bytes += clipboard[idx].text.len; } else { bytes += clipboard[clipboard.len - 1].text.len; root = try do_paste(ed, root, cursel, clipboard[clipboard.len - 1].text, b.allocator); } }; ed.logger.print("paste: {d} bytes", .{bytes}); try ed.update_buf(root); ed.clamp(); ed.need_render(); } fn move_cursor_carriage_return(root: Buffer.Root, cursel: CurSel, cursor: *Cursor, metrics: Buffer.Metrics) error{Stop}!void { if (is_cursel_from_extend_line_below(cursel)) { //The cursor is already beginning next line return; } if (!Editor.is_eol_right(root, cursor, metrics)) { try Editor.move_cursor_end(root, cursor, metrics); } try Editor.move_cursor_right(root, cursor, metrics); } fn select_char_if_no_selection(cursel: *CurSel, root: Buffer.Root, metrics: Buffer.Metrics) !bool { if (cursel.selection) |*sel_| { const sel: *Selection = sel_; if (sel.*.empty()) { sel.*.begin = .{ .row = cursel.cursor.row, .col = cursel.cursor.col + 1, .target = cursel.cursor.target + 1 }; return true; } return false; } else { const sel = cursel.enable_selection(root, metrics); sel.begin = .{ .row = cursel.cursor.row, .col = cursel.cursor.col + 1, .target = cursel.cursor.target + 1 }; return true; } } fn is_cursel_from_extend_line_below(cursel: CurSel) bool { if (cursel.selection) |sel_| { var sel = sel_; sel.normalize(); return sel.end.row != sel.begin.row and sel.end.col == 0; } return false; } const private = @This(); // exports for unittests pub const test_internal = struct { pub const move_cursor_long_word_right = private.move_cursor_long_word_right; pub const move_cursor_long_word_left = private.move_cursor_long_word_left; pub const move_cursor_long_word_right_end = private.move_cursor_long_word_right_end; pub const move_cursor_word_left_helix = private.move_cursor_word_left_helix; pub const move_cursor_word_right_end_helix = private.move_cursor_word_right_end_helix; pub const move_cursor_to_char_left_beyond_eol = private.move_cursor_to_char_left_beyond_eol; pub const move_cursor_to_char_right_beyond_eol = private.move_cursor_to_char_right_beyond_eol; pub const move_cursor_till_char_left_beyond_eol = private.move_cursor_till_char_left_beyond_eol; pub const move_cursor_till_char_right_beyond_eol = private.move_cursor_till_char_right_beyond_eol; pub const insert_before = private.insert_before; pub const insert_replace_selection = private.insert_replace_selection; pub const insert_after = private.insert_after; };