Compare commits
5 commits
0fb10e3aa5
...
c217db02f2
Author | SHA1 | Date | |
---|---|---|---|
c217db02f2 | |||
93ec373ac2 | |||
4ecf46b527 | |||
6ae5dc5c4c | |||
9f117550fa |
4 changed files with 273 additions and 40 deletions
126
src/git.zig
126
src/git.zig
|
@ -47,6 +47,132 @@ pub fn workspace_ignored_files(context: usize) Error!void {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const StatusRecordType = enum {
|
||||||
|
@"#", // header
|
||||||
|
@"1", // ordinary file
|
||||||
|
@"2", // rename or copy
|
||||||
|
u, // unmerged file
|
||||||
|
@"?", // untracked file
|
||||||
|
@"!", // ignored file
|
||||||
|
};
|
||||||
|
|
||||||
|
pub fn status(context_: usize) Error!void {
|
||||||
|
const tag = @src().fn_name;
|
||||||
|
try git_err(context_, .{
|
||||||
|
"--no-optional-locks",
|
||||||
|
"status",
|
||||||
|
"--porcelain=v2",
|
||||||
|
"--branch",
|
||||||
|
"--show-stash",
|
||||||
|
// "--untracked-files=no",
|
||||||
|
"--null",
|
||||||
|
}, struct {
|
||||||
|
fn result(context: usize, parent: tp.pid_ref, output: []const u8) void {
|
||||||
|
var it_ = std.mem.splitScalar(u8, output, 0);
|
||||||
|
while (it_.next()) |line| {
|
||||||
|
var it = std.mem.splitScalar(u8, line, ' ');
|
||||||
|
const rec_type = if (it.next()) |type_tag|
|
||||||
|
std.meta.stringToEnum(StatusRecordType, type_tag) orelse return
|
||||||
|
else
|
||||||
|
return;
|
||||||
|
switch (rec_type) {
|
||||||
|
.@"#" => { // header
|
||||||
|
const name = it.next() orelse return;
|
||||||
|
const value1 = it.next() orelse return;
|
||||||
|
if (it.next()) |value2|
|
||||||
|
parent.send(.{ module_name, context, tag, "#", name, value1, value2 }) catch {}
|
||||||
|
else
|
||||||
|
parent.send(.{ module_name, context, tag, "#", name, value1 }) catch {};
|
||||||
|
},
|
||||||
|
.@"1" => { // ordinary file: <XY> <sub> <mH> <mI> <mW> <hH> <hI> <path>
|
||||||
|
const XY = it.next() orelse return;
|
||||||
|
const sub = it.next() orelse return;
|
||||||
|
const mH = it.next() orelse return;
|
||||||
|
const mI = it.next() orelse return;
|
||||||
|
const mW = it.next() orelse return;
|
||||||
|
const hH = it.next() orelse return;
|
||||||
|
const hI = it.next() orelse return;
|
||||||
|
|
||||||
|
var path: std.ArrayListUnmanaged(u8) = .empty;
|
||||||
|
defer path.deinit(allocator);
|
||||||
|
while (it.next()) |path_part| {
|
||||||
|
if (path.items.len > 0) path.append(allocator, ' ') catch return;
|
||||||
|
path.appendSlice(allocator, path_part) catch return;
|
||||||
|
}
|
||||||
|
|
||||||
|
parent.send(.{ module_name, context, tag, "1", XY, sub, mH, mI, mW, hH, hI, path.items }) catch {};
|
||||||
|
},
|
||||||
|
.@"2" => { // rename or copy: <XY> <sub> <mH> <mI> <mW> <hH> <hI> <X><score> <path><sep><origPath>
|
||||||
|
const XY = it.next() orelse return;
|
||||||
|
const sub = it.next() orelse return;
|
||||||
|
const mH = it.next() orelse return;
|
||||||
|
const mI = it.next() orelse return;
|
||||||
|
const mW = it.next() orelse return;
|
||||||
|
const hH = it.next() orelse return;
|
||||||
|
const hI = it.next() orelse return;
|
||||||
|
const Xscore = it.next() orelse return;
|
||||||
|
|
||||||
|
var path: std.ArrayListUnmanaged(u8) = .empty;
|
||||||
|
defer path.deinit(allocator);
|
||||||
|
while (it.next()) |path_part| {
|
||||||
|
if (path.items.len > 0) path.append(allocator, ' ') catch return;
|
||||||
|
path.appendSlice(allocator, path_part) catch return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const origPath = it_.next() orelse return; // NOTE: this is the next zero terminated part
|
||||||
|
|
||||||
|
parent.send(.{ module_name, context, tag, "2", XY, sub, mH, mI, mW, hH, hI, Xscore, path.items, origPath }) catch {};
|
||||||
|
},
|
||||||
|
.u => { // unmerged file: <XY> <sub> <m1> <m2> <m3> <mW> <h1> <h2> <h3> <path>
|
||||||
|
const XY = it.next() orelse return;
|
||||||
|
const sub = it.next() orelse return;
|
||||||
|
const m1 = it.next() orelse return;
|
||||||
|
const m2 = it.next() orelse return;
|
||||||
|
const m3 = it.next() orelse return;
|
||||||
|
const mW = it.next() orelse return;
|
||||||
|
const h1 = it.next() orelse return;
|
||||||
|
const h2 = it.next() orelse return;
|
||||||
|
const h3 = it.next() orelse return;
|
||||||
|
|
||||||
|
var path: std.ArrayListUnmanaged(u8) = .empty;
|
||||||
|
defer path.deinit(allocator);
|
||||||
|
while (it.next()) |path_part| {
|
||||||
|
if (path.items.len > 0) path.append(allocator, ' ') catch return;
|
||||||
|
path.appendSlice(allocator, path_part) catch return;
|
||||||
|
}
|
||||||
|
|
||||||
|
parent.send(.{ module_name, context, tag, "u", XY, sub, m1, m2, m3, mW, h1, h2, h3, path.items }) catch {};
|
||||||
|
},
|
||||||
|
.@"?" => { // untracked file: <path>
|
||||||
|
var path: std.ArrayListUnmanaged(u8) = .empty;
|
||||||
|
defer path.deinit(allocator);
|
||||||
|
while (it.next()) |path_part| {
|
||||||
|
if (path.items.len > 0) path.append(allocator, ' ') catch return;
|
||||||
|
path.appendSlice(allocator, path_part) catch return;
|
||||||
|
}
|
||||||
|
parent.send(.{ module_name, context, tag, "?", path.items }) catch {};
|
||||||
|
},
|
||||||
|
.@"!" => { // ignored file: <path>
|
||||||
|
var path: std.ArrayListUnmanaged(u8) = .empty;
|
||||||
|
defer path.deinit(allocator);
|
||||||
|
while (it.next()) |path_part| {
|
||||||
|
if (path.items.len > 0) path.append(allocator, ' ') catch return;
|
||||||
|
path.appendSlice(allocator, path_part) catch return;
|
||||||
|
}
|
||||||
|
parent.send(.{ module_name, context, tag, "!", path.items }) catch {};
|
||||||
|
},
|
||||||
|
}
|
||||||
|
// parent.send(.{ module_name, context, tag, value }) catch {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}.result, struct {
|
||||||
|
fn result(_: usize, _: tp.pid_ref, output: []const u8) void {
|
||||||
|
var it = std.mem.splitScalar(u8, output, '\n');
|
||||||
|
while (it.next()) |line| std.log.err("{s}: {s}", .{ module_name, line });
|
||||||
|
}
|
||||||
|
}.result, exit_null(tag));
|
||||||
|
}
|
||||||
|
|
||||||
fn git_line_output(context_: usize, comptime tag: []const u8, cmd: anytype) Error!void {
|
fn git_line_output(context_: usize, comptime tag: []const u8, cmd: anytype) Error!void {
|
||||||
try git_err(context_, cmd, struct {
|
try git_err(context_, cmd, struct {
|
||||||
fn result(context: usize, parent: tp.pid_ref, output: []const u8) void {
|
fn result(context: usize, parent: tp.pid_ref, output: []const u8) void {
|
||||||
|
|
|
@ -57,6 +57,7 @@ pub fn create(ctx_type: type, allocator: std.mem.Allocator, parent: Plane, opts:
|
||||||
.opts = opts,
|
.opts = opts,
|
||||||
};
|
};
|
||||||
self.opts.label = try self.allocator.dupe(u8, opts.label);
|
self.opts.label = try self.allocator.dupe(u8, opts.label);
|
||||||
|
try self.init();
|
||||||
return self;
|
return self;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -75,8 +76,18 @@ pub fn State(ctx_type: type) type {
|
||||||
|
|
||||||
const Self = @This();
|
const Self = @This();
|
||||||
pub const Context = ctx_type;
|
pub const Context = ctx_type;
|
||||||
|
const child: type = switch (@typeInfo(Context)) {
|
||||||
|
.pointer => |p| p.child,
|
||||||
|
.@"struct" => Context,
|
||||||
|
else => struct {},
|
||||||
|
};
|
||||||
|
|
||||||
|
pub fn init(self: *Self) error{OutOfMemory}!void {
|
||||||
|
if (@hasDecl(child, "ctx_init")) return self.opts.ctx.ctx_init();
|
||||||
|
}
|
||||||
|
|
||||||
pub fn deinit(self: *Self, allocator: std.mem.Allocator) void {
|
pub fn deinit(self: *Self, allocator: std.mem.Allocator) void {
|
||||||
|
if (@hasDecl(child, "ctx_deinit")) self.opts.ctx.ctx_deinit();
|
||||||
self.allocator.free(self.opts.label);
|
self.allocator.free(self.opts.label);
|
||||||
self.plane.deinit();
|
self.plane.deinit();
|
||||||
allocator.destroy(self);
|
allocator.destroy(self);
|
||||||
|
|
|
@ -114,7 +114,7 @@ pub const List = struct {
|
||||||
self.list.deinit();
|
self.list.deinit();
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn add(self: *List, h: MessageFilter) !void {
|
pub fn add(self: *List, h: MessageFilter) error{OutOfMemory}!void {
|
||||||
(try self.list.addOne()).* = h;
|
(try self.list.addOne()).* = h;
|
||||||
// @import("log").print("MessageFilter", "add: {d} {s}", .{ self.list.items.len, self.list.items[self.list.items.len - 1].vtable.type_name });
|
// @import("log").print("MessageFilter", "add: {d} {s}", .{ self.list.items.len, self.list.items[self.list.items.len - 1].vtable.type_name });
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,40 +3,66 @@ const tp = @import("thespian");
|
||||||
|
|
||||||
const EventHandler = @import("EventHandler");
|
const EventHandler = @import("EventHandler");
|
||||||
const Plane = @import("renderer").Plane;
|
const Plane = @import("renderer").Plane;
|
||||||
|
const command = @import("command");
|
||||||
const git = @import("git");
|
const git = @import("git");
|
||||||
|
|
||||||
const Widget = @import("../Widget.zig");
|
const Widget = @import("../Widget.zig");
|
||||||
|
const Button = @import("../Button.zig");
|
||||||
const MessageFilter = @import("../MessageFilter.zig");
|
const MessageFilter = @import("../MessageFilter.zig");
|
||||||
const tui = @import("../tui.zig");
|
const tui = @import("../tui.zig");
|
||||||
|
|
||||||
const branch_symbol = "";
|
const branch_symbol = " ";
|
||||||
|
const ahead_symbol = "⇡";
|
||||||
|
const behind_symbol = "⇣";
|
||||||
|
const stash_symbol = "*";
|
||||||
|
const changed_symbol = "+";
|
||||||
|
const untracked_symbol = "?";
|
||||||
|
|
||||||
allocator: std.mem.Allocator,
|
allocator: std.mem.Allocator,
|
||||||
plane: Plane,
|
workspace_path: ?[]const u8 = null,
|
||||||
branch: ?[]const u8 = null,
|
branch: ?[]const u8 = null,
|
||||||
|
ahead: ?[]const u8 = null,
|
||||||
|
behind: ?[]const u8 = null,
|
||||||
|
stash: ?[]const u8 = null,
|
||||||
|
changed: usize = 0,
|
||||||
|
untracked: usize = 0,
|
||||||
|
done: bool = true,
|
||||||
|
|
||||||
const Self = @This();
|
const Self = @This();
|
||||||
|
|
||||||
pub fn create(
|
pub fn create(
|
||||||
allocator: std.mem.Allocator,
|
allocator: std.mem.Allocator,
|
||||||
parent: Plane,
|
parent: Plane,
|
||||||
_: ?EventHandler,
|
event_handler: ?EventHandler,
|
||||||
_: ?[]const u8,
|
_: ?[]const u8,
|
||||||
) @import("widget.zig").CreateError!Widget {
|
) @import("widget.zig").CreateError!Widget {
|
||||||
const self: *Self = try allocator.create(Self);
|
return Button.create_widget(Self, allocator, parent, .{
|
||||||
self.* = .{
|
.ctx = .{
|
||||||
.allocator = allocator,
|
.allocator = allocator,
|
||||||
.plane = try Plane.init(&(Widget.Box{}).opts(@typeName(Self)), parent),
|
},
|
||||||
};
|
.label = "",
|
||||||
try tui.message_filters().add(MessageFilter.bind(self, receive_git));
|
.on_click = on_click,
|
||||||
git.workspace_path(0) catch {};
|
.on_layout = layout,
|
||||||
return Widget.to(self);
|
.on_render = render,
|
||||||
|
.on_event = event_handler,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn deinit(self: *Self, allocator: std.mem.Allocator) void {
|
pub fn ctx_init(self: *Self) error{OutOfMemory}!void {
|
||||||
|
try tui.message_filters().add(MessageFilter.bind(self, receive_git));
|
||||||
|
git.workspace_path(0) catch {};
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn ctx_deinit(self: *Self) void {
|
||||||
|
tui.message_filters().remove_ptr(self);
|
||||||
if (self.branch) |p| self.allocator.free(p);
|
if (self.branch) |p| self.allocator.free(p);
|
||||||
self.plane.deinit();
|
if (self.ahead) |p| self.allocator.free(p);
|
||||||
allocator.destroy(self);
|
if (self.behind) |p| self.allocator.free(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_click(_: *Self, _: *Button.State(Self)) void {
|
||||||
|
git.status(0) catch {};
|
||||||
|
command.executeName("show_git_status", .{}) catch {};
|
||||||
}
|
}
|
||||||
|
|
||||||
fn receive_git(self: *Self, _: tp.pid_ref, m: tp.message) MessageFilter.Error!bool {
|
fn receive_git(self: *Self, _: tp.pid_ref, m: tp.message) MessageFilter.Error!bool {
|
||||||
|
@ -46,45 +72,115 @@ fn receive_git(self: *Self, _: tp.pid_ref, m: tp.message) MessageFilter.Error!bo
|
||||||
false;
|
false;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn process_git(
|
fn process_git(self: *Self, m: tp.message) MessageFilter.Error!bool {
|
||||||
self: *Self,
|
var value: []const u8 = undefined;
|
||||||
m: tp.message,
|
|
||||||
) MessageFilter.Error!bool {
|
|
||||||
var branch: []const u8 = undefined;
|
|
||||||
if (try match(m.buf, .{ any, any, "workspace_path", null_ })) {
|
if (try match(m.buf, .{ any, any, "workspace_path", null_ })) {
|
||||||
// do nothing, we do not have a git workspace
|
// do nothing, we do not have a git workspace
|
||||||
} else if (try match(m.buf, .{ any, any, "workspace_path", string })) {
|
} else if (try match(m.buf, .{ any, any, "workspace_path", extract(&value) })) {
|
||||||
git.current_branch(0) catch {};
|
if (self.workspace_path) |p| self.allocator.free(p);
|
||||||
} else if (try match(m.buf, .{ any, any, "current_branch", extract(&branch) })) {
|
self.workspace_path = try self.allocator.dupe(u8, value);
|
||||||
|
// git.current_branch(0) catch {};
|
||||||
|
git.status(0) catch {};
|
||||||
|
} else if (try match(m.buf, .{ any, any, "current_branch", extract(&value) })) {
|
||||||
if (self.branch) |p| self.allocator.free(p);
|
if (self.branch) |p| self.allocator.free(p);
|
||||||
self.branch = try self.allocator.dupe(u8, branch);
|
self.branch = try self.allocator.dupe(u8, value);
|
||||||
|
} else if (try match(m.buf, .{ any, any, "status", tp.more })) {
|
||||||
|
return self.process_status(m);
|
||||||
} else {
|
} else {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
const format = " {s} {s} ";
|
fn process_status(self: *Self, m: tp.message) MessageFilter.Error!bool {
|
||||||
|
var value: []const u8 = undefined;
|
||||||
|
var ahead: []const u8 = undefined;
|
||||||
|
var behind: []const u8 = undefined;
|
||||||
|
if (self.done) {
|
||||||
|
self.done = false;
|
||||||
|
self.changed = 0;
|
||||||
|
self.untracked = 0;
|
||||||
|
if (self.ahead) |p| self.allocator.free(p);
|
||||||
|
self.ahead = null;
|
||||||
|
if (self.behind) |p| self.allocator.free(p);
|
||||||
|
self.behind = null;
|
||||||
|
if (self.stash) |p| self.allocator.free(p);
|
||||||
|
self.stash = null;
|
||||||
|
}
|
||||||
|
|
||||||
pub fn layout(self: *Self) Widget.Layout {
|
if (try match(m.buf, .{ any, any, "status", "#", "branch.oid", extract(&value) })) {
|
||||||
const branch = self.branch orelse return .{ .static = 0 };
|
// commit | (initial)
|
||||||
var buf: [256]u8 = undefined;
|
} else if (try match(m.buf, .{ any, any, "status", "#", "branch.head", extract(&value) })) {
|
||||||
var fbs = std.io.fixedBufferStream(&buf);
|
if (self.branch) |p| self.allocator.free(p);
|
||||||
|
self.branch = try self.allocator.dupe(u8, value);
|
||||||
|
} else if (try match(m.buf, .{ any, any, "status", "#", "branch.upstream", extract(&value) })) {
|
||||||
|
// upstream-branch
|
||||||
|
} else if (try match(m.buf, .{ any, any, "status", "#", "branch.ab", extract(&ahead), extract(&behind) })) {
|
||||||
|
if (self.ahead) |p| self.allocator.free(p);
|
||||||
|
self.ahead = try self.allocator.dupe(u8, ahead);
|
||||||
|
if (self.behind) |p| self.allocator.free(p);
|
||||||
|
self.behind = try self.allocator.dupe(u8, behind);
|
||||||
|
} else if (try match(m.buf, .{ any, any, "status", "#", "stash", extract(&value) })) {
|
||||||
|
if (self.stash) |p| self.allocator.free(p);
|
||||||
|
self.stash = try self.allocator.dupe(u8, value);
|
||||||
|
} else if (try match(m.buf, .{ any, any, "status", "1", tp.more })) {
|
||||||
|
// ordinary file: <XY> <sub> <mH> <mI> <mW> <hH> <hI> <path>
|
||||||
|
self.changed += 1;
|
||||||
|
} else if (try match(m.buf, .{ any, any, "status", "2", tp.more })) {
|
||||||
|
// rename or copy: <XY> <sub> <mH> <mI> <mW> <hH> <hI> <X><score> <path><sep><origPath>
|
||||||
|
self.changed += 1;
|
||||||
|
} else if (try match(m.buf, .{ any, any, "status", "u", tp.more })) {
|
||||||
|
// unmerged file: <XY> <sub> <m1> <m2> <m3> <mW> <h1> <h2> <h3> <path>
|
||||||
|
self.changed += 1;
|
||||||
|
} else if (try match(m.buf, .{ any, any, "status", "?", tp.more })) {
|
||||||
|
// untracked file: <path>
|
||||||
|
self.untracked += 1;
|
||||||
|
} else if (try match(m.buf, .{ any, any, "status", "!", tp.more })) {
|
||||||
|
// ignored file: <path>
|
||||||
|
} else if (try match(m.buf, .{ any, any, "status", null_ })) {
|
||||||
|
self.done = true;
|
||||||
|
} else return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn format(self: *Self, buf: []u8) []const u8 {
|
||||||
|
const branch = self.branch orelse return "";
|
||||||
|
var fbs = std.io.fixedBufferStream(buf);
|
||||||
const writer = fbs.writer();
|
const writer = fbs.writer();
|
||||||
writer.print(format, .{ branch_symbol, branch }) catch {};
|
writer.print(" {s}{s}", .{ branch_symbol, branch }) catch {};
|
||||||
const len = self.plane.egc_chunk_width(fbs.getWritten(), 0, 1);
|
if (self.ahead) |ahead| if (ahead.len > 1 and ahead[1] != '0')
|
||||||
|
writer.print(" {s}{s}", .{ ahead_symbol, ahead[1..] }) catch {};
|
||||||
|
if (self.behind) |behind| if (behind.len > 1 and behind[1] != '0')
|
||||||
|
writer.print(" {s}{s}", .{ behind_symbol, behind[1..] }) catch {};
|
||||||
|
if (self.stash) |stash| if (stash.len > 0 and stash[0] != '0')
|
||||||
|
writer.print(" {s}{s}", .{ stash_symbol, stash }) catch {};
|
||||||
|
if (self.changed > 0)
|
||||||
|
writer.print(" {s}{d}", .{ changed_symbol, self.changed }) catch {};
|
||||||
|
if (self.untracked > 0)
|
||||||
|
writer.print(" {s}{d}", .{ untracked_symbol, self.untracked }) catch {};
|
||||||
|
writer.print(" ", .{}) catch {};
|
||||||
|
return fbs.getWritten();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn layout(self: *Self, btn: *Button.State(Self)) Widget.Layout {
|
||||||
|
var buf: [256]u8 = undefined;
|
||||||
|
const text = self.format(&buf);
|
||||||
|
const len = btn.plane.egc_chunk_width(text, 0, 1);
|
||||||
return .{ .static = len };
|
return .{ .static = len };
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn render(self: *Self, theme: *const Widget.Theme) bool {
|
pub fn render(self: *Self, btn: *Button.State(Self), theme: *const Widget.Theme) bool {
|
||||||
const branch = self.branch orelse return false;
|
var buf: [256]u8 = undefined;
|
||||||
self.plane.set_base_style(theme.editor);
|
const text = self.format(&buf);
|
||||||
self.plane.erase();
|
if (text.len == 0) return false;
|
||||||
self.plane.home();
|
const bg_style = if (btn.active) theme.editor_cursor else if (btn.hover) theme.statusbar_hover else theme.statusbar;
|
||||||
self.plane.set_style(theme.statusbar);
|
btn.plane.set_base_style(theme.editor);
|
||||||
self.plane.fill(" ");
|
btn.plane.erase();
|
||||||
self.plane.home();
|
btn.plane.home();
|
||||||
_ = self.plane.print(format, .{ branch_symbol, branch }) catch {};
|
btn.plane.set_style(bg_style);
|
||||||
|
btn.plane.fill(" ");
|
||||||
|
btn.plane.home();
|
||||||
|
_ = btn.plane.putstr(text) catch {};
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue