nix update flake and finish keyboard handling
This commit is contained in:
parent
4739675642
commit
23dd09bbf9
6 changed files with 397 additions and 116 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -9,6 +9,7 @@
|
||||||
# Cheers!
|
# Cheers!
|
||||||
# -andrewrk
|
# -andrewrk
|
||||||
|
|
||||||
|
.zig-cache/
|
||||||
zig-cache/
|
zig-cache/
|
||||||
zig-out/
|
zig-out/
|
||||||
/release/
|
/release/
|
||||||
|
|
81
build.zig
81
build.zig
|
@ -1,97 +1,68 @@
|
||||||
const std = @import("std");
|
const std = @import("std");
|
||||||
|
|
||||||
// Although this function looks imperative, note that its job is to
|
|
||||||
// declaratively construct a build graph that will be executed by an external
|
|
||||||
// runner.
|
|
||||||
pub fn build(b: *std.Build) void {
|
pub fn build(b: *std.Build) void {
|
||||||
// Standard target options allows the person running `zig build` to choose
|
|
||||||
// what target to build for. Here we do not override the defaults, which
|
|
||||||
// means any target is allowed, and the default is native. Other options
|
|
||||||
// for restricting supported target set are available.
|
|
||||||
const target = b.standardTargetOptions(.{});
|
const target = b.standardTargetOptions(.{});
|
||||||
|
|
||||||
// Standard optimization options allow the person running `zig build` to select
|
|
||||||
// between Debug, ReleaseSafe, ReleaseFast, and ReleaseSmall. Here we do not
|
|
||||||
// set a preferred release mode, allowing the user to decide how to optimize.
|
|
||||||
const optimize = b.standardOptimizeOption(.{});
|
const optimize = b.standardOptimizeOption(.{});
|
||||||
|
|
||||||
_ = b.addModule("silkdot", .{
|
_ = b.addModule("silkdot", .{
|
||||||
.root_source_file = .{ .path = "src/main.zig" },
|
.root_source_file = b.path("src/main.zig"),
|
||||||
.target = target,
|
.target = target,
|
||||||
.optimize = optimize,
|
.optimize = optimize,
|
||||||
});
|
});
|
||||||
|
|
||||||
const lib = b.addStaticLibrary(.{
|
// const lib = b.addStaticLibrary(.{
|
||||||
.name = "master",
|
// .name = "master",
|
||||||
// In this case the main source file is merely a path, however, in more
|
// .root_source_file = b.path("src/root.zig"),
|
||||||
// complicated build scripts, this could be a generated file.
|
// .target = target,
|
||||||
.root_source_file = .{ .path = "src/root.zig" },
|
// .optimize = optimize,
|
||||||
.target = target,
|
// });
|
||||||
.optimize = optimize,
|
|
||||||
});
|
|
||||||
|
|
||||||
// This declares intent for the library to be installed into the standard
|
// b.installArtifact(lib);
|
||||||
// location when the user invokes the "install" step (the default step when
|
|
||||||
// running `zig build`).
|
|
||||||
b.installArtifact(lib);
|
|
||||||
|
|
||||||
const exe = b.addExecutable(.{
|
const exe = b.addExecutable(.{
|
||||||
.name = "master",
|
.name = "master",
|
||||||
.root_source_file = .{ .path = "src/main.zig" },
|
.root_source_file = b.path("src/main.zig"),
|
||||||
.target = target,
|
.target = target,
|
||||||
.optimize = optimize,
|
.optimize = optimize,
|
||||||
});
|
});
|
||||||
|
|
||||||
// This declares intent for the executable to be installed into the
|
|
||||||
// standard location when the user invokes the "install" step (the default
|
|
||||||
// step when running `zig build`).
|
|
||||||
b.installArtifact(exe);
|
b.installArtifact(exe);
|
||||||
|
|
||||||
// This *creates* a Run step in the build graph, to be executed when another
|
|
||||||
// step is evaluated that depends on it. The next line below will establish
|
|
||||||
// such a dependency.
|
|
||||||
const run_cmd = b.addRunArtifact(exe);
|
const run_cmd = b.addRunArtifact(exe);
|
||||||
|
|
||||||
// By making the run step depend on the install step, it will be run from the
|
|
||||||
// installation directory rather than directly from within the cache directory.
|
|
||||||
// This is not necessary, however, if the application depends on other installed
|
|
||||||
// files, this ensures they will be present and in the expected location.
|
|
||||||
run_cmd.step.dependOn(b.getInstallStep());
|
run_cmd.step.dependOn(b.getInstallStep());
|
||||||
|
|
||||||
// This allows the user to pass arguments to the application in the build
|
|
||||||
// command itself, like this: `zig build run -- arg1 arg2 etc`
|
|
||||||
if (b.args) |args| {
|
if (b.args) |args| {
|
||||||
run_cmd.addArgs(args);
|
run_cmd.addArgs(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
// This creates a build step. It will be visible in the `zig build --help` menu,
|
|
||||||
// and can be selected like this: `zig build run`
|
|
||||||
// This will evaluate the `run` step rather than the default, which is "install".
|
|
||||||
const run_step = b.step("run", "Run the app");
|
const run_step = b.step("run", "Run the app");
|
||||||
run_step.dependOn(&run_cmd.step);
|
run_step.dependOn(&run_cmd.step);
|
||||||
|
|
||||||
// Creates a step for unit testing. This only builds the test executable
|
// const lib_unit_tests = b.addTest(.{
|
||||||
// but does not run it.
|
// .root_source_file = b.path("src/root.zig"),
|
||||||
const lib_unit_tests = b.addTest(.{
|
// .target = target,
|
||||||
.root_source_file = .{ .path = "src/root.zig" },
|
// .optimize = optimize,
|
||||||
.target = target,
|
// });
|
||||||
.optimize = optimize,
|
|
||||||
});
|
|
||||||
|
|
||||||
const run_lib_unit_tests = b.addRunArtifact(lib_unit_tests);
|
// const run_lib_unit_tests = b.addRunArtifact(lib_unit_tests);
|
||||||
|
|
||||||
const exe_unit_tests = b.addTest(.{
|
const exe_unit_tests = b.addTest(.{
|
||||||
.root_source_file = .{ .path = "src/main.zig" },
|
.root_source_file = b.path("src/main.zig"),
|
||||||
.target = target,
|
.target = target,
|
||||||
.optimize = optimize,
|
.optimize = optimize,
|
||||||
});
|
});
|
||||||
|
|
||||||
const run_exe_unit_tests = b.addRunArtifact(exe_unit_tests);
|
const run_exe_unit_tests = b.addRunArtifact(exe_unit_tests);
|
||||||
|
|
||||||
// Similar to creating the run step earlier, this exposes a `test` step to
|
|
||||||
// the `zig build --help` menu, providing a way for the user to request
|
|
||||||
// running the unit tests.
|
|
||||||
const test_step = b.step("test", "Run unit tests");
|
const test_step = b.step("test", "Run unit tests");
|
||||||
test_step.dependOn(&run_lib_unit_tests.step);
|
// test_step.dependOn(&run_lib_unit_tests.step);
|
||||||
test_step.dependOn(&run_exe_unit_tests.step);
|
test_step.dependOn(&run_exe_unit_tests.step);
|
||||||
|
|
||||||
|
const install_docs = b.addInstallDirectory(.{
|
||||||
|
.source_dir = exe.getEmittedDocs(),
|
||||||
|
.install_dir = .prefix,
|
||||||
|
.install_subdir = "docs",
|
||||||
|
});
|
||||||
|
|
||||||
|
const docs_step = b.step("docs", "Copy documentation artifacts to prefix path");
|
||||||
|
docs_step.dependOn(&install_docs.step);
|
||||||
}
|
}
|
||||||
|
|
18
flake.lock
generated
18
flake.lock
generated
|
@ -5,11 +5,11 @@
|
||||||
"systems": "systems"
|
"systems": "systems"
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1705309234,
|
"lastModified": 1710146030,
|
||||||
"narHash": "sha256-uNRRNRKmJyCRC/8y1RqBkqWBLM034y4qN7EprSdmgyA=",
|
"narHash": "sha256-SZ5L6eA7HJ/nmkzGG7/ISclqe6oZdOZTNoesiInkXPQ=",
|
||||||
"owner": "numtide",
|
"owner": "numtide",
|
||||||
"repo": "flake-utils",
|
"repo": "flake-utils",
|
||||||
"rev": "1ef2e671c3b0c19053962c07dbda38332dcebf26",
|
"rev": "b1d9ab70662946ef0850d488da1c9019f3a9752a",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
@ -20,11 +20,11 @@
|
||||||
},
|
},
|
||||||
"nixpkgs": {
|
"nixpkgs": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1706063522,
|
"lastModified": 1718622336,
|
||||||
"narHash": "sha256-o1m9en7ovSjyktXgX3n/6GJEwG06WYa/9Mfx5hTTf5g=",
|
"narHash": "sha256-lywfxWRBn+lwdKaBy5x5uTkbCcEPUonCn6bK8OQPsw4=",
|
||||||
"owner": "nixos",
|
"owner": "nixos",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "95c1439b205d507f3cb88aae76e02cd6a01ac504",
|
"rev": "d0fc4188d246ab953653f00e9ce0cf51d10d5eda",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
@ -59,11 +59,11 @@
|
||||||
"nixpkgs": "nixpkgs"
|
"nixpkgs": "nixpkgs"
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1712106553,
|
"lastModified": 1721870682,
|
||||||
"narHash": "sha256-p7xlViVQh/916u28pTtPw3L+s2PkkgA5MAMk/Vfibac=",
|
"narHash": "sha256-NAIeaZpJR4ynuw1pUhhukPqbKGfOoXPcATAWVtmkiiU=",
|
||||||
"owner": "Cloudef",
|
"owner": "Cloudef",
|
||||||
"repo": "zig2nix",
|
"repo": "zig2nix",
|
||||||
"rev": "32bae779ad1712a83919e8a61da1ea8e60e328e1",
|
"rev": "4be136cc3615ba0047ded273d693116c8b05d0d4",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
|
393
src/main.zig
393
src/main.zig
|
@ -93,7 +93,7 @@ pub const Terminal = struct {
|
||||||
try term.uncook();
|
try term.uncook();
|
||||||
try attachSignalHandlers();
|
try attachSignalHandlers();
|
||||||
try term.updateWinSize();
|
try term.updateWinSize();
|
||||||
try term.print("\x1b[>1u", .{});
|
try term.print("\x1b[>{d}u", .{KittyFlags.asInt(.{})});
|
||||||
return term;
|
return term;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -142,10 +142,10 @@ pub const Terminal = struct {
|
||||||
.mask = posix.empty_sigset,
|
.mask = posix.empty_sigset,
|
||||||
.flags = 0,
|
.flags = 0,
|
||||||
};
|
};
|
||||||
// try posix.sigaction(posix.SIG.INT, &act, null);
|
// posix.sigaction(posix.SIG.INT, &act, null);
|
||||||
try posix.sigaction(posix.SIG.USR1, &act, null);
|
posix.sigaction(posix.SIG.USR1, &act, null);
|
||||||
try posix.sigaction(posix.SIG.USR2, &act, null);
|
posix.sigaction(posix.SIG.USR2, &act, null);
|
||||||
try posix.sigaction(posix.SIG.WINCH, &act, null);
|
posix.sigaction(posix.SIG.WINCH, &act, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn getEvents(self: *Terminal) ![]Event {
|
pub fn getEvents(self: *Terminal) ![]Event {
|
||||||
|
@ -174,21 +174,56 @@ pub const Terminal = struct {
|
||||||
if (key[0] == '[') {
|
if (key[0] == '[') {
|
||||||
var code_list = std.ArrayList(u8).init(self.allocator);
|
var code_list = std.ArrayList(u8).init(self.allocator);
|
||||||
defer code_list.deinit();
|
defer code_list.deinit();
|
||||||
try self.tty.reader().streamUntilDelimiter(code_list.writer(), 'u', 1024);
|
var delim: u8 = 0;
|
||||||
|
while (code_list.items.len < 1024) {
|
||||||
|
const c = self.tty.reader().readByte() catch |e| {
|
||||||
|
if (e == error.EndOfStream) {
|
||||||
|
std.debug.print("\n\r\n{s}\r\n", .{fmt.fmtSliceHexLower(code_list.items)});
|
||||||
|
return error.UnknownControlCode;
|
||||||
|
} else return e;
|
||||||
|
};
|
||||||
|
switch (c) {
|
||||||
|
'u', '~', 'A', 'B', 'C', 'D', 'E', 'F', 'H', 'P', 'Q', 'S' => {
|
||||||
|
delim = c;
|
||||||
|
break;
|
||||||
|
},
|
||||||
|
else => try code_list.append(c),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// std.debug.print("{s}{s}\r\n", .{ code_list.items, &[1]u8{delim} });
|
||||||
|
|
||||||
const idx1 = mem.indexOfScalar(u8, code_list.items, ';');
|
var keycode: u21 = 1;
|
||||||
// TODO: alternate key code
|
var alt_keycode: ?u21 = null;
|
||||||
if (idx1) |idx| {
|
var modifiers: u9 = 1;
|
||||||
const idx2 = mem.indexOfScalarPos(u8, code_list.items, idx, ';');
|
var event_type: EventType = .press;
|
||||||
// if (idx2) |ix| {
|
var text: ?u21 = null;
|
||||||
const keycode = try fmt.parseInt(u21, code_list.items[0..idx], 10);
|
|
||||||
const modifiers = try fmt.parseInt(u9, if (idx2) |i| code_list.items[idx..i] else code_list.items[idx..], 10);
|
var section: u8 = 0;
|
||||||
|
var semicolon_it = mem.splitScalar(u8, code_list.items, ';');
|
||||||
|
while (semicolon_it.next()) |semi| : (section += 1) try switch (section) {
|
||||||
|
0 => if (semi.len > 0) {
|
||||||
|
const colon = mem.indexOfScalar(u8, semi, ':');
|
||||||
|
keycode = try fmt.parseInt(u21, if (colon) |c| semi[0..c] else semi, 10);
|
||||||
|
if (colon) |c| alt_keycode = try fmt.parseInt(u21, semi[c + 1 ..], 10);
|
||||||
|
},
|
||||||
|
1 => if (semi.len > 0) {
|
||||||
|
const colon = mem.indexOfScalar(u8, semi, ':');
|
||||||
|
modifiers = try fmt.parseInt(u9, if (colon) |c| semi[0..c] else semi, 10);
|
||||||
|
if (colon) |c| event_type = EventType.init(try fmt.parseInt(u2, semi[c + 1 ..], 10));
|
||||||
|
},
|
||||||
|
2 => text = try fmt.parseInt(u21, semi, 10),
|
||||||
|
else => error.InvalidKittyEscape,
|
||||||
|
};
|
||||||
try self.events.append(Event{ .keyboard = .{
|
try self.events.append(Event{ .keyboard = .{
|
||||||
.code = keycode,
|
.code = keycode,
|
||||||
|
.altcode = alt_keycode,
|
||||||
.mods = Modifiers.init(modifiers),
|
.mods = Modifiers.init(modifiers),
|
||||||
|
.evtype = event_type,
|
||||||
|
.text = text,
|
||||||
|
.fnkey = FunctionalKey.init(keycode, delim),
|
||||||
} });
|
} });
|
||||||
// } else return error.InvalidKittyEscape;
|
// std.debug.print("{any}\r\n", .{self.events.items[self.events.items.len - 1].keyboard});
|
||||||
} else return error.InvalidKittyEscape; // TODO better error name!!!
|
// std.debug.print("keycode: {}\r\nalt_keycode: {?}\r\nmodifiers: {}\r\n{}\r\ntext: {?}\r\n\nspecial: {?}\r\n\n", .{ keycode, alt_keycode, modifiers, event_type, text, FunctionalKey.init(keycode, delim) });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -218,7 +253,11 @@ pub const Terminal = struct {
|
||||||
},
|
},
|
||||||
keyboard: struct {
|
keyboard: struct {
|
||||||
code: u21,
|
code: u21,
|
||||||
|
altcode: ?u21 = null,
|
||||||
mods: Modifiers,
|
mods: Modifiers,
|
||||||
|
evtype: EventType = .press,
|
||||||
|
text: ?u21 = null,
|
||||||
|
fnkey: ?FunctionalKey = null,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -245,28 +284,271 @@ pub const Terminal = struct {
|
||||||
.num_lock = (b & 0b10000000) > 0,
|
.num_lock = (b & 0b10000000) > 0,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
// pub const Bits = enum(u8) {
|
};
|
||||||
// shift = 0b1,
|
|
||||||
// alt = 0b10,
|
pub const EventType = enum(u2) {
|
||||||
// ctrl = 0b100,
|
press = 1,
|
||||||
// super = 0b1000,
|
repeat = 2,
|
||||||
// hyper = 0b10000,
|
release = 3,
|
||||||
// meta = 0b100000,
|
|
||||||
// caps_lock = 0b1000000,
|
pub fn init(int: u2) EventType {
|
||||||
// num_lock = 0b10000000,
|
return if (int == 0) .press else @enumFromInt(int);
|
||||||
// };
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
pub const FunctionalKey = enum {
|
||||||
|
escape,
|
||||||
|
enter,
|
||||||
|
tab,
|
||||||
|
backspace,
|
||||||
|
insert,
|
||||||
|
delete,
|
||||||
|
left,
|
||||||
|
right,
|
||||||
|
up,
|
||||||
|
down,
|
||||||
|
page_up,
|
||||||
|
page_down,
|
||||||
|
home,
|
||||||
|
end,
|
||||||
|
caps_lock,
|
||||||
|
scroll_lock,
|
||||||
|
num_lock,
|
||||||
|
print_screen,
|
||||||
|
pause,
|
||||||
|
menu,
|
||||||
|
f1,
|
||||||
|
f2,
|
||||||
|
f3,
|
||||||
|
f4,
|
||||||
|
f5,
|
||||||
|
f6,
|
||||||
|
f7,
|
||||||
|
f8,
|
||||||
|
f9,
|
||||||
|
f10,
|
||||||
|
f11,
|
||||||
|
f12,
|
||||||
|
f13,
|
||||||
|
f14,
|
||||||
|
f15,
|
||||||
|
f16,
|
||||||
|
f17,
|
||||||
|
f18,
|
||||||
|
f19,
|
||||||
|
f20,
|
||||||
|
f21,
|
||||||
|
f22,
|
||||||
|
f23,
|
||||||
|
f24,
|
||||||
|
f25,
|
||||||
|
f26,
|
||||||
|
f27,
|
||||||
|
f28,
|
||||||
|
f29,
|
||||||
|
f30,
|
||||||
|
f31,
|
||||||
|
f32,
|
||||||
|
f33,
|
||||||
|
f34,
|
||||||
|
f35,
|
||||||
|
kp_0,
|
||||||
|
kp_1,
|
||||||
|
kp_2,
|
||||||
|
kp_3,
|
||||||
|
kp_4,
|
||||||
|
kp_5,
|
||||||
|
kp_6,
|
||||||
|
kp_7,
|
||||||
|
kp_8,
|
||||||
|
kp_9,
|
||||||
|
kp_decimal,
|
||||||
|
kp_divide,
|
||||||
|
kp_multiply,
|
||||||
|
kp_subtract,
|
||||||
|
kp_add,
|
||||||
|
kp_enter,
|
||||||
|
kp_equal,
|
||||||
|
kp_separator,
|
||||||
|
kp_left,
|
||||||
|
kp_right,
|
||||||
|
kp_up,
|
||||||
|
kp_down,
|
||||||
|
kp_page_up,
|
||||||
|
kp_page_down,
|
||||||
|
kp_home,
|
||||||
|
kp_end,
|
||||||
|
kp_insert,
|
||||||
|
kp_delete,
|
||||||
|
kp_begin,
|
||||||
|
media_play,
|
||||||
|
media_pause,
|
||||||
|
media_play_pause,
|
||||||
|
media_reverse,
|
||||||
|
media_stop,
|
||||||
|
media_fast_forward,
|
||||||
|
media_rewind,
|
||||||
|
media_track_next,
|
||||||
|
media_track_previous,
|
||||||
|
media_record,
|
||||||
|
lower_volume,
|
||||||
|
raise_volume,
|
||||||
|
mute_volume,
|
||||||
|
left_shift,
|
||||||
|
left_control,
|
||||||
|
left_alt,
|
||||||
|
left_super,
|
||||||
|
left_hyper,
|
||||||
|
left_meta,
|
||||||
|
right_shift,
|
||||||
|
right_control,
|
||||||
|
right_alt,
|
||||||
|
right_super,
|
||||||
|
right_hyper,
|
||||||
|
right_meta,
|
||||||
|
iso_level3_shift,
|
||||||
|
iso_level5_shift,
|
||||||
|
|
||||||
|
fn init(keycode: u21, delim: u8) ?FunctionalKey {
|
||||||
|
return switch (delim) {
|
||||||
|
'~' => switch (keycode) {
|
||||||
|
2 => .insert,
|
||||||
|
3 => .delete,
|
||||||
|
5 => .page_up,
|
||||||
|
6 => .page_down,
|
||||||
|
7 => .home,
|
||||||
|
8 => .end,
|
||||||
|
11 => .f1,
|
||||||
|
12 => .f2,
|
||||||
|
13 => .f3,
|
||||||
|
14 => .f4,
|
||||||
|
15 => .f5,
|
||||||
|
17 => .f6,
|
||||||
|
18 => .f7,
|
||||||
|
19 => .f8,
|
||||||
|
20 => .f9,
|
||||||
|
21 => .f10,
|
||||||
|
23 => .f11,
|
||||||
|
24 => .f12,
|
||||||
|
57427 => .kp_begin,
|
||||||
|
else => null,
|
||||||
|
},
|
||||||
|
'A' => if (keycode == 1) .up else null,
|
||||||
|
'B' => if (keycode == 1) .down else null,
|
||||||
|
'C' => if (keycode == 1) .right else null,
|
||||||
|
'D' => if (keycode == 1) .left else null,
|
||||||
|
'E' => if (keycode == 1) .kp_begin else null,
|
||||||
|
'F' => if (keycode == 1) .end else null,
|
||||||
|
'H' => if (keycode == 1) .home else null,
|
||||||
|
'P' => if (keycode == 1) .f1 else null,
|
||||||
|
'Q' => if (keycode == 1) .f2 else null,
|
||||||
|
'S' => if (keycode == 1) .f4 else null,
|
||||||
|
'u' => switch (keycode) {
|
||||||
|
27 => .escape,
|
||||||
|
13 => .enter,
|
||||||
|
9 => .tab,
|
||||||
|
127 => .backspace,
|
||||||
|
57358 => .caps_lock,
|
||||||
|
57359 => .scroll_lock,
|
||||||
|
57360 => .num_lock,
|
||||||
|
57361 => .print_screen,
|
||||||
|
57362 => .pause,
|
||||||
|
57363 => .menu,
|
||||||
|
57376 => .f13,
|
||||||
|
57377 => .f14,
|
||||||
|
57378 => .f15,
|
||||||
|
57379 => .f16,
|
||||||
|
57380 => .f17,
|
||||||
|
57381 => .f18,
|
||||||
|
57382 => .f19,
|
||||||
|
57383 => .f20,
|
||||||
|
57384 => .f21,
|
||||||
|
57385 => .f22,
|
||||||
|
57386 => .f23,
|
||||||
|
57387 => .f24,
|
||||||
|
57388 => .f25,
|
||||||
|
57389 => .f26,
|
||||||
|
57390 => .f27,
|
||||||
|
57391 => .f28,
|
||||||
|
57392 => .f29,
|
||||||
|
57393 => .f30,
|
||||||
|
57394 => .f31,
|
||||||
|
57395 => .f32,
|
||||||
|
57396 => .f33,
|
||||||
|
57397 => .f34,
|
||||||
|
57398 => .f35,
|
||||||
|
57399 => .kp_0,
|
||||||
|
57400 => .kp_1,
|
||||||
|
57401 => .kp_2,
|
||||||
|
57402 => .kp_3,
|
||||||
|
57403 => .kp_4,
|
||||||
|
57404 => .kp_5,
|
||||||
|
57405 => .kp_6,
|
||||||
|
57406 => .kp_7,
|
||||||
|
57407 => .kp_8,
|
||||||
|
57408 => .kp_9,
|
||||||
|
57409 => .kp_decimal,
|
||||||
|
57410 => .kp_divide,
|
||||||
|
57411 => .kp_multiply,
|
||||||
|
57412 => .kp_subtract,
|
||||||
|
57413 => .kp_add,
|
||||||
|
57414 => .kp_enter,
|
||||||
|
57415 => .kp_equal,
|
||||||
|
57416 => .kp_separator,
|
||||||
|
57417 => .kp_left,
|
||||||
|
57418 => .kp_right,
|
||||||
|
57419 => .kp_up,
|
||||||
|
57420 => .kp_down,
|
||||||
|
57421 => .kp_page_up,
|
||||||
|
57422 => .kp_page_down,
|
||||||
|
57423 => .kp_home,
|
||||||
|
57424 => .kp_end,
|
||||||
|
57425 => .kp_insert,
|
||||||
|
57426 => .kp_delete,
|
||||||
|
57428 => .media_play,
|
||||||
|
57429 => .media_pause,
|
||||||
|
57430 => .media_play_pause,
|
||||||
|
57431 => .media_reverse,
|
||||||
|
57432 => .media_stop,
|
||||||
|
57433 => .media_fast_forward,
|
||||||
|
57434 => .media_rewind,
|
||||||
|
57435 => .media_track_next,
|
||||||
|
57436 => .media_track_previous,
|
||||||
|
57437 => .media_record,
|
||||||
|
57438 => .lower_volume,
|
||||||
|
57439 => .raise_volume,
|
||||||
|
57440 => .mute_volume,
|
||||||
|
57441 => .left_shift,
|
||||||
|
57442 => .left_control,
|
||||||
|
57443 => .left_alt,
|
||||||
|
57444 => .left_super,
|
||||||
|
57445 => .left_hyper,
|
||||||
|
57446 => .left_meta,
|
||||||
|
57447 => .right_shift,
|
||||||
|
57448 => .right_control,
|
||||||
|
57449 => .right_alt,
|
||||||
|
57450 => .right_super,
|
||||||
|
57451 => .right_hyper,
|
||||||
|
57452 => .right_meta,
|
||||||
|
57453 => .iso_level3_shift,
|
||||||
|
57454 => .iso_level5_shift,
|
||||||
|
else => null,
|
||||||
|
},
|
||||||
|
else => null,
|
||||||
|
};
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
fn updateWinSize(self: *Terminal) !void {
|
fn updateWinSize(self: *Terminal) !void {
|
||||||
var sz: os.linux.winsize = undefined;
|
var sz: posix.winsize = undefined;
|
||||||
const ret = os.linux.ioctl(0, os.linux.T.IOCGWINSZ, @intFromPtr(&sz));
|
const ret = os.linux.ioctl(0, os.linux.T.IOCGWINSZ, @intFromPtr(&sz));
|
||||||
// std.debug.print("ret: {d}, {any}\r\n", .{ ret, sz });
|
// std.debug.print("ret: {d}, {any}\r\n", .{ ret, sz });
|
||||||
if (ret == 0) {
|
if (ret == 0) {
|
||||||
self.box.rect = .{
|
self.box.rect = .{
|
||||||
.x = 0,
|
.x = 0,
|
||||||
.y = 0,
|
.y = 0,
|
||||||
.h = @intCast(sz.ws_row),
|
.h = @intCast(sz.row),
|
||||||
.w = @intCast(sz.ws_col),
|
.w = @intCast(sz.col),
|
||||||
};
|
};
|
||||||
self.box.makeDirty();
|
self.box.makeDirty();
|
||||||
} else unreachable; // TODO: handle else case
|
} else unreachable; // TODO: handle else case
|
||||||
|
@ -482,19 +764,30 @@ pub const Terminal = struct {
|
||||||
var x: u32 = 0;
|
var x: u32 = 0;
|
||||||
while (x < rect.w) : (x += 1) try term.print(" ", .{});
|
while (x < rect.w) : (x += 1) try term.print(" ", .{});
|
||||||
}
|
}
|
||||||
try term.cursorSet(
|
if (self.content.len > 0) {
|
||||||
switch (self.content_halign) {
|
const lines = mem.count(u8, self.content, "\n");
|
||||||
|
const x = switch (self.content_halign) {
|
||||||
.left => rect.x,
|
.left => rect.x,
|
||||||
.center => rect.x + (rect.w / 2) - (@as(u32, @intCast(self.content.len)) / 2),
|
.center => rect.x + (rect.w / 2) - (@as(u32, @intCast(self.content.len)) / 2),
|
||||||
.right => rect.x + rect.w - (@as(u32, @intCast(self.content.len))),
|
.right => rect.x + rect.w - (@as(u32, @intCast(self.content.len))),
|
||||||
},
|
};
|
||||||
switch (self.content_valign) {
|
y = switch (self.content_valign) {
|
||||||
.top => rect.y,
|
.top => rect.y,
|
||||||
.center => rect.y + (rect.h / 2),
|
.center => rect.y + (rect.h / 2) - (@as(u32, @intCast(lines)) / 2),
|
||||||
.bottom => rect.y + rect.h - 1,
|
.bottom => rect.y + rect.h - 1 - @as(u32, @intCast(lines)),
|
||||||
},
|
};
|
||||||
);
|
try term.cursorSet(x, y);
|
||||||
try term.print("{s}", .{self.content});
|
|
||||||
|
var split = mem.splitScalar(u8, self.content, '\n');
|
||||||
|
while (split.next()) |s| {
|
||||||
|
var spliit = mem.splitScalar(u8, s, '\r');
|
||||||
|
while (spliit.next()) |c| {
|
||||||
|
try term.cursorSet(x, y);
|
||||||
|
try term.print("{s}", .{c});
|
||||||
|
}
|
||||||
|
y += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
self.dirty = false;
|
self.dirty = false;
|
||||||
|
@ -529,4 +822,30 @@ pub const Terminal = struct {
|
||||||
} else return self.border_bg;
|
} else return self.border_bg;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const KittyFlags = struct {
|
||||||
|
/// Disambiguate escape codes
|
||||||
|
disambiguate: bool = true,
|
||||||
|
/// Report event types
|
||||||
|
event_types: bool = true,
|
||||||
|
/// Report alternate keys
|
||||||
|
alt_keys: bool = true,
|
||||||
|
/// Report all keys as escape codes
|
||||||
|
all_escapes: bool = true,
|
||||||
|
/// Report associated text
|
||||||
|
associated_text: bool = true,
|
||||||
|
|
||||||
|
fn asInt(flags: KittyFlags) u5 {
|
||||||
|
const d: u5 = if (flags.disambiguate) 1 else 0;
|
||||||
|
const e: u5 = if (flags.event_types) 1 else 0;
|
||||||
|
const k: u5 = if (flags.alt_keys) 1 else 0;
|
||||||
|
const a: u5 = if (flags.all_escapes) 1 else 0;
|
||||||
|
const t: u5 = if (flags.associated_text) 1 else 0;
|
||||||
|
return d << 0 |
|
||||||
|
e << 1 |
|
||||||
|
k << 2 |
|
||||||
|
a << 3 |
|
||||||
|
t << 4;
|
||||||
|
}
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
10
src/root.zig
10
src/root.zig
|
@ -1,10 +0,0 @@
|
||||||
const std = @import("std");
|
|
||||||
const testing = std.testing;
|
|
||||||
|
|
||||||
export fn add(a: i32, b: i32) i32 {
|
|
||||||
return a + b;
|
|
||||||
}
|
|
||||||
|
|
||||||
test "basic add functionality" {
|
|
||||||
try testing.expect(add(3, 7) == 10);
|
|
||||||
}
|
|
|
@ -6,7 +6,7 @@ const io = std.io;
|
||||||
const Self = @This();
|
const Self = @This();
|
||||||
|
|
||||||
pub fn init(allocator: mem.Allocator) !Self {
|
pub fn init(allocator: mem.Allocator) !Self {
|
||||||
const result = try std.ChildProcess.run(.{
|
const result = try std.process.Child.run(.{
|
||||||
.allocator = allocator,
|
.allocator = allocator,
|
||||||
.argv = &[_][]const u8{ "infocmp", "-x" },
|
.argv = &[_][]const u8{ "infocmp", "-x" },
|
||||||
});
|
});
|
||||||
|
|
Loading…
Add table
Reference in a new issue