feat: save and restore tabbar state

This commit is contained in:
CJ van den Berg 2025-08-08 20:40:00 +02:00
parent a98d4e02a7
commit 1ebdae310e
Signed by: neurocyte
GPG key ID: 8EB1E1BB660E3FB9
2 changed files with 69 additions and 1 deletions

View file

@ -1303,6 +1303,10 @@ pub fn write_restore_info(self: *Self) void {
const buffer_manager = tui.get_buffer_manager() orelse @panic("tabs no buffer manager");
buffer_manager.write_state(writer) catch return;
if (self.widgets.get("tabs")) |tabs_widget|
if (tabs_widget.dynamic_cast(@import("status/tabs.zig").TabBar)) |tabs|
tabs.write_state(writer) catch return;
const file_name = root.get_restore_file_name() catch return;
var file = std.fs.createFileAbsolute(file_name, .{ .truncate = true }) catch return;
defer file.close();
@ -1324,6 +1328,14 @@ fn read_restore_info(self: *Self) !void {
if (!try cbor.matchValue(&iter, cbor.extract(&editor_file_path))) return error.Stop;
try self.buffer_manager.extract_state(&iter);
if (self.widgets.get("tabs")) |tabs_widget|
if (tabs_widget.dynamic_cast(@import("status/tabs.zig").TabBar)) |tabs|
tabs.extract_state(&iter) catch |e| {
const logger = log.logger("mainview");
defer logger.deinit();
logger.print_err("mainview", "failed to restore tabs: {}", .{e});
};
if (editor_file_path) |file_path| {
try tp.self_pid().send(.{ "cmd", "navigate", .{ .file = file_path } });
} else {

View file

@ -1,4 +1,5 @@
const std = @import("std");
const cbor = @import("cbor");
const tp = @import("thespian");
const root = @import("root");
@ -64,7 +65,7 @@ pub fn create(allocator: std.mem.Allocator, parent: Plane, event_handler: ?Event
return Widget.to(self);
}
const TabBar = struct {
pub const TabBar = struct {
allocator: std.mem.Allocator,
plane: Plane,
widget_list: *WidgetList,
@ -267,6 +268,45 @@ const TabBar = struct {
if (buffer_manager.buffer_from_ref(tab.buffer_ref)) |buffer|
tp.self_pid().send(.{ "cmd", "navigate", .{ .file = buffer.get_file_path() } }) catch {};
}
pub fn write_state(self: *const Self, writer: Buffer.MetaWriter) error{OutOfMemory}!void {
try cbor.writeArrayHeader(writer, self.tabs.len);
for (self.tabs) |tab| try cbor.writeValue(writer, ref_to_name(tab.buffer_ref));
}
fn ref_to_name(buffer_ref: usize) ?[]const u8 {
const buffer_manager = tui.get_buffer_manager() orelse @panic("tabs no buffer manager");
return if (buffer_manager.buffer_from_ref(buffer_ref)) |buffer| buffer.get_file_path() else null;
}
pub fn extract_state(self: *Self, iter: *[]const u8) !void {
var iter2 = iter.*;
self.allocator.free(self.tabs);
self.tabs = &.{};
var result: std.ArrayListUnmanaged(TabBarTab) = .{};
errdefer result.deinit(self.allocator);
var count = cbor.decodeArrayHeader(&iter2) catch return error.MatchTabArrayFailed;
while (count > 0) : (count -= 1) {
var buffer_name: ?[]const u8 = undefined;
if (!(cbor.matchValue(&iter2, cbor.extract(&buffer_name)) catch false)) return error.MatchTabBufferNameFailed;
if (buffer_name) |name| if (name_to_ref(name)) |buffer_ref| {
(try result.addOne(self.allocator)).* = .{
.buffer_ref = buffer_ref,
.widget = try Tab.create(self, buffer_ref, &self.tab_style, self.event_handler),
};
};
}
self.tabs = try result.toOwnedSlice(self.allocator);
iter.* = iter2;
}
fn name_to_ref(buffer_name: []const u8) ?usize {
const buffer_manager = tui.get_buffer_manager() orelse @panic("tabs no buffer manager");
return if (buffer_manager.get_buffer_for_file(buffer_name)) |buffer| buffer_manager.buffer_to_ref(buffer) else null;
}
};
const Tab = struct {
@ -457,6 +497,22 @@ const Tab = struct {
const basename = if (basename_begin) |begin| file_path[begin + 1 ..] else file_path;
return basename;
}
fn write_state(self: *const @This(), writer: Buffer.MetaWriter) error{OutOfMemory}!void {
try cbor.writeArrayHeader(writer, 9);
try cbor.writeValue(writer, self.get_file_path());
try cbor.writeValue(writer, self.file_exists);
try cbor.writeValue(writer, self.file_eol_mode);
try cbor.writeValue(writer, self.hidden);
try cbor.writeValue(writer, self.ephemeral);
try cbor.writeValue(writer, self.meta);
try cbor.writeValue(writer, self.file_type_name);
}
fn extract_state(self: *@This(), iter: *[]const u8) !void {
_ = self;
_ = iter;
}
};
const spacer = struct {