feat: add mouse resizing of panel views
This commit is contained in:
parent
95bf2ac9a7
commit
50d47a73f2
15 changed files with 138 additions and 65 deletions
|
@ -23,6 +23,7 @@ pub fn Options(context: type) type {
|
||||||
on_render: *const fn (ctx: *context, button: *State(Context), theme: *const Widget.Theme) bool = on_render_default,
|
on_render: *const fn (ctx: *context, button: *State(Context), theme: *const Widget.Theme) bool = on_render_default,
|
||||||
on_layout: *const fn (ctx: *context, button: *State(Context)) Widget.Layout = on_layout_default,
|
on_layout: *const fn (ctx: *context, button: *State(Context)) Widget.Layout = on_layout_default,
|
||||||
on_receive: *const fn (ctx: *context, button: *State(Context), from: tp.pid_ref, m: tp.message) error{Exit}!bool = on_receive_default,
|
on_receive: *const fn (ctx: *context, button: *State(Context), from: tp.pid_ref, m: tp.message) error{Exit}!bool = on_receive_default,
|
||||||
|
on_event: ?Widget.EventHandler = null,
|
||||||
|
|
||||||
pub const Context = context;
|
pub const Context = context;
|
||||||
pub fn do_nothing(_: *context, _: *State(Context)) void {}
|
pub fn do_nothing(_: *context, _: *State(Context)) void {}
|
||||||
|
@ -109,7 +110,17 @@ pub fn State(ctx_type: type) type {
|
||||||
self.call_click_handler(btn);
|
self.call_click_handler(btn);
|
||||||
tui.need_render();
|
tui.need_render();
|
||||||
return true;
|
return true;
|
||||||
|
} else if (try m.match(.{ "D", event_type.PRESS, tp.extract(&btn), tp.more })) {
|
||||||
|
if (self.opts.on_event) |h| {
|
||||||
|
self.active = false;
|
||||||
|
h.send(from, m) catch {};
|
||||||
|
}
|
||||||
|
return true;
|
||||||
} else if (try m.match(.{ "D", event_type.RELEASE, tp.extract(&btn), tp.more })) {
|
} else if (try m.match(.{ "D", event_type.RELEASE, tp.extract(&btn), tp.more })) {
|
||||||
|
if (self.opts.on_event) |h| {
|
||||||
|
self.active = false;
|
||||||
|
h.send(from, m) catch {};
|
||||||
|
}
|
||||||
self.call_click_handler(btn);
|
self.call_click_handler(btn);
|
||||||
tui.need_render();
|
tui.need_render();
|
||||||
return true;
|
return true;
|
||||||
|
@ -122,7 +133,10 @@ pub fn State(ctx_type: type) type {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn call_click_handler(self: *Self, btn: u32) void {
|
fn call_click_handler(self: *Self, btn: u32) void {
|
||||||
if (btn == key.BUTTON1) self.active = false;
|
if (btn == key.BUTTON1) {
|
||||||
|
if (!self.active) return;
|
||||||
|
self.active = false;
|
||||||
|
}
|
||||||
if (!self.hover) return;
|
if (!self.hover) return;
|
||||||
switch (btn) {
|
switch (btn) {
|
||||||
key.BUTTON1 => self.opts.on_click(&self.opts.ctx, self),
|
key.BUTTON1 => self.opts.on_click(&self.opts.ctx, self),
|
||||||
|
|
|
@ -49,6 +49,7 @@ pub const VTable = struct {
|
||||||
unsubscribe: *const fn (ctx: *anyopaque, h: EventHandler) error{NotSupported}!void,
|
unsubscribe: *const fn (ctx: *anyopaque, h: EventHandler) error{NotSupported}!void,
|
||||||
get: *const fn (ctx: *anyopaque, name_: []const u8) ?*Self,
|
get: *const fn (ctx: *anyopaque, name_: []const u8) ?*Self,
|
||||||
walk: *const fn (ctx: *anyopaque, walk_ctx: *anyopaque, f: WalkFn, self_widget: *Self) bool,
|
walk: *const fn (ctx: *anyopaque, walk_ctx: *anyopaque, f: WalkFn, self_widget: *Self) bool,
|
||||||
|
hover: *const fn (ctx: *anyopaque) bool,
|
||||||
type_name: []const u8,
|
type_name: []const u8,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -136,6 +137,11 @@ pub fn to(pimpl: anytype) Self {
|
||||||
return if (comptime @hasDecl(child, "walk")) child.walk(@as(*child, @ptrCast(@alignCast(ctx))), walk_ctx, f, self) else false;
|
return if (comptime @hasDecl(child, "walk")) child.walk(@as(*child, @ptrCast(@alignCast(ctx))), walk_ctx, f, self) else false;
|
||||||
}
|
}
|
||||||
}.walk,
|
}.walk,
|
||||||
|
.hover = struct {
|
||||||
|
pub fn hover(ctx: *anyopaque) bool {
|
||||||
|
return if (comptime @hasField(child, "hover")) @as(*child, @ptrCast(@alignCast(ctx))).hover else false;
|
||||||
|
}
|
||||||
|
}.hover,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -211,6 +217,10 @@ pub fn walk(self: *Self, walk_ctx: *anyopaque, f: WalkFn) bool {
|
||||||
return if (self.vtable.walk(self.ptr, walk_ctx, f, self)) true else f(walk_ctx, self);
|
return if (self.vtable.walk(self.ptr, walk_ctx, f, self)) true else f(walk_ctx, self);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn hover(self: *Self) bool {
|
||||||
|
return self.vtable.hover(self.ptr);
|
||||||
|
}
|
||||||
|
|
||||||
pub fn empty(a: Allocator, parent: Plane, layout_: Layout) !Self {
|
pub fn empty(a: Allocator, parent: Plane, layout_: Layout) !Self {
|
||||||
const child: type = struct { plane: Plane, layout: Layout };
|
const child: type = struct { plane: Plane, layout: Layout };
|
||||||
const widget = try a.create(child);
|
const widget = try a.create(child);
|
||||||
|
@ -270,6 +280,11 @@ pub fn empty(a: Allocator, parent: Plane, layout_: Layout) !Self {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}.walk,
|
}.walk,
|
||||||
|
.hover = struct {
|
||||||
|
pub fn hover(_: *anyopaque) bool {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}.hover,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,6 +28,7 @@ direction: Direction,
|
||||||
box: ?Widget.Box = null,
|
box: ?Widget.Box = null,
|
||||||
ctx: ?*anyopaque = null,
|
ctx: ?*anyopaque = null,
|
||||||
on_render: *const fn (ctx: ?*anyopaque, theme: *const Widget.Theme) void = on_render_default,
|
on_render: *const fn (ctx: ?*anyopaque, theme: *const Widget.Theme) void = on_render_default,
|
||||||
|
after_render: *const fn (ctx: ?*anyopaque, theme: *const Widget.Theme) void = on_render_default,
|
||||||
on_resize: *const fn (ctx: ?*anyopaque, self: *Self, pos_: Widget.Box) void = on_resize_default,
|
on_resize: *const fn (ctx: ?*anyopaque, self: *Self, pos_: Widget.Box) void = on_resize_default,
|
||||||
|
|
||||||
pub fn createH(a: Allocator, parent: Widget, name: [:0]const u8, layout_: Layout) !*Self {
|
pub fn createH(a: Allocator, parent: Widget, name: [:0]const u8, layout_: Layout) !*Self {
|
||||||
|
@ -135,6 +136,8 @@ pub fn render(self: *Self, theme: *const Widget.Theme) bool {
|
||||||
if (w.widget.render(theme)) {
|
if (w.widget.render(theme)) {
|
||||||
more = true;
|
more = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
self.after_render(self.ctx, theme);
|
||||||
return more;
|
return more;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -190,6 +193,8 @@ fn on_resize_default(_: ?*anyopaque, self: *Self, pos: Widget.Box) void {
|
||||||
pub fn resize(self: *Self, pos_: Widget.Box) void {
|
pub fn resize(self: *Self, pos_: Widget.Box) void {
|
||||||
self.box = pos_;
|
self.box = pos_;
|
||||||
var pos = pos_;
|
var pos = pos_;
|
||||||
|
self.plane.move_yx(@intCast(pos.y), @intCast(pos.x)) catch return;
|
||||||
|
self.plane.resize_simple(@intCast(pos.h), @intCast(pos.w)) catch return;
|
||||||
const total = self.get_size_a(&pos).*;
|
const total = self.get_size_a(&pos).*;
|
||||||
var avail = total;
|
var avail = total;
|
||||||
var statics: usize = 0;
|
var statics: usize = 0;
|
||||||
|
@ -243,3 +248,8 @@ pub fn walk(self: *Self, ctx: *anyopaque, f: Widget.WalkFn, self_w: *Widget) boo
|
||||||
if (w.widget.walk(ctx, f)) return true;
|
if (w.widget.walk(ctx, f)) return true;
|
||||||
return f(ctx, self_w);
|
return f(ctx, self_w);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn hover(self: *Self) bool {
|
||||||
|
for (self.widgets.items) |*w| if (w.widget.hover()) return true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
|
@ -140,9 +140,6 @@ fn handle_render_menu(self: *Self, button: *Button.State(*Menu.State(*Self)), th
|
||||||
const style_info: Widget.Theme.Style = .{ .fg = theme.editor_information.fg, .fs = theme.editor_information.fs, .bg = style_base.bg };
|
const style_info: Widget.Theme.Style = .{ .fg = theme.editor_information.fg, .fs = theme.editor_information.fs, .bg = style_base.bg };
|
||||||
const style_separator: Widget.Theme.Style = .{ .fg = theme.editor_selection.bg, .bg = style_base.bg };
|
const style_separator: Widget.Theme.Style = .{ .fg = theme.editor_selection.bg, .bg = style_base.bg };
|
||||||
// const style_error: Widget.Theme.Style = .{ .fg = theme.editor_error.fg, .fs = theme.editor_error.fs, .bg = style_base.bg };
|
// const style_error: Widget.Theme.Style = .{ .fg = theme.editor_error.fg, .fs = theme.editor_error.fs, .bg = style_base.bg };
|
||||||
button.plane.set_base_style(" ", style_base);
|
|
||||||
button.plane.erase();
|
|
||||||
button.plane.home();
|
|
||||||
var idx: usize = undefined;
|
var idx: usize = undefined;
|
||||||
var iter = button.opts.label; // label contains cbor, just the index
|
var iter = button.opts.label; // label contains cbor, just the index
|
||||||
if (!(cbor.matchValue(&iter, cbor.extract(&idx)) catch false)) {
|
if (!(cbor.matchValue(&iter, cbor.extract(&idx)) catch false)) {
|
||||||
|
@ -155,6 +152,9 @@ fn handle_render_menu(self: *Self, button: *Button.State(*Menu.State(*Self)), th
|
||||||
if (idx >= self.entries.items.len) {
|
if (idx >= self.entries.items.len) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
button.plane.set_base_style(" ", style_base);
|
||||||
|
button.plane.erase();
|
||||||
|
button.plane.home();
|
||||||
const entry = &self.entries.items[idx];
|
const entry = &self.entries.items[idx];
|
||||||
const pointer = if (selected) "⏵" else " ";
|
const pointer = if (selected) "⏵" else " ";
|
||||||
_ = button.plane.print("{s} ", .{pointer}) catch {};
|
_ = button.plane.print("{s} ", .{pointer}) catch {};
|
||||||
|
@ -229,10 +229,7 @@ fn handle_menu_action(menu: **Menu.State(*Self), button: *Button.State(*Menu.Sta
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
idx += self.view_pos;
|
idx += self.view_pos;
|
||||||
if (idx >= self.entries.items.len) {
|
if (idx >= self.entries.items.len) return;
|
||||||
self.logger.print_err(name, "table entry index out of range: {d}/{d}", .{ idx, self.entries.items.len });
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
self.selected = idx;
|
self.selected = idx;
|
||||||
self.update_selected();
|
self.update_selected();
|
||||||
const entry = &self.entries.items[idx];
|
const entry = &self.entries.items[idx];
|
||||||
|
|
|
@ -8,6 +8,8 @@ const location_history = @import("location_history");
|
||||||
const project_manager = @import("project_manager");
|
const project_manager = @import("project_manager");
|
||||||
|
|
||||||
const Plane = @import("renderer").Plane;
|
const Plane = @import("renderer").Plane;
|
||||||
|
const key = @import("renderer").input.key;
|
||||||
|
const event_type = @import("renderer").input.event_type;
|
||||||
|
|
||||||
const tui = @import("tui.zig");
|
const tui = @import("tui.zig");
|
||||||
const command = @import("command.zig");
|
const command = @import("command.zig");
|
||||||
|
@ -35,6 +37,7 @@ last_match_text: ?[]const u8 = null,
|
||||||
location_history: location_history,
|
location_history: location_history,
|
||||||
file_stack: std.ArrayList([]const u8),
|
file_stack: std.ArrayList([]const u8),
|
||||||
find_in_files_done: bool = false,
|
find_in_files_done: bool = false,
|
||||||
|
panel_height: ?usize = null,
|
||||||
|
|
||||||
const NavState = struct {
|
const NavState = struct {
|
||||||
time: i64 = 0,
|
time: i64 = 0,
|
||||||
|
@ -45,11 +48,11 @@ const NavState = struct {
|
||||||
matches: usize = 0,
|
matches: usize = 0,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub fn create(a: std.mem.Allocator, n: Plane) !Widget {
|
pub fn create(a: std.mem.Allocator) !Widget {
|
||||||
const self = try a.create(Self);
|
const self = try a.create(Self);
|
||||||
self.* = .{
|
self.* = .{
|
||||||
.a = a,
|
.a = a,
|
||||||
.plane = n,
|
.plane = tui.current().stdplane(),
|
||||||
.widgets = undefined,
|
.widgets = undefined,
|
||||||
.widgets_widget = undefined,
|
.widgets_widget = undefined,
|
||||||
.floating_views = WidgetStack.init(a),
|
.floating_views = WidgetStack.init(a),
|
||||||
|
@ -62,8 +65,8 @@ pub fn create(a: std.mem.Allocator, n: Plane) !Widget {
|
||||||
const widgets = try WidgetList.createV(a, w, @typeName(Self), .dynamic);
|
const widgets = try WidgetList.createV(a, w, @typeName(Self), .dynamic);
|
||||||
self.widgets = widgets;
|
self.widgets = widgets;
|
||||||
self.widgets_widget = widgets.widget();
|
self.widgets_widget = widgets.widget();
|
||||||
try widgets.add(try Widget.empty(a, n, .dynamic));
|
try widgets.add(try Widget.empty(a, self.widgets_widget.plane.*, .dynamic));
|
||||||
self.statusbar = try widgets.addP(try @import("status/statusbar.zig").create(a, w));
|
self.statusbar = try widgets.addP(try @import("status/statusbar.zig").create(a, w, EventHandler.bind(self, handle_statusbar_event)));
|
||||||
if (tp.env.get().is("show-input"))
|
if (tp.env.get().is("show-input"))
|
||||||
self.toggle_inputview_async();
|
self.toggle_inputview_async();
|
||||||
if (tp.env.get().is("show-log"))
|
if (tp.env.get().is("show-log"))
|
||||||
|
@ -125,6 +128,10 @@ pub fn render(self: *Self, theme: *const Widget.Theme) bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn handle_resize(self: *Self, pos: Box) void {
|
pub fn handle_resize(self: *Self, pos: Box) void {
|
||||||
|
self.plane = tui.current().stdplane();
|
||||||
|
if (self.panel_height) |h| if (h >= self.box().h) {
|
||||||
|
self.panel_height = null;
|
||||||
|
};
|
||||||
self.widgets.handle_resize(pos);
|
self.widgets.handle_resize(pos);
|
||||||
self.floating_views.resize(pos);
|
self.floating_views.resize(pos);
|
||||||
}
|
}
|
||||||
|
@ -133,6 +140,22 @@ pub fn box(self: *const Self) Box {
|
||||||
return Box.from(self.plane);
|
return Box.from(self.plane);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn handle_statusbar_event(self: *Self, _: tp.pid_ref, m: tp.message) tp.result {
|
||||||
|
var y: usize = undefined;
|
||||||
|
if (try m.match(.{ "D", event_type.PRESS, key.BUTTON1, tp.any, tp.any, tp.extract(&y), tp.any, tp.any }))
|
||||||
|
return self.statusbar_primary_drag(y);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn statusbar_primary_drag(self: *Self, y: usize) tp.result {
|
||||||
|
const panels = self.panels orelse blk: {
|
||||||
|
cmds.toggle_panel(self, .{}) catch return;
|
||||||
|
break :blk self.panels.?;
|
||||||
|
};
|
||||||
|
const h = self.plane.dim_y();
|
||||||
|
self.panel_height = @max(2, h - @min(h, y + 1));
|
||||||
|
panels.layout = .{ .static = self.panel_height.? };
|
||||||
|
}
|
||||||
|
|
||||||
fn toggle_panel_view(self: *Self, view: anytype, enable_only: bool) !bool {
|
fn toggle_panel_view(self: *Self, view: anytype, enable_only: bool) !bool {
|
||||||
var enabled = true;
|
var enabled = true;
|
||||||
if (self.panels) |panels| {
|
if (self.panels) |panels| {
|
||||||
|
@ -149,7 +172,7 @@ fn toggle_panel_view(self: *Self, view: anytype, enable_only: bool) !bool {
|
||||||
try panels.add(try view.create(self.a, self.widgets.plane));
|
try panels.add(try view.create(self.a, self.widgets.plane));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const panels = try WidgetList.createH(self.a, self.widgets.widget(), "panel", .{ .static = self.box().h / 5 });
|
const panels = try WidgetList.createH(self.a, self.widgets.widget(), "panel", .{ .static = self.panel_height orelse self.box().h / 5 });
|
||||||
try self.widgets.add(panels.widget());
|
try self.widgets.add(panels.widget());
|
||||||
try panels.add(try view.create(self.a, self.widgets.plane));
|
try panels.add(try view.create(self.a, self.widgets.plane));
|
||||||
self.panels = panels;
|
self.panels = panels;
|
||||||
|
|
|
@ -19,7 +19,7 @@ rendered: [:0]const u8 = "",
|
||||||
|
|
||||||
const Self = @This();
|
const Self = @This();
|
||||||
|
|
||||||
pub fn create(a: Allocator, parent: Plane) !Widget {
|
pub fn create(a: Allocator, parent: Plane, event_handler: ?Widget.EventHandler) !Widget {
|
||||||
return Button.create_widget(Self, a, parent, .{
|
return Button.create_widget(Self, a, parent, .{
|
||||||
.ctx = .{},
|
.ctx = .{},
|
||||||
.label = "",
|
.label = "",
|
||||||
|
@ -27,6 +27,7 @@ pub fn create(a: Allocator, parent: Plane) !Widget {
|
||||||
.on_layout = layout,
|
.on_layout = layout,
|
||||||
.on_render = render,
|
.on_render = render,
|
||||||
.on_receive = receive,
|
.on_receive = receive,
|
||||||
|
.on_event = event_handler,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -34,7 +34,7 @@ file: bool = false,
|
||||||
const project_icon = "";
|
const project_icon = "";
|
||||||
const Self = @This();
|
const Self = @This();
|
||||||
|
|
||||||
pub fn create(a: Allocator, parent: Plane) !Widget {
|
pub fn create(a: Allocator, parent: Plane, event_handler: ?Widget.EventHandler) !Widget {
|
||||||
const btn = try Button.create(Self, a, parent, .{
|
const btn = try Button.create(Self, a, parent, .{
|
||||||
.ctx = .{
|
.ctx = .{
|
||||||
.a = a,
|
.a = a,
|
||||||
|
@ -52,6 +52,7 @@ pub fn create(a: Allocator, parent: Plane) !Widget {
|
||||||
.on_layout = layout,
|
.on_layout = layout,
|
||||||
.on_render = render,
|
.on_render = render,
|
||||||
.on_receive = receive,
|
.on_receive = receive,
|
||||||
|
.on_event = event_handler,
|
||||||
});
|
});
|
||||||
return Widget.to(btn);
|
return Widget.to(btn);
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,7 +15,6 @@ const EventHandler = @import("../EventHandler.zig");
|
||||||
|
|
||||||
const history = 8;
|
const history = 8;
|
||||||
|
|
||||||
parent: Plane,
|
|
||||||
plane: Plane,
|
plane: Plane,
|
||||||
frame: u64 = 0,
|
frame: u64 = 0,
|
||||||
idle_frame: u64 = 0,
|
idle_frame: u64 = 0,
|
||||||
|
@ -33,23 +32,15 @@ const idle_msg = "🐶";
|
||||||
pub const width = idle_msg.len + 20;
|
pub const width = idle_msg.len + 20;
|
||||||
|
|
||||||
pub fn create(a: Allocator, parent: Plane) !Widget {
|
pub fn create(a: Allocator, parent: Plane) !Widget {
|
||||||
const self: *Self = try a.create(Self);
|
|
||||||
self.* = try init(parent);
|
|
||||||
try tui.current().input_listeners.add(EventHandler.bind(self, listen));
|
|
||||||
return self.widget();
|
|
||||||
}
|
|
||||||
|
|
||||||
fn init(parent: Plane) !Self {
|
|
||||||
var n = try Plane.init(&(Widget.Box{}).opts(@typeName(Self)), parent);
|
|
||||||
errdefer n.deinit();
|
|
||||||
var frame_rate = tp.env.get().num("frame-rate");
|
var frame_rate = tp.env.get().num("frame-rate");
|
||||||
if (frame_rate == 0) frame_rate = 60;
|
if (frame_rate == 0) frame_rate = 60;
|
||||||
|
const self: *Self = try a.create(Self);
|
||||||
return .{
|
self.* = .{
|
||||||
.parent = parent,
|
.plane = try Plane.init(&(Widget.Box{}).opts(@typeName(Self)), parent),
|
||||||
.plane = n,
|
|
||||||
.wipe_after_frames = @divTrunc(frame_rate, 2),
|
.wipe_after_frames = @divTrunc(frame_rate, 2),
|
||||||
};
|
};
|
||||||
|
try tui.current().input_listeners.add(EventHandler.bind(self, listen));
|
||||||
|
return self.widget();
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn widget(self: *Self) Widget {
|
pub fn widget(self: *Self) Widget {
|
||||||
|
|
|
@ -18,7 +18,7 @@ rendered: [:0]const u8 = "",
|
||||||
|
|
||||||
const Self = @This();
|
const Self = @This();
|
||||||
|
|
||||||
pub fn create(a: Allocator, parent: Plane) !Widget {
|
pub fn create(a: Allocator, parent: Plane, event_handler: ?Widget.EventHandler) !Widget {
|
||||||
return Button.create_widget(Self, a, parent, .{
|
return Button.create_widget(Self, a, parent, .{
|
||||||
.ctx = .{},
|
.ctx = .{},
|
||||||
.label = "",
|
.label = "",
|
||||||
|
@ -26,6 +26,7 @@ pub fn create(a: Allocator, parent: Plane) !Widget {
|
||||||
.on_layout = layout,
|
.on_layout = layout,
|
||||||
.on_render = render,
|
.on_render = render,
|
||||||
.on_receive = receive,
|
.on_receive = receive,
|
||||||
|
.on_event = event_handler,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -10,12 +10,12 @@ const tui = @import("../tui.zig");
|
||||||
const mainview = @import("../mainview.zig");
|
const mainview = @import("../mainview.zig");
|
||||||
const logview = @import("../logview.zig");
|
const logview = @import("../logview.zig");
|
||||||
|
|
||||||
parent: Plane,
|
|
||||||
plane: Plane,
|
plane: Plane,
|
||||||
msg: std.ArrayList(u8),
|
msg: std.ArrayList(u8),
|
||||||
msg_counter: usize = 0,
|
msg_counter: usize = 0,
|
||||||
clear_timer: ?tp.Cancellable = null,
|
clear_timer: ?tp.Cancellable = null,
|
||||||
level: Level = .info,
|
level: Level = .info,
|
||||||
|
on_event: ?Widget.EventHandler,
|
||||||
|
|
||||||
const message_display_time_seconds = 2;
|
const message_display_time_seconds = 2;
|
||||||
const error_display_time_seconds = 4;
|
const error_display_time_seconds = 4;
|
||||||
|
@ -26,12 +26,12 @@ const Level = enum {
|
||||||
err,
|
err,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub fn create(a: std.mem.Allocator, parent: Plane) !Widget {
|
pub fn create(a: std.mem.Allocator, parent: Plane, event_handler: ?Widget.EventHandler) !Widget {
|
||||||
const self: *Self = try a.create(Self);
|
const self: *Self = try a.create(Self);
|
||||||
self.* = .{
|
self.* = .{
|
||||||
.parent = parent,
|
|
||||||
.plane = try Plane.init(&(Widget.Box{}).opts(@typeName(Self)), parent),
|
.plane = try Plane.init(&(Widget.Box{}).opts(@typeName(Self)), parent),
|
||||||
.msg = std.ArrayList(u8).init(a),
|
.msg = std.ArrayList(u8).init(a),
|
||||||
|
.on_event = event_handler,
|
||||||
};
|
};
|
||||||
logview.init(a);
|
logview.init(a);
|
||||||
try tui.current().message_filters.add(MessageFilter.bind(self, receive_log));
|
try tui.current().message_filters.add(MessageFilter.bind(self, receive_log));
|
||||||
|
@ -52,6 +52,15 @@ pub fn deinit(self: *Self, a: std.mem.Allocator) void {
|
||||||
a.destroy(self);
|
a.destroy(self);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn receive(self: *Self, from: tp.pid_ref, m: tp.message) error{Exit}!bool {
|
||||||
|
var btn: u32 = 0;
|
||||||
|
if (try m.match(.{ "D", tp.any, tp.extract(&btn), tp.more })) {
|
||||||
|
if (self.on_event) |h| h.send(from, m) catch {};
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
pub fn layout(self: *Self) Widget.Layout {
|
pub fn layout(self: *Self) Widget.Layout {
|
||||||
return .{ .static = if (self.msg.items.len > 0) self.msg.items.len + 2 else 1 };
|
return .{ .static = if (self.msg.items.len > 0) self.msg.items.len + 2 else 1 };
|
||||||
}
|
}
|
||||||
|
@ -92,7 +101,7 @@ fn process_log(self: *Self, m: tp.message) !void {
|
||||||
const err_stop = "error.Stop";
|
const err_stop = "error.Stop";
|
||||||
if (std.mem.eql(u8, msg, err_stop))
|
if (std.mem.eql(u8, msg, err_stop))
|
||||||
return;
|
return;
|
||||||
if (msg.len >= err_stop.len + 1 and std.mem.eql(u8, msg[0..err_stop.len + 1], err_stop ++ "\n"))
|
if (msg.len >= err_stop.len + 1 and std.mem.eql(u8, msg[0 .. err_stop.len + 1], err_stop ++ "\n"))
|
||||||
return;
|
return;
|
||||||
try self.set(msg, .err);
|
try self.set(msg, .err);
|
||||||
} else if (try m.match(.{ "log", tp.extract(&src), tp.more })) {
|
} else if (try m.match(.{ "log", tp.extract(&src), tp.more })) {
|
||||||
|
|
|
@ -14,13 +14,14 @@ const command = @import("../command.zig");
|
||||||
const ed = @import("../editor.zig");
|
const ed = @import("../editor.zig");
|
||||||
const tui = @import("../tui.zig");
|
const tui = @import("../tui.zig");
|
||||||
|
|
||||||
pub fn create(a: Allocator, parent: Plane) !Widget {
|
pub fn create(a: Allocator, parent: Plane, event_handler: ?Widget.EventHandler) !Widget {
|
||||||
return Button.create_widget(void, a, parent, .{
|
return Button.create_widget(void, a, parent, .{
|
||||||
.ctx = {},
|
.ctx = {},
|
||||||
.label = tui.get_mode(),
|
.label = tui.get_mode(),
|
||||||
.on_click = on_click,
|
.on_click = on_click,
|
||||||
.on_layout = layout,
|
.on_layout = layout,
|
||||||
.on_render = render,
|
.on_render = render,
|
||||||
|
.on_event = event_handler,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -25,19 +25,13 @@ pub const width = 5;
|
||||||
|
|
||||||
pub fn create(a: Allocator, parent: Plane) !Widget {
|
pub fn create(a: Allocator, parent: Plane) !Widget {
|
||||||
const self: *Self = try a.create(Self);
|
const self: *Self = try a.create(Self);
|
||||||
self.* = try init(parent);
|
self.* = .{
|
||||||
|
.plane = try Plane.init(&(Widget.Box{}).opts(@typeName(Self)), parent),
|
||||||
|
};
|
||||||
try tui.current().input_listeners.add(EventHandler.bind(self, listen));
|
try tui.current().input_listeners.add(EventHandler.bind(self, listen));
|
||||||
return self.widget();
|
return self.widget();
|
||||||
}
|
}
|
||||||
|
|
||||||
fn init(parent: Plane) !Self {
|
|
||||||
var n = try Plane.init(&(Widget.Box{}).opts(@typeName(Self)), parent);
|
|
||||||
errdefer n.deinit();
|
|
||||||
return .{
|
|
||||||
.plane = n,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn widget(self: *Self) Widget {
|
pub fn widget(self: *Self) Widget {
|
||||||
return Widget.to(self);
|
return Widget.to(self);
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,22 +15,17 @@ cursels: usize = 0,
|
||||||
selection: ?ed.Selection = null,
|
selection: ?ed.Selection = null,
|
||||||
buf: [256]u8 = undefined,
|
buf: [256]u8 = undefined,
|
||||||
rendered: [:0]const u8 = "",
|
rendered: [:0]const u8 = "",
|
||||||
|
on_event: ?Widget.EventHandler,
|
||||||
|
|
||||||
const Self = @This();
|
const Self = @This();
|
||||||
|
|
||||||
pub fn create(a: Allocator, parent: Plane) !Widget {
|
pub fn create(a: Allocator, parent: Plane, event_handler: ?Widget.EventHandler) !Widget {
|
||||||
const self: *Self = try a.create(Self);
|
const self: *Self = try a.create(Self);
|
||||||
self.* = try init(parent);
|
self.* = .{
|
||||||
return Widget.to(self);
|
.plane = try Plane.init(&(Widget.Box{}).opts(@typeName(Self)), parent),
|
||||||
}
|
.on_event = event_handler,
|
||||||
|
|
||||||
fn init(parent: Plane) !Self {
|
|
||||||
var n = try Plane.init(&(Widget.Box{}).opts(@typeName(Self)), parent);
|
|
||||||
errdefer n.deinit();
|
|
||||||
|
|
||||||
return .{
|
|
||||||
.plane = n,
|
|
||||||
};
|
};
|
||||||
|
return Widget.to(self);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn deinit(self: *Self, a: Allocator) void {
|
pub fn deinit(self: *Self, a: Allocator) void {
|
||||||
|
@ -81,7 +76,12 @@ fn format(self: *Self) void {
|
||||||
self.buf[self.rendered.len] = 0;
|
self.buf[self.rendered.len] = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn receive(self: *Self, _: tp.pid_ref, m: tp.message) error{Exit}!bool {
|
pub fn receive(self: *Self, from: tp.pid_ref, m: tp.message) error{Exit}!bool {
|
||||||
|
var btn: u32 = 0;
|
||||||
|
if (try m.match(.{ "D", tp.any, tp.extract(&btn), tp.more })) {
|
||||||
|
if (self.on_event) |h| h.send(from, m) catch {};
|
||||||
|
return true;
|
||||||
|
}
|
||||||
if (try m.match(.{ "E", "match", tp.extract(&self.matches) }))
|
if (try m.match(.{ "E", "match", tp.extract(&self.matches) }))
|
||||||
self.format();
|
self.format();
|
||||||
if (try m.match(.{ "E", "cursels", tp.extract(&self.cursels) }))
|
if (try m.match(.{ "E", "cursels", tp.extract(&self.cursels) }))
|
||||||
|
|
|
@ -6,15 +6,28 @@ const tui = @import("../tui.zig");
|
||||||
|
|
||||||
const Self = @This();
|
const Self = @This();
|
||||||
|
|
||||||
pub fn create(a: std.mem.Allocator, parent: Widget) !Widget {
|
pub fn create(a: std.mem.Allocator, parent: Widget, event_handler: ?Widget.EventHandler) !Widget {
|
||||||
var w = try WidgetList.createH(a, parent, "statusbar", .{ .static = 1 });
|
var w = try WidgetList.createH(a, parent, "statusbar", .{ .static = 1 });
|
||||||
if (tui.current().config.modestate_show) try w.add(try @import("modestate.zig").create(a, w.plane));
|
w.after_render = render_grip;
|
||||||
try w.add(try @import("filestate.zig").create(a, w.plane));
|
w.ctx = w;
|
||||||
try w.add(try @import("minilog.zig").create(a, w.plane));
|
if (tui.current().config.modestate_show) try w.add(try @import("modestate.zig").create(a, w.plane, event_handler));
|
||||||
if (tui.current().config.selectionstate_show) try w.add(try @import("selectionstate.zig").create(a, w.plane));
|
try w.add(try @import("filestate.zig").create(a, w.plane, event_handler));
|
||||||
try w.add(try @import("diagstate.zig").create(a, w.plane));
|
try w.add(try @import("minilog.zig").create(a, w.plane, event_handler));
|
||||||
try w.add(try @import("linenumstate.zig").create(a, w.plane));
|
if (tui.current().config.selectionstate_show) try w.add(try @import("selectionstate.zig").create(a, w.plane, event_handler));
|
||||||
|
try w.add(try @import("diagstate.zig").create(a, w.plane, event_handler));
|
||||||
|
try w.add(try @import("linenumstate.zig").create(a, w.plane, event_handler));
|
||||||
if (tui.current().config.modstate_show) try w.add(try @import("modstate.zig").create(a, w.plane));
|
if (tui.current().config.modstate_show) try w.add(try @import("modstate.zig").create(a, w.plane));
|
||||||
if (tui.current().config.keystate_show) try w.add(try @import("keystate.zig").create(a, w.plane));
|
if (tui.current().config.keystate_show) try w.add(try @import("keystate.zig").create(a, w.plane));
|
||||||
return w.widget();
|
return w.widget();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn render_grip(ctx: ?*anyopaque, theme: *const Widget.Theme) void {
|
||||||
|
const w: *WidgetList = @ptrCast(@alignCast(ctx.?));
|
||||||
|
if (w.hover()) {
|
||||||
|
w.plane.set_style(theme.statusbar_hover);
|
||||||
|
const width = w.plane.dim_x();
|
||||||
|
const grip_pos = width / 2;
|
||||||
|
w.plane.cursor_move_yx(0, @intCast(grip_pos)) catch {};
|
||||||
|
_ = w.plane.putstr(" ") catch {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -119,7 +119,6 @@ fn init(a: Allocator) !*Self {
|
||||||
self.rdr.dispatch_mouse_drag = dispatch_mouse_drag;
|
self.rdr.dispatch_mouse_drag = dispatch_mouse_drag;
|
||||||
self.rdr.dispatch_event = dispatch_event;
|
self.rdr.dispatch_event = dispatch_event;
|
||||||
try self.rdr.run();
|
try self.rdr.run();
|
||||||
const n = self.rdr.stdplane();
|
|
||||||
|
|
||||||
try frame_clock.start();
|
try frame_clock.start();
|
||||||
try self.commands.init(self);
|
try self.commands.init(self);
|
||||||
|
@ -132,7 +131,7 @@ fn init(a: Allocator) !*Self {
|
||||||
try self.listen_sigwinch();
|
try self.listen_sigwinch();
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
self.mainview = try mainview.create(a, n);
|
self.mainview = try mainview.create(a);
|
||||||
self.resize();
|
self.resize();
|
||||||
self.rdr.set_terminal_style(self.theme.editor);
|
self.rdr.set_terminal_style(self.theme.editor);
|
||||||
try self.rdr.render();
|
try self.rdr.render();
|
||||||
|
@ -701,6 +700,10 @@ pub fn resize(self: *Self) void {
|
||||||
need_render();
|
need_render();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn stdplane(self: *Self) renderer.Plane {
|
||||||
|
return self.rdr.stdplane();
|
||||||
|
}
|
||||||
|
|
||||||
pub fn screen(self: *Self) Widget.Box {
|
pub fn screen(self: *Self) Widget.Box {
|
||||||
return Widget.Box.from(self.rdr.stdplane());
|
return Widget.Box.from(self.rdr.stdplane());
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue