refactor: move goto mini mode keybindings to keybind module
This commit is contained in:
parent
16c5471126
commit
49319d6207
5 changed files with 176 additions and 90 deletions
12
build.zig
12
build.zig
|
@ -156,6 +156,17 @@ pub fn build(b: *std.Build) void {
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const keybind_static_mod = b.createModule(.{
|
||||||
|
.root_source_file = b.path("src/keybind/static/root.zig"),
|
||||||
|
.imports = &.{
|
||||||
|
.{ .name = "cbor", .module = cbor_mod },
|
||||||
|
.{ .name = "command", .module = command_mod },
|
||||||
|
.{ .name = "EventHandler", .module = EventHandler_mod },
|
||||||
|
.{ .name = "renderer", .module = renderer_mod },
|
||||||
|
.{ .name = "thespian", .module = thespian_mod },
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const ripgrep_mod = b.createModule(.{
|
const ripgrep_mod = b.createModule(.{
|
||||||
.root_source_file = b.path("src/ripgrep.zig"),
|
.root_source_file = b.path("src/ripgrep.zig"),
|
||||||
.imports = &.{
|
.imports = &.{
|
||||||
|
@ -218,6 +229,7 @@ pub fn build(b: *std.Build) void {
|
||||||
.{ .name = "syntax", .module = syntax_mod },
|
.{ .name = "syntax", .module = syntax_mod },
|
||||||
.{ .name = "text_manip", .module = text_manip_mod },
|
.{ .name = "text_manip", .module = text_manip_mod },
|
||||||
.{ .name = "Buffer", .module = Buffer_mod },
|
.{ .name = "Buffer", .module = Buffer_mod },
|
||||||
|
.{ .name = "keybind", .module = keybind_static_mod },
|
||||||
.{ .name = "ripgrep", .module = ripgrep_mod },
|
.{ .name = "ripgrep", .module = ripgrep_mod },
|
||||||
.{ .name = "theme", .module = themes_dep.module("theme") },
|
.{ .name = "theme", .module = themes_dep.module("theme") },
|
||||||
.{ .name = "themes", .module = themes_dep.module("themes") },
|
.{ .name = "themes", .module = themes_dep.module("themes") },
|
||||||
|
|
72
src/keybind/static/goto.zig
Normal file
72
src/keybind/static/goto.zig
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
const tp = @import("thespian");
|
||||||
|
const key = @import("renderer").input.key;
|
||||||
|
const mod = @import("renderer").input.modifier;
|
||||||
|
const event_type = @import("renderer").input.event_type;
|
||||||
|
const command = @import("command");
|
||||||
|
const EventHandler = @import("EventHandler");
|
||||||
|
|
||||||
|
const Allocator = @import("std").mem.Allocator;
|
||||||
|
const fmt = @import("std").fmt;
|
||||||
|
|
||||||
|
const Mode = @import("root.zig").Mode;
|
||||||
|
|
||||||
|
const name = "#goto";
|
||||||
|
|
||||||
|
pub fn create(_: Allocator) error{OutOfMemory}!Mode {
|
||||||
|
return .{
|
||||||
|
.handler = EventHandler.static(@This()),
|
||||||
|
.name = name,
|
||||||
|
.description = name,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn receive(_: tp.pid_ref, m: tp.message) error{Exit}!bool {
|
||||||
|
var evtype: u32 = undefined;
|
||||||
|
var keypress: u32 = undefined;
|
||||||
|
var modifiers: u32 = undefined;
|
||||||
|
if (try m.match(.{ "I", tp.extract(&evtype), tp.extract(&keypress), tp.any, tp.string, tp.extract(&modifiers) }))
|
||||||
|
try mapEvent(evtype, keypress, modifiers);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn mapEvent(evtype: u32, keypress: u32, modifiers: u32) tp.result {
|
||||||
|
switch (evtype) {
|
||||||
|
event_type.PRESS => try mapPress(keypress, modifiers),
|
||||||
|
event_type.REPEAT => try mapPress(keypress, modifiers),
|
||||||
|
event_type.RELEASE => try mapRelease(keypress, modifiers),
|
||||||
|
else => {},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn mapPress(keypress: u32, modifiers: u32) tp.result {
|
||||||
|
const keynormal = if ('a' <= keypress and keypress <= 'z') keypress - ('a' - 'A') else keypress;
|
||||||
|
return switch (modifiers) {
|
||||||
|
mod.CTRL => switch (keynormal) {
|
||||||
|
'Q' => command.executeName("quit", .{}),
|
||||||
|
'U' => command.executeName("mini_mode_reset", .{}),
|
||||||
|
'G' => command.executeName("mini_mode_cancel", .{}),
|
||||||
|
'C' => command.executeName("mini_mode_cancel", .{}),
|
||||||
|
'L' => command.executeName("scroll_view_center", .{}),
|
||||||
|
key.SPACE => command.executeName("mini_mode_cancel", .{}),
|
||||||
|
else => {},
|
||||||
|
},
|
||||||
|
0 => switch (keypress) {
|
||||||
|
key.LCTRL, key.RCTRL => command.executeName("enable_fast_scroll", .{}),
|
||||||
|
key.LALT, key.RALT => command.executeName("enable_fast_scroll", .{}),
|
||||||
|
key.ESC => command.executeName("mini_mode_cancel", .{}),
|
||||||
|
key.ENTER => command.executeName("exit_mini_mode", .{}),
|
||||||
|
key.BACKSPACE => command.executeName("mini_mode_delete_backwards", .{}),
|
||||||
|
'0'...'9' => command.executeName("mini_mode_insert_code_point", command.fmt(.{keypress})),
|
||||||
|
else => {},
|
||||||
|
},
|
||||||
|
else => {},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
fn mapRelease(keypress: u32, _: u32) tp.result {
|
||||||
|
return switch (keypress) {
|
||||||
|
key.LCTRL, key.RCTRL => command.executeName("disable_fast_scroll", .{}),
|
||||||
|
key.LALT, key.RALT => command.executeName("disable_fast_scroll", .{}),
|
||||||
|
else => {},
|
||||||
|
};
|
||||||
|
}
|
24
src/keybind/static/root.zig
Normal file
24
src/keybind/static/root.zig
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
pub const mode = struct {
|
||||||
|
pub const mini = struct {
|
||||||
|
pub const goto = @import("goto.zig");
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
pub const Mode = struct {
|
||||||
|
handler: EventHandler,
|
||||||
|
name: []const u8,
|
||||||
|
description: []const u8,
|
||||||
|
line_numbers: enum { absolute, relative } = .absolute,
|
||||||
|
keybind_hints: ?*const KeybindHints = null,
|
||||||
|
cursor_shape: renderer.CursorShape = .block,
|
||||||
|
|
||||||
|
pub fn deinit(self: *Mode) void {
|
||||||
|
self.handler.deinit();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
pub const KeybindHints = std.static_string_map.StaticStringMap([]const u8);
|
||||||
|
|
||||||
|
const renderer = @import("renderer");
|
||||||
|
const EventHandler = @import("EventHandler");
|
||||||
|
const std = @import("std");
|
|
@ -3,11 +3,12 @@ const tp = @import("thespian");
|
||||||
const key = @import("renderer").input.key;
|
const key = @import("renderer").input.key;
|
||||||
const mod = @import("renderer").input.modifier;
|
const mod = @import("renderer").input.modifier;
|
||||||
const event_type = @import("renderer").input.event_type;
|
const event_type = @import("renderer").input.event_type;
|
||||||
|
const keybind = @import("keybind");
|
||||||
|
const command = @import("command");
|
||||||
|
const EventHandler = @import("EventHandler");
|
||||||
|
|
||||||
const tui = @import("../../tui.zig");
|
const tui = @import("../../tui.zig");
|
||||||
const mainview = @import("../../mainview.zig");
|
const mainview = @import("../../mainview.zig");
|
||||||
const command = @import("../../command.zig");
|
|
||||||
const EventHandler = @import("../../EventHandler.zig");
|
|
||||||
|
|
||||||
const Allocator = @import("std").mem.Allocator;
|
const Allocator = @import("std").mem.Allocator;
|
||||||
const fmt = @import("std").fmt;
|
const fmt = @import("std").fmt;
|
||||||
|
@ -15,10 +16,13 @@ const fmt = @import("std").fmt;
|
||||||
const Self = @This();
|
const Self = @This();
|
||||||
const name = "#goto";
|
const name = "#goto";
|
||||||
|
|
||||||
|
const Commands = command.Collection(cmds);
|
||||||
|
|
||||||
allocator: Allocator,
|
allocator: Allocator,
|
||||||
buf: [30]u8 = undefined,
|
buf: [30]u8 = undefined,
|
||||||
input: ?usize = null,
|
input: ?usize = null,
|
||||||
start: usize,
|
start: usize,
|
||||||
|
commands: Commands = undefined,
|
||||||
|
|
||||||
pub fn create(allocator: Allocator, _: command.Context) !struct { tui.Mode, tui.MiniMode } {
|
pub fn create(allocator: Allocator, _: command.Context) !struct { tui.Mode, tui.MiniMode } {
|
||||||
const self: *Self = try allocator.create(Self);
|
const self: *Self = try allocator.create(Self);
|
||||||
|
@ -27,100 +31,85 @@ pub fn create(allocator: Allocator, _: command.Context) !struct { tui.Mode, tui.
|
||||||
.allocator = allocator,
|
.allocator = allocator,
|
||||||
.start = editor.get_primary().cursor.row + 1,
|
.start = editor.get_primary().cursor.row + 1,
|
||||||
};
|
};
|
||||||
|
try self.commands.init(self);
|
||||||
return .{
|
return .{
|
||||||
|
try keybind.mode.mini.goto.create(allocator),
|
||||||
.{
|
.{
|
||||||
.handler = EventHandler.to_owned(self),
|
.event_handler = EventHandler.to_owned(self),
|
||||||
.name = name,
|
|
||||||
.description = name,
|
|
||||||
},
|
},
|
||||||
.{},
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
return error.NotFound;
|
return error.NotFound;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn deinit(self: *Self) void {
|
pub fn deinit(self: *Self) void {
|
||||||
|
self.commands.deinit();
|
||||||
self.allocator.destroy(self);
|
self.allocator.destroy(self);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn receive(self: *Self, _: tp.pid_ref, m: tp.message) error{Exit}!bool {
|
pub fn receive(_: *Self, _: tp.pid_ref, _: tp.message) error{Exit}!bool {
|
||||||
var evtype: u32 = undefined;
|
|
||||||
var keypress: u32 = undefined;
|
|
||||||
var modifiers: u32 = undefined;
|
|
||||||
defer {
|
|
||||||
if (tui.current().mini_mode) |*mini_mode| {
|
|
||||||
mini_mode.text = if (self.input) |linenum|
|
|
||||||
(fmt.bufPrint(&self.buf, "{d}", .{linenum}) catch "")
|
|
||||||
else
|
|
||||||
"";
|
|
||||||
mini_mode.cursor = mini_mode.text.len;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (try m.match(.{ "I", tp.extract(&evtype), tp.extract(&keypress), tp.any, tp.string, tp.extract(&modifiers) }))
|
|
||||||
try self.mapEvent(evtype, keypress, modifiers);
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn mapEvent(self: *Self, evtype: u32, keypress: u32, modifiers: u32) tp.result {
|
fn update_mini_mode_text(self: *Self) void {
|
||||||
switch (evtype) {
|
if (tui.current().mini_mode) |*mini_mode| {
|
||||||
event_type.PRESS => try self.mapPress(keypress, modifiers),
|
mini_mode.text = if (self.input) |linenum|
|
||||||
event_type.REPEAT => try self.mapPress(keypress, modifiers),
|
(fmt.bufPrint(&self.buf, "{d}", .{linenum}) catch "")
|
||||||
event_type.RELEASE => try self.mapRelease(keypress, modifiers),
|
else
|
||||||
else => {},
|
"";
|
||||||
|
mini_mode.cursor = mini_mode.text.len;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn mapPress(self: *Self, keypress: u32, modifiers: u32) tp.result {
|
|
||||||
const keynormal = if ('a' <= keypress and keypress <= 'z') keypress - ('a' - 'A') else keypress;
|
|
||||||
return switch (modifiers) {
|
|
||||||
mod.CTRL => switch (keynormal) {
|
|
||||||
'Q' => command.executeName("quit", .{}),
|
|
||||||
'U' => self.input = null,
|
|
||||||
'G' => self.cancel(),
|
|
||||||
'C' => self.cancel(),
|
|
||||||
'L' => command.executeName("scroll_view_center", .{}),
|
|
||||||
key.SPACE => self.cancel(),
|
|
||||||
else => {},
|
|
||||||
},
|
|
||||||
0 => switch (keypress) {
|
|
||||||
key.LCTRL, key.RCTRL => command.executeName("enable_fast_scroll", .{}),
|
|
||||||
key.LALT, key.RALT => command.executeName("enable_fast_scroll", .{}),
|
|
||||||
key.ESC => self.cancel(),
|
|
||||||
key.ENTER => command.executeName("exit_mini_mode", .{}),
|
|
||||||
key.BACKSPACE => if (self.input) |linenum| {
|
|
||||||
const newval = if (linenum < 10) 0 else linenum / 10;
|
|
||||||
self.input = if (newval == 0) null else newval;
|
|
||||||
self.goto();
|
|
||||||
},
|
|
||||||
'0' => {
|
|
||||||
if (self.input) |linenum| self.input = linenum * 10;
|
|
||||||
self.goto();
|
|
||||||
},
|
|
||||||
'1'...'9' => {
|
|
||||||
const digit: usize = @intCast(keypress - '0');
|
|
||||||
self.input = if (self.input) |x| x * 10 + digit else digit;
|
|
||||||
self.goto();
|
|
||||||
},
|
|
||||||
else => {},
|
|
||||||
},
|
|
||||||
else => {},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
fn mapRelease(_: *Self, keypress: u32, _: u32) tp.result {
|
|
||||||
return switch (keypress) {
|
|
||||||
key.LCTRL, key.RCTRL => command.executeName("disable_fast_scroll", .{}),
|
|
||||||
key.LALT, key.RALT => command.executeName("disable_fast_scroll", .{}),
|
|
||||||
else => {},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
fn goto(self: *Self) void {
|
fn goto(self: *Self) void {
|
||||||
command.executeName("goto_line", command.fmt(.{self.input orelse self.start})) catch {};
|
command.executeName("goto_line", command.fmt(.{self.input orelse self.start})) catch {};
|
||||||
}
|
}
|
||||||
|
|
||||||
fn cancel(self: *Self) void {
|
const cmds = struct {
|
||||||
self.input = null;
|
pub const Target = Self;
|
||||||
self.goto();
|
const Ctx = command.Context;
|
||||||
command.executeName("exit_mini_mode", .{}) catch {};
|
const Result = command.Result;
|
||||||
}
|
|
||||||
|
pub fn mini_mode_reset(self: *Self, _: Ctx) Result {
|
||||||
|
self.input = null;
|
||||||
|
self.update_mini_mode_text();
|
||||||
|
}
|
||||||
|
pub const mini_mode_reset_meta = .{ .description = "Clear input" };
|
||||||
|
|
||||||
|
pub fn mini_mode_cancel(self: *Self, _: Ctx) Result {
|
||||||
|
self.input = null;
|
||||||
|
self.update_mini_mode_text();
|
||||||
|
self.goto();
|
||||||
|
command.executeName("exit_mini_mode", .{}) catch {};
|
||||||
|
}
|
||||||
|
pub const mini_mode_cancel_meta = .{ .description = "Cancel input" };
|
||||||
|
|
||||||
|
pub fn mini_mode_delete_backwards(self: *Self, _: Ctx) Result {
|
||||||
|
if (self.input) |linenum| {
|
||||||
|
const newval = if (linenum < 10) 0 else linenum / 10;
|
||||||
|
self.input = if (newval == 0) null else newval;
|
||||||
|
self.update_mini_mode_text();
|
||||||
|
self.goto();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub const mini_mode_delete_backwards_meta = .{ .description = "Delete backwards" };
|
||||||
|
|
||||||
|
pub fn mini_mode_insert_code_point(self: *Self, ctx: Ctx) Result {
|
||||||
|
var keypress: usize = 0;
|
||||||
|
if (!try ctx.args.match(.{tp.extract(&keypress)}))
|
||||||
|
return error.InvalidArgument;
|
||||||
|
switch (keypress) {
|
||||||
|
'0' => {
|
||||||
|
if (self.input) |linenum| self.input = linenum * 10;
|
||||||
|
},
|
||||||
|
'1'...'9' => {
|
||||||
|
const digit: usize = @intCast(keypress - '0');
|
||||||
|
self.input = if (self.input) |x| x * 10 + digit else digit;
|
||||||
|
},
|
||||||
|
else => {},
|
||||||
|
}
|
||||||
|
self.update_mini_mode_text();
|
||||||
|
self.goto();
|
||||||
|
}
|
||||||
|
pub const mini_mode_insert_code_point_meta = .{ .interactive = false };
|
||||||
|
};
|
||||||
|
|
|
@ -10,6 +10,7 @@ const builtin = @import("builtin");
|
||||||
pub const renderer = @import("renderer");
|
pub const renderer = @import("renderer");
|
||||||
const command = @import("command");
|
const command = @import("command");
|
||||||
const EventHandler = @import("EventHandler");
|
const EventHandler = @import("EventHandler");
|
||||||
|
const keybind = @import("keybind");
|
||||||
|
|
||||||
const Widget = @import("Widget.zig");
|
const Widget = @import("Widget.zig");
|
||||||
const MessageFilter = @import("MessageFilter.zig");
|
const MessageFilter = @import("MessageFilter.zig");
|
||||||
|
@ -755,25 +756,13 @@ const cmds = struct {
|
||||||
pub const exit_mini_mode_meta = .{ .interactive = false };
|
pub const exit_mini_mode_meta = .{ .interactive = false };
|
||||||
};
|
};
|
||||||
|
|
||||||
pub const Mode = struct {
|
|
||||||
handler: EventHandler,
|
|
||||||
name: []const u8,
|
|
||||||
description: []const u8,
|
|
||||||
line_numbers: enum { absolute, relative } = .absolute,
|
|
||||||
keybind_hints: ?*const KeybindHints = null,
|
|
||||||
cursor_shape: renderer.CursorShape = .block,
|
|
||||||
|
|
||||||
fn deinit(self: *Mode) void {
|
|
||||||
self.handler.deinit();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
pub const MiniMode = struct {
|
pub const MiniMode = struct {
|
||||||
event_handler: ?EventHandler = null,
|
event_handler: ?EventHandler = null,
|
||||||
text: []const u8 = "",
|
text: []const u8 = "",
|
||||||
cursor: ?usize = null,
|
cursor: ?usize = null,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
pub const Mode = keybind.Mode;
|
||||||
pub const KeybindHints = std.static_string_map.StaticStringMap([]const u8);
|
pub const KeybindHints = std.static_string_map.StaticStringMap([]const u8);
|
||||||
|
|
||||||
threadlocal var instance_: ?*Self = null;
|
threadlocal var instance_: ?*Self = null;
|
||||||
|
|
Loading…
Add table
Reference in a new issue