3146 lines
113 KiB
Zig
3146 lines
113 KiB
Zig
const streampkg = @This();
|
|
const std = @import("std");
|
|
const build_options = @import("terminal_options");
|
|
const assert = std.debug.assert;
|
|
const testing = std.testing;
|
|
const simd = @import("../simd/main.zig");
|
|
const lib = @import("../lib/main.zig");
|
|
const Parser = @import("Parser.zig");
|
|
const ansi = @import("ansi.zig");
|
|
const charsets = @import("charsets.zig");
|
|
const device_status = @import("device_status.zig");
|
|
const csi = @import("csi.zig");
|
|
const kitty = @import("kitty.zig");
|
|
const modes = @import("modes.zig");
|
|
const osc = @import("osc.zig");
|
|
const sgr = @import("sgr.zig");
|
|
const UTF8Decoder = @import("UTF8Decoder.zig");
|
|
const MouseShape = @import("mouse_shape.zig").MouseShape;
|
|
|
|
const log = std.log.scoped(.stream);
|
|
|
|
/// Flip this to true when you want verbose debug output for
|
|
/// debugging terminal stream issues. In addition to louder
|
|
/// output this will also disable the SIMD optimizations in
|
|
/// order to make it easier to see every byte. So if you're
|
|
/// debugging an issue in the SIMD code then you'll need to
|
|
/// do something else.
|
|
const debug = false;
|
|
|
|
const lib_target: lib.Target = if (build_options.c_abi) .c else .zig;
|
|
|
|
pub const Action = union(Key) {
|
|
print: Print,
|
|
print_repeat: usize,
|
|
bell,
|
|
backspace,
|
|
horizontal_tab: u16,
|
|
horizontal_tab_back: u16,
|
|
linefeed,
|
|
carriage_return,
|
|
enquiry,
|
|
invoke_charset: InvokeCharset,
|
|
cursor_up: CursorMovement,
|
|
cursor_down: CursorMovement,
|
|
cursor_left: CursorMovement,
|
|
cursor_right: CursorMovement,
|
|
cursor_col: CursorMovement,
|
|
cursor_row: CursorMovement,
|
|
cursor_col_relative: CursorMovement,
|
|
cursor_row_relative: CursorMovement,
|
|
cursor_pos: CursorPos,
|
|
erase_display_below: bool,
|
|
erase_display_above: bool,
|
|
erase_display_complete: bool,
|
|
erase_display_scrollback: bool,
|
|
erase_display_scroll_complete: bool,
|
|
erase_line_right: bool,
|
|
erase_line_left: bool,
|
|
erase_line_complete: bool,
|
|
erase_line_right_unless_pending_wrap: bool,
|
|
delete_chars: usize,
|
|
erase_chars: usize,
|
|
insert_lines: usize,
|
|
insert_blanks: usize,
|
|
delete_lines: usize,
|
|
scroll_up: usize,
|
|
scroll_down: usize,
|
|
tab_clear_current,
|
|
tab_clear_all,
|
|
tab_set,
|
|
tab_reset,
|
|
set_mode: Mode,
|
|
reset_mode: Mode,
|
|
save_mode: Mode,
|
|
restore_mode: Mode,
|
|
request_mode: Mode,
|
|
request_mode_unknown: RawMode,
|
|
top_and_bottom_margin: Margin,
|
|
left_and_right_margin: Margin,
|
|
left_and_right_margin_ambiguous,
|
|
save_cursor,
|
|
restore_cursor,
|
|
modify_key_format: ansi.ModifyKeyFormat,
|
|
protected_mode_off,
|
|
protected_mode_iso,
|
|
protected_mode_dec,
|
|
xtversion,
|
|
kitty_keyboard_query,
|
|
kitty_keyboard_push: KittyKeyboardFlags,
|
|
kitty_keyboard_pop: u16,
|
|
kitty_keyboard_set: KittyKeyboardFlags,
|
|
kitty_keyboard_set_or: KittyKeyboardFlags,
|
|
kitty_keyboard_set_not: KittyKeyboardFlags,
|
|
prompt_end,
|
|
end_of_input,
|
|
end_hyperlink,
|
|
decaln,
|
|
|
|
pub const Key = lib.Enum(
|
|
lib_target,
|
|
&.{
|
|
"print",
|
|
"print_repeat",
|
|
"bell",
|
|
"backspace",
|
|
"horizontal_tab",
|
|
"horizontal_tab_back",
|
|
"linefeed",
|
|
"carriage_return",
|
|
"enquiry",
|
|
"invoke_charset",
|
|
"cursor_up",
|
|
"cursor_down",
|
|
"cursor_left",
|
|
"cursor_right",
|
|
"cursor_col",
|
|
"cursor_row",
|
|
"cursor_col_relative",
|
|
"cursor_row_relative",
|
|
"cursor_pos",
|
|
"erase_display_below",
|
|
"erase_display_above",
|
|
"erase_display_complete",
|
|
"erase_display_scrollback",
|
|
"erase_display_scroll_complete",
|
|
"erase_line_right",
|
|
"erase_line_left",
|
|
"erase_line_complete",
|
|
"erase_line_right_unless_pending_wrap",
|
|
"delete_chars",
|
|
"erase_chars",
|
|
"insert_lines",
|
|
"insert_blanks",
|
|
"delete_lines",
|
|
"scroll_up",
|
|
"scroll_down",
|
|
"tab_clear_current",
|
|
"tab_clear_all",
|
|
"tab_set",
|
|
"tab_reset",
|
|
"set_mode",
|
|
"reset_mode",
|
|
"save_mode",
|
|
"restore_mode",
|
|
"request_mode",
|
|
"request_mode_unknown",
|
|
"top_and_bottom_margin",
|
|
"left_and_right_margin",
|
|
"left_and_right_margin_ambiguous",
|
|
"save_cursor",
|
|
"restore_cursor",
|
|
"modify_key_format",
|
|
"protected_mode_off",
|
|
"protected_mode_iso",
|
|
"protected_mode_dec",
|
|
"xtversion",
|
|
"kitty_keyboard_query",
|
|
"kitty_keyboard_push",
|
|
"kitty_keyboard_pop",
|
|
"kitty_keyboard_set",
|
|
"kitty_keyboard_set_or",
|
|
"kitty_keyboard_set_not",
|
|
"prompt_end",
|
|
"end_of_input",
|
|
"end_hyperlink",
|
|
"decaln",
|
|
},
|
|
);
|
|
|
|
/// C ABI functions.
|
|
const c_union = lib.TaggedUnion(
|
|
lib_target,
|
|
@This(),
|
|
// TODO: Before shipping an ABI-compatible libghostty, verify this.
|
|
// This was just arbitrarily chosen for now.
|
|
[8]u64,
|
|
);
|
|
pub const Tag = c_union.Tag;
|
|
pub const Value = c_union.Value;
|
|
pub const C = c_union.C;
|
|
pub const CValue = c_union.CValue;
|
|
pub const cval = c_union.cval;
|
|
|
|
/// Field types
|
|
pub const Print = struct {
|
|
cp: u21,
|
|
|
|
pub const C = extern struct {
|
|
cp: u32,
|
|
};
|
|
|
|
pub fn cval(self: Print) Print.C {
|
|
return .{ .cp = @intCast(self.cp) };
|
|
}
|
|
};
|
|
|
|
pub const InvokeCharset = lib.Struct(lib_target, struct {
|
|
bank: charsets.ActiveSlot,
|
|
charset: charsets.Slots,
|
|
locking: bool,
|
|
});
|
|
|
|
pub const CursorMovement = extern struct {
|
|
/// The value of the cursor movement. Depending on the tag of this
|
|
/// union this may be an absolute value or it may be a relative
|
|
/// value. For example, `cursor_up` is relative, but `cursor_row`
|
|
/// is absolute.
|
|
value: u16,
|
|
};
|
|
|
|
pub const CursorPos = extern struct {
|
|
row: u16,
|
|
col: u16,
|
|
};
|
|
|
|
pub const Mode = struct {
|
|
mode: modes.Mode,
|
|
|
|
pub const C = u16;
|
|
|
|
pub fn cval(self: Mode) Mode.C {
|
|
return @bitCast(self.mode);
|
|
}
|
|
};
|
|
|
|
pub const RawMode = extern struct {
|
|
mode: u16,
|
|
ansi: bool,
|
|
};
|
|
|
|
pub const Margin = extern struct {
|
|
top_left: u16,
|
|
bottom_right: u16,
|
|
};
|
|
|
|
pub const KittyKeyboardFlags = struct {
|
|
flags: kitty.KeyFlags,
|
|
|
|
pub const C = u8;
|
|
|
|
pub fn cval(self: KittyKeyboardFlags) KittyKeyboardFlags.C {
|
|
return @intCast(self.flags.int());
|
|
}
|
|
};
|
|
};
|
|
|
|
/// Returns a type that can process a stream of tty control characters.
|
|
/// This will call various callback functions on type T. Type T only has to
|
|
/// implement the callbacks it cares about; any unimplemented callbacks will
|
|
/// logged at runtime.
|
|
///
|
|
/// To figure out what callbacks exist, search the source for "hasDecl". This
|
|
/// isn't ideal but for now that's the best approach.
|
|
///
|
|
/// This is implemented this way because we purposely do NOT want dynamic
|
|
/// dispatch for performance reasons. The way this is implemented forces
|
|
/// comptime resolution for all function calls.
|
|
pub fn Stream(comptime Handler: type) type {
|
|
return struct {
|
|
const Self = @This();
|
|
|
|
pub const Action = streampkg.Action;
|
|
|
|
// We use T with @hasDecl so it needs to be a struct. Unwrap the
|
|
// pointer if we were given one.
|
|
const T = switch (@typeInfo(Handler)) {
|
|
.pointer => |p| p.child,
|
|
else => Handler,
|
|
};
|
|
|
|
handler: Handler,
|
|
parser: Parser,
|
|
utf8decoder: UTF8Decoder,
|
|
|
|
pub fn init(h: Handler) Self {
|
|
return .{
|
|
.handler = h,
|
|
.parser = .init(),
|
|
.utf8decoder = .{},
|
|
};
|
|
}
|
|
|
|
pub fn deinit(self: *Self) void {
|
|
self.parser.deinit();
|
|
}
|
|
|
|
/// Process a string of characters.
|
|
pub inline fn nextSlice(self: *Self, input: []const u8) !void {
|
|
// Disable SIMD optimizations if build requests it or if our
|
|
// manual debug mode is on.
|
|
if (comptime debug or !build_options.simd) {
|
|
for (input) |c| try self.next(c);
|
|
return;
|
|
}
|
|
|
|
// This is the maximum number of codepoints we can decode
|
|
// at one time for this function call. This is somewhat arbitrary
|
|
// so if someone can demonstrate a better number then we can switch.
|
|
var cp_buf: [4096]u32 = undefined;
|
|
|
|
// Split the input into chunks that fit into cp_buf.
|
|
var i: usize = 0;
|
|
while (true) {
|
|
const len = @min(cp_buf.len, input.len - i);
|
|
try self.nextSliceCapped(input[i .. i + len], &cp_buf);
|
|
i += len;
|
|
if (i >= input.len) break;
|
|
}
|
|
}
|
|
|
|
inline fn nextSliceCapped(self: *Self, input: []const u8, cp_buf: []u32) !void {
|
|
assert(input.len <= cp_buf.len);
|
|
|
|
var offset: usize = 0;
|
|
|
|
// If the scalar UTF-8 decoder was in the middle of processing
|
|
// a code sequence, we continue until it's not.
|
|
while (self.utf8decoder.state != 0) {
|
|
if (offset >= input.len) return;
|
|
try self.nextUtf8(input[offset]);
|
|
offset += 1;
|
|
}
|
|
if (offset >= input.len) return;
|
|
|
|
// If we're not in the ground state then we process until
|
|
// we are. This can happen if the last chunk of input put us
|
|
// in the middle of a control sequence.
|
|
offset += try self.consumeUntilGround(input[offset..]);
|
|
if (offset >= input.len) return;
|
|
offset += try self.consumeAllEscapes(input[offset..]);
|
|
|
|
// If we're in the ground state then we can use SIMD to process
|
|
// input until we see an ESC (0x1B), since all other characters
|
|
// up to that point are just UTF-8.
|
|
while (self.parser.state == .ground and offset < input.len) {
|
|
const res = simd.vt.utf8DecodeUntilControlSeq(input[offset..], cp_buf);
|
|
for (cp_buf[0..res.decoded]) |cp| {
|
|
if (cp <= 0xF) {
|
|
try self.execute(@intCast(cp));
|
|
} else {
|
|
try self.print(@intCast(cp));
|
|
}
|
|
}
|
|
// Consume the bytes we just processed.
|
|
offset += res.consumed;
|
|
|
|
if (offset >= input.len) return;
|
|
|
|
// If our offset is NOT an escape then we must have a
|
|
// partial UTF-8 sequence. In that case, we pass it off
|
|
// to the scalar parser.
|
|
if (input[offset] != 0x1B) {
|
|
const rem = input[offset..];
|
|
for (rem) |c| try self.nextUtf8(c);
|
|
return;
|
|
}
|
|
|
|
// Process control sequences until we run out.
|
|
offset += try self.consumeAllEscapes(input[offset..]);
|
|
}
|
|
}
|
|
|
|
/// Parses back-to-back escape sequences until none are left.
|
|
/// Returns the number of bytes consumed from the provided input.
|
|
///
|
|
/// Expects input to start with 0x1B, use consumeUntilGround first
|
|
/// if the stream may be in the middle of an escape sequence.
|
|
inline fn consumeAllEscapes(self: *Self, input: []const u8) !usize {
|
|
var offset: usize = 0;
|
|
while (input[offset] == 0x1B) {
|
|
self.parser.state = .escape;
|
|
self.parser.clear();
|
|
offset += 1;
|
|
offset += try self.consumeUntilGround(input[offset..]);
|
|
if (offset >= input.len) return input.len;
|
|
}
|
|
return offset;
|
|
}
|
|
|
|
/// Parses escape sequences until the parser reaches the ground state.
|
|
/// Returns the number of bytes consumed from the provided input.
|
|
inline fn consumeUntilGround(self: *Self, input: []const u8) !usize {
|
|
var offset: usize = 0;
|
|
while (self.parser.state != .ground) {
|
|
if (offset >= input.len) return input.len;
|
|
try self.nextNonUtf8(input[offset]);
|
|
offset += 1;
|
|
}
|
|
return offset;
|
|
}
|
|
|
|
/// Like nextSlice but takes one byte and is necessarily a scalar
|
|
/// operation that can't use SIMD. Prefer nextSlice if you can and
|
|
/// try to get multiple bytes at once.
|
|
pub inline fn next(self: *Self, c: u8) !void {
|
|
// The scalar path can be responsible for decoding UTF-8.
|
|
if (self.parser.state == .ground) {
|
|
try self.nextUtf8(c);
|
|
return;
|
|
}
|
|
|
|
try self.nextNonUtf8(c);
|
|
}
|
|
|
|
/// Process the next byte and print as necessary.
|
|
///
|
|
/// This assumes we're in the UTF-8 decoding state. If we may not
|
|
/// be in the UTF-8 decoding state call nextSlice or next.
|
|
inline fn nextUtf8(self: *Self, c: u8) !void {
|
|
assert(self.parser.state == .ground);
|
|
|
|
const res = self.utf8decoder.next(c);
|
|
const consumed = res[1];
|
|
if (res[0]) |codepoint| {
|
|
try self.handleCodepoint(codepoint);
|
|
}
|
|
if (!consumed) {
|
|
const retry = self.utf8decoder.next(c);
|
|
// It should be impossible for the decoder
|
|
// to not consume the byte twice in a row.
|
|
assert(retry[1] == true);
|
|
if (retry[0]) |codepoint| {
|
|
try self.handleCodepoint(codepoint);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// To be called whenever the utf-8 decoder produces a codepoint.
|
|
///
|
|
/// This function is abstracted this way to handle the case where
|
|
/// the decoder emits a 0x1B after rejecting an ill-formed sequence.
|
|
inline fn handleCodepoint(self: *Self, c: u21) !void {
|
|
if (c <= 0xF) {
|
|
try self.execute(@intCast(c));
|
|
return;
|
|
}
|
|
if (c == 0x1B) {
|
|
try self.nextNonUtf8(@intCast(c));
|
|
return;
|
|
}
|
|
try self.print(@intCast(c));
|
|
}
|
|
|
|
/// Process the next character and call any callbacks if necessary.
|
|
///
|
|
/// This assumes that we're not in the UTF-8 decoding state. If
|
|
/// we may be in the UTF-8 decoding state call nextSlice or next.
|
|
fn nextNonUtf8(self: *Self, c: u8) !void {
|
|
assert(self.parser.state != .ground or c == 0x1B);
|
|
|
|
// Fast path for ESC
|
|
if (self.parser.state == .ground and c == 0x1B) {
|
|
self.parser.state = .escape;
|
|
self.parser.clear();
|
|
return;
|
|
}
|
|
// Fast path for CSI entry.
|
|
if (self.parser.state == .escape and c == '[') {
|
|
self.parser.state = .csi_entry;
|
|
return;
|
|
}
|
|
// Fast path for CSI params.
|
|
if (self.parser.state == .csi_param) csi_param: {
|
|
switch (c) {
|
|
// A C0 escape (yes, this is valid):
|
|
0x00...0x0F => try self.execute(c),
|
|
// We ignore C0 escapes > 0xF since execute
|
|
// doesn't have processing for them anyway:
|
|
0x10...0x17, 0x19, 0x1C...0x1F => {},
|
|
// We don't currently have any handling for
|
|
// 0x18 or 0x1A, but they should still move
|
|
// the parser state to ground.
|
|
0x18, 0x1A => self.parser.state = .ground,
|
|
// A parameter digit:
|
|
'0'...'9' => if (self.parser.params_idx < Parser.MAX_PARAMS) {
|
|
self.parser.param_acc *|= 10;
|
|
self.parser.param_acc +|= c - '0';
|
|
// The parser's CSI param action uses param_acc_idx
|
|
// to decide if there's a final param that needs to
|
|
// be consumed or not, but it doesn't matter really
|
|
// what it is as long as it's not 0.
|
|
self.parser.param_acc_idx |= 1;
|
|
},
|
|
// A parameter separator:
|
|
':', ';' => if (self.parser.params_idx < Parser.MAX_PARAMS) {
|
|
self.parser.params[self.parser.params_idx] = self.parser.param_acc;
|
|
if (c == ':') self.parser.params_sep.set(self.parser.params_idx);
|
|
self.parser.params_idx += 1;
|
|
|
|
self.parser.param_acc = 0;
|
|
self.parser.param_acc_idx = 0;
|
|
},
|
|
// Explicitly ignored:
|
|
0x7F => {},
|
|
// Defer to the state machine to
|
|
// handle any other characters:
|
|
else => break :csi_param,
|
|
}
|
|
return;
|
|
}
|
|
|
|
// We explicitly inline this call here for performance reasons.
|
|
//
|
|
// We do this rather than mark Parser.next as inline because doing
|
|
// that causes weird behavior in some tests- I'm not sure if they
|
|
// miscompile or it's just very counter-intuitive comptime stuff,
|
|
// but regardless, this is the easy solution.
|
|
const actions = @call(.always_inline, Parser.next, .{ &self.parser, c });
|
|
|
|
for (actions) |action_opt| {
|
|
const action = action_opt orelse continue;
|
|
if (comptime debug) log.info("action: {f}", .{action});
|
|
|
|
// If this handler handles everything manually then we do nothing
|
|
// if it can be processed.
|
|
if (@hasDecl(T, "handleManually")) {
|
|
const processed = self.handler.handleManually(action) catch |err| err: {
|
|
log.warn("error handling action manually err={} action={f}", .{
|
|
err,
|
|
action,
|
|
});
|
|
|
|
break :err false;
|
|
};
|
|
|
|
if (processed) continue;
|
|
}
|
|
|
|
switch (action) {
|
|
.print => |p| try self.print(p),
|
|
.execute => |code| try self.execute(code),
|
|
.csi_dispatch => |csi_action| try self.csiDispatch(csi_action),
|
|
.esc_dispatch => |esc| try self.escDispatch(esc),
|
|
.osc_dispatch => |cmd| try self.oscDispatch(cmd),
|
|
.dcs_hook => |dcs| if (@hasDecl(T, "dcsHook")) {
|
|
try self.handler.dcsHook(dcs);
|
|
} else log.warn("unimplemented DCS hook", .{}),
|
|
.dcs_put => |code| if (@hasDecl(T, "dcsPut")) {
|
|
try self.handler.dcsPut(code);
|
|
} else log.warn("unimplemented DCS put: {x}", .{code}),
|
|
.dcs_unhook => if (@hasDecl(T, "dcsUnhook")) {
|
|
try self.handler.dcsUnhook();
|
|
} else log.warn("unimplemented DCS unhook", .{}),
|
|
.apc_start => if (@hasDecl(T, "apcStart")) {
|
|
try self.handler.apcStart();
|
|
} else log.warn("unimplemented APC start", .{}),
|
|
.apc_put => |code| if (@hasDecl(T, "apcPut")) {
|
|
try self.handler.apcPut(code);
|
|
} else log.warn("unimplemented APC put: {x}", .{code}),
|
|
.apc_end => if (@hasDecl(T, "apcEnd")) {
|
|
try self.handler.apcEnd();
|
|
} else log.warn("unimplemented APC end", .{}),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub inline fn print(self: *Self, c: u21) !void {
|
|
try self.handler.vt(.print, .{ .cp = c });
|
|
}
|
|
|
|
pub inline fn execute(self: *Self, c: u8) !void {
|
|
const c0: ansi.C0 = @enumFromInt(c);
|
|
if (comptime debug) log.info("execute: {f}", .{c0});
|
|
switch (c0) {
|
|
// We ignore SOH/STX: https://github.com/microsoft/terminal/issues/10786
|
|
.NUL, .SOH, .STX => {},
|
|
|
|
.ENQ => try self.handler.vt(.enquiry, {}),
|
|
.BEL => try self.handler.vt(.bell, {}),
|
|
.BS => try self.handler.vt(.backspace, {}),
|
|
.HT => try self.handler.vt(.horizontal_tab, 1),
|
|
.LF, .VT, .FF => try self.handler.vt(.linefeed, {}),
|
|
.CR => try self.handler.vt(.carriage_return, {}),
|
|
.SO => try self.handler.vt(.invoke_charset, .{ .bank = .GL, .charset = .G1, .locking = false }),
|
|
.SI => try self.handler.vt(.invoke_charset, .{ .bank = .GL, .charset = .G0, .locking = false }),
|
|
|
|
else => log.warn("invalid C0 character, ignoring: 0x{x}", .{c}),
|
|
}
|
|
}
|
|
|
|
inline fn csiDispatch(self: *Self, input: Parser.Action.CSI) !void {
|
|
switch (input.final) {
|
|
// CUU - Cursor Up
|
|
'A', 'k' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.cursor_up, .{
|
|
.value = switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid cursor up command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI A with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// CUD - Cursor Down
|
|
'B' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.cursor_down, .{
|
|
.value = switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid cursor down command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI B with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// CUF - Cursor Right
|
|
'C' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.cursor_right, .{
|
|
.value = switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid cursor right command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI C with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// CUB - Cursor Left
|
|
'D', 'j' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.cursor_left, .{
|
|
.value = switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid cursor left command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI D with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// CNL - Cursor Next Line
|
|
'E' => switch (input.intermediates.len) {
|
|
0 => {
|
|
try self.handler.vt(.cursor_down, .{
|
|
.value = switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid cursor up command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
});
|
|
try self.handler.vt(.carriage_return, {});
|
|
},
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI E with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// CPL - Cursor Previous Line
|
|
'F' => switch (input.intermediates.len) {
|
|
0 => {
|
|
try self.handler.vt(.cursor_up, .{
|
|
.value = switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid cursor down command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
});
|
|
try self.handler.vt(.carriage_return, {});
|
|
},
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI F with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// HPA - Cursor Horizontal Position Absolute
|
|
// TODO: test
|
|
'G', '`' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.cursor_col, .{
|
|
.value = switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid HPA command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI G with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// CUP - Set Cursor Position.
|
|
// TODO: test
|
|
'H', 'f' => switch (input.intermediates.len) {
|
|
0 => {
|
|
const pos: streampkg.Action.CursorPos = switch (input.params.len) {
|
|
0 => .{ .row = 1, .col = 1 },
|
|
1 => .{ .row = input.params[0], .col = 1 },
|
|
2 => .{ .row = input.params[0], .col = input.params[1] },
|
|
else => {
|
|
log.warn("invalid CUP command: {f}", .{input});
|
|
return;
|
|
},
|
|
};
|
|
try self.handler.vt(.cursor_pos, pos);
|
|
},
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI H with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// CHT - Cursor Horizontal Tabulation
|
|
'I' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.horizontal_tab, switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid horizontal tab command: {f}", .{input});
|
|
return;
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI I with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// Erase Display
|
|
'J' => {
|
|
const protected_: ?bool = switch (input.intermediates.len) {
|
|
0 => false,
|
|
1 => if (input.intermediates[0] == '?') true else null,
|
|
else => null,
|
|
};
|
|
|
|
const protected = protected_ orelse {
|
|
log.warn("invalid erase display command: {f}", .{input});
|
|
return;
|
|
};
|
|
|
|
const mode_: ?csi.EraseDisplay = switch (input.params.len) {
|
|
0 => .below,
|
|
1 => std.meta.intToEnum(csi.EraseDisplay, input.params[0]) catch null,
|
|
else => null,
|
|
};
|
|
|
|
const mode = mode_ orelse {
|
|
log.warn("invalid erase display command: {f}", .{input});
|
|
return;
|
|
};
|
|
|
|
switch (mode) {
|
|
.below => try self.handler.vt(.erase_display_below, protected),
|
|
.above => try self.handler.vt(.erase_display_above, protected),
|
|
.complete => try self.handler.vt(.erase_display_complete, protected),
|
|
.scrollback => try self.handler.vt(.erase_display_scrollback, protected),
|
|
.scroll_complete => try self.handler.vt(.erase_display_scroll_complete, protected),
|
|
}
|
|
},
|
|
|
|
// Erase Line
|
|
'K' => {
|
|
const protected_: ?bool = switch (input.intermediates.len) {
|
|
0 => false,
|
|
1 => if (input.intermediates[0] == '?') true else null,
|
|
else => null,
|
|
};
|
|
|
|
const protected = protected_ orelse {
|
|
log.warn("invalid erase line command: {f}", .{input});
|
|
return;
|
|
};
|
|
|
|
const mode_: ?csi.EraseLine = switch (input.params.len) {
|
|
0 => .right,
|
|
1 => if (input.params[0] < 3) @enumFromInt(input.params[0]) else null,
|
|
else => null,
|
|
};
|
|
|
|
const mode = mode_ orelse {
|
|
log.warn("invalid erase line command: {f}", .{input});
|
|
return;
|
|
};
|
|
|
|
switch (mode) {
|
|
.right => try self.handler.vt(.erase_line_right, protected),
|
|
.left => try self.handler.vt(.erase_line_left, protected),
|
|
.complete => try self.handler.vt(.erase_line_complete, protected),
|
|
.right_unless_pending_wrap => try self.handler.vt(.erase_line_right_unless_pending_wrap, protected),
|
|
_ => log.warn("invalid erase line mode: {}", .{mode}),
|
|
}
|
|
},
|
|
|
|
// IL - Insert Lines
|
|
// TODO: test
|
|
'L' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.insert_lines, switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid IL command: {f}", .{input});
|
|
return;
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI L with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// DL - Delete Lines
|
|
// TODO: test
|
|
'M' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.delete_lines, switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid DL command: {f}", .{input});
|
|
return;
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI M with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// Delete Character (DCH)
|
|
'P' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.delete_chars, switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid delete characters command: {f}", .{input});
|
|
return;
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI P with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// Scroll Up (SD)
|
|
|
|
'S' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.scroll_up, switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid scroll up command: {f}", .{input});
|
|
return;
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI S with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// Scroll Down (SD)
|
|
'T' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.scroll_down, switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid scroll down command: {f}", .{input});
|
|
return;
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI T with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// Cursor Tabulation Control
|
|
'W' => switch (input.intermediates.len) {
|
|
0 => {
|
|
if (input.params.len == 0 or
|
|
(input.params.len == 1 and input.params[0] == 0))
|
|
{
|
|
try self.handler.vt(.tab_set, {});
|
|
return;
|
|
}
|
|
|
|
switch (input.params.len) {
|
|
0 => unreachable,
|
|
|
|
1 => switch (input.params[0]) {
|
|
0 => unreachable,
|
|
|
|
2 => try self.handler.vt(.tab_clear_current, {}),
|
|
|
|
5 => try self.handler.vt(.tab_clear_all, {}),
|
|
|
|
else => {},
|
|
},
|
|
|
|
else => {},
|
|
}
|
|
|
|
log.warn("invalid cursor tabulation control: {f}", .{input});
|
|
return;
|
|
},
|
|
|
|
1 => if (input.intermediates[0] == '?' and input.params[0] == 5) {
|
|
try self.handler.vt(.tab_reset, {});
|
|
} else log.warn("invalid cursor tabulation control: {f}", .{input}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI W with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// Erase Characters (ECH)
|
|
'X' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.erase_chars, switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid erase characters command: {f}", .{input});
|
|
return;
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI X with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// CHT - Cursor Horizontal Tabulation Back
|
|
'Z' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.horizontal_tab_back, switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid horizontal tab back command: {f}", .{input});
|
|
return;
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI Z with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// HPR - Cursor Horizontal Position Relative
|
|
'a' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.cursor_col_relative, .{
|
|
.value = switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid HPR command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI a with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// Repeat Previous Char (REP)
|
|
'b' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.print_repeat, switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid print repeat command: {f}", .{input});
|
|
return;
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI b with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// c - Device Attributes (DA1)
|
|
'c' => if (@hasDecl(T, "deviceAttributes")) {
|
|
const req: ansi.DeviceAttributeReq = switch (input.intermediates.len) {
|
|
0 => ansi.DeviceAttributeReq.primary,
|
|
1 => switch (input.intermediates[0]) {
|
|
'>' => ansi.DeviceAttributeReq.secondary,
|
|
'=' => ansi.DeviceAttributeReq.tertiary,
|
|
else => null,
|
|
},
|
|
else => @as(?ansi.DeviceAttributeReq, null),
|
|
} orelse {
|
|
log.warn("invalid device attributes command: {f}", .{input});
|
|
return;
|
|
};
|
|
|
|
try self.handler.deviceAttributes(req, input.params);
|
|
} else log.warn("unimplemented CSI callback: {f}", .{input}),
|
|
|
|
// VPA - Cursor Vertical Position Absolute
|
|
'd' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.cursor_row, .{
|
|
.value = switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid VPA command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI d with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// VPR - Cursor Vertical Position Relative
|
|
'e' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.cursor_row_relative, .{
|
|
.value = switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid VPR command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI e with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// TBC - Tab Clear
|
|
// TODO: test
|
|
'g' => switch (input.intermediates.len) {
|
|
0 => {
|
|
const mode: csi.TabClear = switch (input.params.len) {
|
|
1 => @enumFromInt(input.params[0]),
|
|
else => {
|
|
log.warn("invalid tab clear command: {f}", .{input});
|
|
return;
|
|
},
|
|
};
|
|
switch (mode) {
|
|
.current => try self.handler.vt(.tab_clear_current, {}),
|
|
.all => try self.handler.vt(.tab_clear_all, {}),
|
|
_ => log.warn("unknown tab clear mode: {}", .{mode}),
|
|
}
|
|
},
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI g with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
// SM - Set Mode
|
|
'h' => mode: {
|
|
const ansi_mode = ansi: {
|
|
if (input.intermediates.len == 0) break :ansi true;
|
|
if (input.intermediates.len == 1 and
|
|
input.intermediates[0] == '?') break :ansi false;
|
|
|
|
log.warn("invalid set mode command: {f}", .{input});
|
|
break :mode;
|
|
};
|
|
|
|
for (input.params) |mode_int| {
|
|
if (modes.modeFromInt(mode_int, ansi_mode)) |mode| {
|
|
try self.handler.vt(.set_mode, .{ .mode = mode });
|
|
} else {
|
|
log.warn("unimplemented mode: {}", .{mode_int});
|
|
}
|
|
}
|
|
},
|
|
|
|
// RM - Reset Mode
|
|
'l' => mode: {
|
|
const ansi_mode = ansi: {
|
|
if (input.intermediates.len == 0) break :ansi true;
|
|
if (input.intermediates.len == 1 and
|
|
input.intermediates[0] == '?') break :ansi false;
|
|
|
|
log.warn("invalid set mode command: {f}", .{input});
|
|
break :mode;
|
|
};
|
|
|
|
for (input.params) |mode_int| {
|
|
if (modes.modeFromInt(mode_int, ansi_mode)) |mode| {
|
|
try self.handler.vt(.reset_mode, .{ .mode = mode });
|
|
} else {
|
|
log.warn("unimplemented mode: {}", .{mode_int});
|
|
}
|
|
}
|
|
},
|
|
|
|
// SGR - Select Graphic Rendition
|
|
'm' => switch (input.intermediates.len) {
|
|
0 => if (@hasDecl(T, "setAttribute")) {
|
|
// log.info("parse SGR params={any}", .{input.params});
|
|
var p: sgr.Parser = .{
|
|
.params = input.params,
|
|
.params_sep = input.params_sep,
|
|
};
|
|
while (p.next()) |attr| {
|
|
// log.info("SGR attribute: {}", .{attr});
|
|
try self.handler.setAttribute(attr);
|
|
}
|
|
} else log.warn("unimplemented CSI callback: {f}", .{input}),
|
|
|
|
1 => switch (input.intermediates[0]) {
|
|
'>' => blk: {
|
|
if (input.params.len == 0) {
|
|
// Reset
|
|
try self.handler.vt(.modify_key_format, .legacy);
|
|
break :blk;
|
|
}
|
|
|
|
var format: ansi.ModifyKeyFormat = switch (input.params[0]) {
|
|
0 => .legacy,
|
|
1 => .cursor_keys,
|
|
2 => .function_keys,
|
|
4 => .other_keys_none,
|
|
else => {
|
|
log.warn("invalid setModifyKeyFormat: {f}", .{input});
|
|
break :blk;
|
|
},
|
|
};
|
|
|
|
if (input.params.len > 2) {
|
|
log.warn("invalid setModifyKeyFormat: {f}", .{input});
|
|
break :blk;
|
|
}
|
|
|
|
if (input.params.len == 2) {
|
|
switch (format) {
|
|
// We don't support any of the subparams yet for these.
|
|
.legacy => {},
|
|
.cursor_keys => {},
|
|
.function_keys => {},
|
|
|
|
// We only support the numeric form.
|
|
.other_keys_none => switch (input.params[1]) {
|
|
2 => format = .other_keys_numeric,
|
|
else => {},
|
|
},
|
|
.other_keys_numeric_except => {},
|
|
.other_keys_numeric => {},
|
|
}
|
|
}
|
|
|
|
try self.handler.vt(.modify_key_format, format);
|
|
},
|
|
|
|
else => log.warn(
|
|
"unknown CSI m with intermediate: {}",
|
|
.{input.intermediates[0]},
|
|
),
|
|
},
|
|
|
|
else => {
|
|
// Nothing, but I wanted a place to put this comment:
|
|
// there are others forms of CSI m that have intermediates.
|
|
// `vim --clean` uses `CSI ? 4 m` and I don't know what
|
|
// that means. And there is also `CSI > m` which is used
|
|
// to control modifier key reporting formats that we don't
|
|
// support yet.
|
|
log.warn(
|
|
"ignoring unimplemented CSI m with intermediates: {s}",
|
|
.{input.intermediates},
|
|
);
|
|
},
|
|
},
|
|
|
|
// TODO: test
|
|
'n' => {
|
|
// Handle deviceStatusReport first
|
|
if (input.intermediates.len == 0 or
|
|
input.intermediates[0] == '?')
|
|
{
|
|
if (!@hasDecl(T, "deviceStatusReport")) {
|
|
log.warn("unimplemented CSI callback: {f}", .{input});
|
|
return;
|
|
}
|
|
|
|
if (input.params.len != 1) {
|
|
log.warn("invalid device status report command: {f}", .{input});
|
|
return;
|
|
}
|
|
|
|
const question = question: {
|
|
if (input.intermediates.len == 0) break :question false;
|
|
if (input.intermediates.len == 1 and
|
|
input.intermediates[0] == '?') break :question true;
|
|
|
|
log.warn("invalid set mode command: {f}", .{input});
|
|
return;
|
|
};
|
|
|
|
const req = device_status.reqFromInt(input.params[0], question) orelse {
|
|
log.warn("invalid device status report command: {f}", .{input});
|
|
return;
|
|
};
|
|
|
|
try self.handler.deviceStatusReport(req);
|
|
return;
|
|
}
|
|
|
|
// Handle other forms of CSI n
|
|
switch (input.intermediates.len) {
|
|
0 => unreachable, // handled above
|
|
|
|
1 => switch (input.intermediates[0]) {
|
|
'>' => {
|
|
// This isn't strictly correct. CSI > n has parameters that
|
|
// control what exactly is being disabled. However, we
|
|
// only support reverting back to modify other keys in
|
|
// numeric except format.
|
|
try self.handler.vt(.modify_key_format, .other_keys_numeric_except);
|
|
},
|
|
|
|
else => log.warn(
|
|
"unknown CSI n with intermediate: {}",
|
|
.{input.intermediates[0]},
|
|
),
|
|
},
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI n with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
}
|
|
},
|
|
|
|
// DECRQM - Request Mode
|
|
'p' => switch (input.intermediates.len) {
|
|
2 => decrqm: {
|
|
const ansi_mode = ansi: {
|
|
switch (input.intermediates.len) {
|
|
1 => if (input.intermediates[0] == '$') break :ansi true,
|
|
2 => if (input.intermediates[0] == '?' and
|
|
input.intermediates[1] == '$') break :ansi false,
|
|
else => {},
|
|
}
|
|
|
|
log.warn(
|
|
"ignoring unimplemented CSI p with intermediates: {s}",
|
|
.{input.intermediates},
|
|
);
|
|
break :decrqm;
|
|
};
|
|
|
|
if (input.params.len != 1) {
|
|
log.warn("invalid DECRQM command: {f}", .{input});
|
|
break :decrqm;
|
|
}
|
|
|
|
const mode_raw = input.params[0];
|
|
const mode = modes.modeFromInt(mode_raw, ansi_mode);
|
|
if (mode) |m| {
|
|
try self.handler.vt(.request_mode, .{ .mode = m });
|
|
} else {
|
|
try self.handler.vt(.request_mode_unknown, .{
|
|
.mode = mode_raw,
|
|
.ansi = ansi_mode,
|
|
});
|
|
}
|
|
},
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI p with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
'q' => switch (input.intermediates.len) {
|
|
1 => switch (input.intermediates[0]) {
|
|
// DECSCUSR - Select Cursor Style
|
|
// TODO: test
|
|
' ' => {
|
|
if (@hasDecl(T, "setCursorStyle")) try self.handler.setCursorStyle(
|
|
switch (input.params.len) {
|
|
0 => ansi.CursorStyle.default,
|
|
1 => @enumFromInt(input.params[0]),
|
|
else => {
|
|
log.warn("invalid set curor style command: {f}", .{input});
|
|
return;
|
|
},
|
|
},
|
|
) else log.warn("unimplemented CSI callback: {f}", .{input});
|
|
},
|
|
|
|
// DECSCA
|
|
'"' => {
|
|
const mode_: ?ansi.ProtectedMode = switch (input.params.len) {
|
|
else => null,
|
|
0 => .off,
|
|
1 => switch (input.params[0]) {
|
|
0, 2 => .off,
|
|
1 => .dec,
|
|
else => null,
|
|
},
|
|
};
|
|
|
|
const mode = mode_ orelse {
|
|
log.warn("invalid set protected mode command: {f}", .{input});
|
|
return;
|
|
};
|
|
|
|
switch (mode) {
|
|
.off => try self.handler.vt(.protected_mode_off, {}),
|
|
.iso => try self.handler.vt(.protected_mode_iso, {}),
|
|
.dec => try self.handler.vt(.protected_mode_dec, {}),
|
|
}
|
|
},
|
|
|
|
// XTVERSION
|
|
'>' => try self.handler.vt(.xtversion, {}),
|
|
else => {
|
|
log.warn(
|
|
"ignoring unimplemented CSI q with intermediates: {s}",
|
|
.{input.intermediates},
|
|
);
|
|
},
|
|
},
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI p with intermediates: {s}",
|
|
.{input.intermediates},
|
|
),
|
|
},
|
|
|
|
'r' => switch (input.intermediates.len) {
|
|
// DECSTBM - Set Top and Bottom Margins
|
|
0 => switch (input.params.len) {
|
|
0 => try self.handler.vt(.top_and_bottom_margin, .{ .top_left = 0, .bottom_right = 0 }),
|
|
1 => try self.handler.vt(.top_and_bottom_margin, .{ .top_left = input.params[0], .bottom_right = 0 }),
|
|
2 => try self.handler.vt(.top_and_bottom_margin, .{ .top_left = input.params[0], .bottom_right = input.params[1] }),
|
|
else => log.warn("invalid DECSTBM command: {f}", .{input}),
|
|
},
|
|
|
|
1 => switch (input.intermediates[0]) {
|
|
// Restore Mode
|
|
'?' => {
|
|
for (input.params) |mode_int| {
|
|
if (modes.modeFromInt(mode_int, false)) |mode| {
|
|
try self.handler.vt(.restore_mode, .{ .mode = mode });
|
|
} else {
|
|
log.warn(
|
|
"unimplemented restore mode: {}",
|
|
.{mode_int},
|
|
);
|
|
}
|
|
}
|
|
},
|
|
|
|
else => log.warn(
|
|
"unknown CSI s with intermediate: {f}",
|
|
.{input},
|
|
),
|
|
},
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI s with intermediates: {f}",
|
|
.{input},
|
|
),
|
|
},
|
|
|
|
's' => switch (input.intermediates.len) {
|
|
// DECSLRM
|
|
0 => switch (input.params.len) {
|
|
// CSI S is ambiguous with zero params so we defer
|
|
// to our handler to do the proper logic. If mode 69
|
|
// is set, then we should invoke DECSLRM, otherwise
|
|
// we should invoke SC.
|
|
0 => try self.handler.vt(.left_and_right_margin_ambiguous, {}),
|
|
1 => try self.handler.vt(.left_and_right_margin, .{ .top_left = input.params[0], .bottom_right = 0 }),
|
|
2 => try self.handler.vt(.left_and_right_margin, .{ .top_left = input.params[0], .bottom_right = input.params[1] }),
|
|
else => log.warn("invalid DECSLRM command: {f}", .{input}),
|
|
},
|
|
|
|
1 => switch (input.intermediates[0]) {
|
|
'?' => {
|
|
for (input.params) |mode_int| {
|
|
if (modes.modeFromInt(mode_int, false)) |mode| {
|
|
try self.handler.vt(.save_mode, .{ .mode = mode });
|
|
} else {
|
|
log.warn(
|
|
"unimplemented save mode: {}",
|
|
.{mode_int},
|
|
);
|
|
}
|
|
}
|
|
},
|
|
|
|
// XTSHIFTESCAPE
|
|
'>' => if (@hasDecl(T, "setMouseShiftCapture")) capture: {
|
|
const capture = switch (input.params.len) {
|
|
0 => false,
|
|
1 => switch (input.params[0]) {
|
|
0 => false,
|
|
1 => true,
|
|
else => {
|
|
log.warn("invalid XTSHIFTESCAPE command: {f}", .{input});
|
|
break :capture;
|
|
},
|
|
},
|
|
else => {
|
|
log.warn("invalid XTSHIFTESCAPE command: {f}", .{input});
|
|
break :capture;
|
|
},
|
|
};
|
|
|
|
try self.handler.setMouseShiftCapture(capture);
|
|
} else log.warn(
|
|
"unimplemented CSI callback: {f}",
|
|
.{input},
|
|
),
|
|
|
|
else => log.warn(
|
|
"unknown CSI s with intermediate: {f}",
|
|
.{input},
|
|
),
|
|
},
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI s with intermediates: {f}",
|
|
.{input},
|
|
),
|
|
},
|
|
|
|
// XTWINOPS
|
|
't' => switch (input.intermediates.len) {
|
|
0 => {
|
|
if (input.params.len > 0) {
|
|
switch (input.params[0]) {
|
|
14 => if (input.params.len == 1) {
|
|
// report the text area size in pixels
|
|
if (@hasDecl(T, "sendSizeReport")) {
|
|
self.handler.sendSizeReport(.csi_14_t);
|
|
} else log.warn(
|
|
"ignoring unimplemented CSI 14 t",
|
|
.{},
|
|
);
|
|
} else log.warn(
|
|
"ignoring CSI 14 t with extra parameters: {f}",
|
|
.{input},
|
|
),
|
|
16 => if (input.params.len == 1) {
|
|
// report cell size in pixels
|
|
if (@hasDecl(T, "sendSizeReport")) {
|
|
self.handler.sendSizeReport(.csi_16_t);
|
|
} else log.warn(
|
|
"ignoring unimplemented CSI 16 t",
|
|
.{},
|
|
);
|
|
} else log.warn(
|
|
"ignoring CSI 16 t with extra parameters: {f}",
|
|
.{input},
|
|
),
|
|
18 => if (input.params.len == 1) {
|
|
// report screen size in characters
|
|
if (@hasDecl(T, "sendSizeReport")) {
|
|
self.handler.sendSizeReport(.csi_18_t);
|
|
} else log.warn(
|
|
"ignoring unimplemented CSI 18 t",
|
|
.{},
|
|
);
|
|
} else log.warn(
|
|
"ignoring CSI 18 t with extra parameters: {f}",
|
|
.{input},
|
|
),
|
|
21 => if (input.params.len == 1) {
|
|
// report window title
|
|
if (@hasDecl(T, "sendSizeReport")) {
|
|
self.handler.sendSizeReport(.csi_21_t);
|
|
} else log.warn(
|
|
"ignoring unimplemented CSI 21 t",
|
|
.{},
|
|
);
|
|
} else log.warn(
|
|
"ignoring CSI 21 t with extra parameters: {f}",
|
|
.{input},
|
|
),
|
|
inline 22, 23 => |number| if ((input.params.len == 2 or
|
|
input.params.len == 3) and
|
|
// we only support window title
|
|
(input.params[1] == 0 or
|
|
input.params[1] == 2))
|
|
{
|
|
// push/pop title
|
|
if (@hasDecl(T, "pushPopTitle")) {
|
|
self.handler.pushPopTitle(.{
|
|
.op = switch (number) {
|
|
22 => .push,
|
|
23 => .pop,
|
|
else => @compileError("unreachable"),
|
|
},
|
|
.index = if (input.params.len == 3)
|
|
input.params[2]
|
|
else
|
|
0,
|
|
});
|
|
} else log.warn(
|
|
"ignoring unimplemented CSI 22/23 t",
|
|
.{},
|
|
);
|
|
} else log.warn(
|
|
"ignoring CSI 22/23 t with extra parameters: {f}",
|
|
.{input},
|
|
),
|
|
else => log.warn(
|
|
"ignoring CSI t with unimplemented parameter: {f}",
|
|
.{input},
|
|
),
|
|
}
|
|
} else log.err(
|
|
"ignoring CSI t with no parameters: {f}",
|
|
.{input},
|
|
);
|
|
},
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI t with intermediates: {f}",
|
|
.{input},
|
|
),
|
|
},
|
|
|
|
'u' => switch (input.intermediates.len) {
|
|
0 => if (@hasDecl(T, "restoreCursor"))
|
|
try self.handler.restoreCursor()
|
|
else
|
|
log.warn("unimplemented CSI callback: {f}", .{input}),
|
|
|
|
// Kitty keyboard protocol
|
|
1 => switch (input.intermediates[0]) {
|
|
'?' => try self.handler.vt(.kitty_keyboard_query, {}),
|
|
|
|
'>' => push: {
|
|
const flags: u5 = if (input.params.len == 1)
|
|
std.math.cast(u5, input.params[0]) orelse {
|
|
log.warn("invalid pushKittyKeyboard command: {f}", .{input});
|
|
break :push;
|
|
}
|
|
else
|
|
0;
|
|
|
|
try self.handler.vt(.kitty_keyboard_push, .{ .flags = @as(kitty.KeyFlags, @bitCast(flags)) });
|
|
},
|
|
|
|
'<' => {
|
|
const number: u16 = if (input.params.len == 1)
|
|
input.params[0]
|
|
else
|
|
1;
|
|
|
|
try self.handler.vt(.kitty_keyboard_pop, number);
|
|
},
|
|
|
|
'=' => set: {
|
|
const flags: u5 = if (input.params.len >= 1)
|
|
std.math.cast(u5, input.params[0]) orelse {
|
|
log.warn("invalid setKittyKeyboard command: {f}", .{input});
|
|
break :set;
|
|
}
|
|
else
|
|
0;
|
|
|
|
const number: u16 = if (input.params.len >= 2)
|
|
input.params[1]
|
|
else
|
|
1;
|
|
|
|
const action_tag: streampkg.Action.Tag = switch (number) {
|
|
1 => .kitty_keyboard_set,
|
|
2 => .kitty_keyboard_set_or,
|
|
3 => .kitty_keyboard_set_not,
|
|
else => {
|
|
log.warn("invalid setKittyKeyboard command: {f}", .{input});
|
|
break :set;
|
|
},
|
|
};
|
|
|
|
const kitty_flags: streampkg.Action.KittyKeyboardFlags = .{ .flags = @as(kitty.KeyFlags, @bitCast(flags)) };
|
|
switch (action_tag) {
|
|
.kitty_keyboard_set => try self.handler.vt(.kitty_keyboard_set, kitty_flags),
|
|
.kitty_keyboard_set_or => try self.handler.vt(.kitty_keyboard_set_or, kitty_flags),
|
|
.kitty_keyboard_set_not => try self.handler.vt(.kitty_keyboard_set_not, kitty_flags),
|
|
else => unreachable,
|
|
}
|
|
},
|
|
|
|
else => log.warn(
|
|
"unknown CSI s with intermediate: {f}",
|
|
.{input},
|
|
),
|
|
},
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI u: {f}",
|
|
.{input},
|
|
),
|
|
},
|
|
|
|
// ICH - Insert Blanks
|
|
'@' => switch (input.intermediates.len) {
|
|
0 => try self.handler.vt(.insert_blanks, switch (input.params.len) {
|
|
0 => 1,
|
|
1 => input.params[0],
|
|
else => {
|
|
log.warn("invalid ICH command: {f}", .{input});
|
|
return;
|
|
},
|
|
}),
|
|
|
|
else => log.warn(
|
|
"ignoring unimplemented CSI @: {f}",
|
|
.{input},
|
|
),
|
|
},
|
|
|
|
// DECSASD - Select Active Status Display
|
|
'}' => {
|
|
const success = decsasd: {
|
|
// Verify we're getting a DECSASD command
|
|
if (input.intermediates.len != 1 or input.intermediates[0] != '$')
|
|
break :decsasd false;
|
|
if (input.params.len != 1)
|
|
break :decsasd false;
|
|
if (!@hasDecl(T, "setActiveStatusDisplay"))
|
|
break :decsasd false;
|
|
|
|
const display = std.meta.intToEnum(
|
|
ansi.StatusDisplay,
|
|
input.params[0],
|
|
) catch break :decsasd false;
|
|
|
|
try self.handler.setActiveStatusDisplay(display);
|
|
break :decsasd true;
|
|
};
|
|
|
|
if (!success) log.warn("unimplemented CSI callback: {f}", .{input});
|
|
},
|
|
|
|
else => if (@hasDecl(T, "csiUnimplemented"))
|
|
try self.handler.csiUnimplemented(input)
|
|
else
|
|
log.warn("unimplemented CSI action: {f}", .{input}),
|
|
}
|
|
}
|
|
|
|
inline fn oscDispatch(self: *Self, cmd: osc.Command) !void {
|
|
switch (cmd) {
|
|
.change_window_title => |title| {
|
|
if (@hasDecl(T, "changeWindowTitle")) {
|
|
if (!std.unicode.utf8ValidateSlice(title)) {
|
|
log.warn("change title request: invalid utf-8, ignoring request", .{});
|
|
return;
|
|
}
|
|
|
|
try self.handler.changeWindowTitle(title);
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.change_window_icon => |icon| {
|
|
log.info("OSC 1 (change icon) received and ignored icon={s}", .{icon});
|
|
},
|
|
|
|
.clipboard_contents => |clip| {
|
|
if (@hasDecl(T, "clipboardContents")) {
|
|
try self.handler.clipboardContents(clip.kind, clip.data);
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.prompt_start => |v| {
|
|
if (@hasDecl(T, "promptStart")) {
|
|
switch (v.kind) {
|
|
.primary, .right => try self.handler.promptStart(v.aid, v.redraw),
|
|
.continuation, .secondary => try self.handler.promptContinuation(v.aid),
|
|
}
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.prompt_end => {
|
|
try self.handler.vt(.prompt_end, {});
|
|
},
|
|
|
|
.end_of_input => {
|
|
try self.handler.vt(.end_of_input, {});
|
|
},
|
|
|
|
.end_of_command => |end| {
|
|
if (@hasDecl(T, "endOfCommand")) {
|
|
try self.handler.endOfCommand(end.exit_code);
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.report_pwd => |v| {
|
|
if (@hasDecl(T, "reportPwd")) {
|
|
try self.handler.reportPwd(v.value);
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.mouse_shape => |v| {
|
|
if (@hasDecl(T, "setMouseShape")) {
|
|
const shape = MouseShape.fromString(v.value) orelse {
|
|
log.warn("unknown cursor shape: {s}", .{v.value});
|
|
return;
|
|
};
|
|
|
|
try self.handler.setMouseShape(shape);
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.color_operation => |v| {
|
|
if (@hasDecl(T, "handleColorOperation")) {
|
|
try self.handler.handleColorOperation(
|
|
v.op,
|
|
&v.requests,
|
|
v.terminator,
|
|
);
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.kitty_color_protocol => |v| {
|
|
if (@hasDecl(T, "sendKittyColorReport")) {
|
|
try self.handler.sendKittyColorReport(v);
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.show_desktop_notification => |v| {
|
|
if (@hasDecl(T, "showDesktopNotification")) {
|
|
try self.handler.showDesktopNotification(v.title, v.body);
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.hyperlink_start => |v| {
|
|
if (@hasDecl(T, "startHyperlink")) {
|
|
try self.handler.startHyperlink(v.uri, v.id);
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.hyperlink_end => {
|
|
try self.handler.vt(.end_hyperlink, {});
|
|
},
|
|
|
|
.conemu_progress_report => |v| {
|
|
if (@hasDecl(T, "handleProgressReport")) {
|
|
try self.handler.handleProgressReport(v);
|
|
return;
|
|
} else log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.conemu_sleep,
|
|
.conemu_show_message_box,
|
|
.conemu_change_tab_title,
|
|
.conemu_wait_input,
|
|
.conemu_guimacro,
|
|
=> {
|
|
log.warn("unimplemented OSC callback: {}", .{cmd});
|
|
},
|
|
|
|
.invalid => {
|
|
// This is an invalid internal state, not an invalid OSC
|
|
// string being parsed. We shouldn't see this.
|
|
log.warn("invalid OSC, should never happen", .{});
|
|
},
|
|
}
|
|
|
|
// Fall through for when we don't have a handler.
|
|
if (@hasDecl(T, "oscUnimplemented")) {
|
|
try self.handler.oscUnimplemented(cmd);
|
|
} else {
|
|
log.warn("unimplemented OSC command: {s}", .{@tagName(cmd)});
|
|
}
|
|
}
|
|
|
|
inline fn configureCharset(
|
|
self: *Self,
|
|
intermediates: []const u8,
|
|
set: charsets.Charset,
|
|
) !void {
|
|
if (intermediates.len != 1) {
|
|
log.warn("invalid charset intermediate: {any}", .{intermediates});
|
|
return;
|
|
}
|
|
|
|
const slot: charsets.Slots = switch (intermediates[0]) {
|
|
// TODO: support slots '-', '.', '/'
|
|
|
|
'(' => .G0,
|
|
')' => .G1,
|
|
'*' => .G2,
|
|
'+' => .G3,
|
|
else => {
|
|
log.warn("invalid charset intermediate: {any}", .{intermediates});
|
|
return;
|
|
},
|
|
};
|
|
|
|
if (@hasDecl(T, "configureCharset")) {
|
|
try self.handler.configureCharset(slot, set);
|
|
return;
|
|
}
|
|
|
|
log.warn("unimplemented configureCharset callback slot={} set={}", .{
|
|
slot,
|
|
set,
|
|
});
|
|
}
|
|
|
|
inline fn escDispatch(
|
|
self: *Self,
|
|
action: Parser.Action.ESC,
|
|
) !void {
|
|
switch (action.final) {
|
|
// Charsets
|
|
'B' => try self.configureCharset(action.intermediates, .ascii),
|
|
'A' => try self.configureCharset(action.intermediates, .british),
|
|
'0' => try self.configureCharset(action.intermediates, .dec_special),
|
|
|
|
// DECSC - Save Cursor
|
|
'7' => switch (action.intermediates.len) {
|
|
0 => try self.handler.vt(.save_cursor, {}),
|
|
else => {
|
|
log.warn("invalid command: {f}", .{action});
|
|
return;
|
|
},
|
|
},
|
|
|
|
'8' => blk: {
|
|
switch (action.intermediates.len) {
|
|
// DECRC - Restore Cursor
|
|
0 => {
|
|
try self.handler.vt(.restore_cursor, {});
|
|
break :blk {};
|
|
},
|
|
|
|
1 => switch (action.intermediates[0]) {
|
|
// DECALN - Fill Screen with E
|
|
'#' => {
|
|
try self.handler.vt(.decaln, {});
|
|
break :blk {};
|
|
},
|
|
|
|
else => {},
|
|
},
|
|
|
|
else => {}, // fall through
|
|
}
|
|
|
|
log.warn("unimplemented ESC action: {f}", .{action});
|
|
},
|
|
|
|
// IND - Index
|
|
'D' => if (@hasDecl(T, "index")) switch (action.intermediates.len) {
|
|
0 => try self.handler.index(),
|
|
else => {
|
|
log.warn("invalid index command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented ESC callback: {f}", .{action}),
|
|
|
|
// NEL - Next Line
|
|
'E' => if (@hasDecl(T, "nextLine")) switch (action.intermediates.len) {
|
|
0 => try self.handler.nextLine(),
|
|
else => {
|
|
log.warn("invalid next line command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented ESC callback: {f}", .{action}),
|
|
|
|
// HTS - Horizontal Tab Set
|
|
'H' => switch (action.intermediates.len) {
|
|
0 => try self.handler.vt(.tab_set, {}),
|
|
else => {
|
|
log.warn("invalid tab set command: {f}", .{action});
|
|
return;
|
|
},
|
|
},
|
|
|
|
// RI - Reverse Index
|
|
'M' => if (@hasDecl(T, "reverseIndex")) switch (action.intermediates.len) {
|
|
0 => try self.handler.reverseIndex(),
|
|
else => {
|
|
log.warn("invalid reverse index command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented ESC callback: {f}", .{action}),
|
|
|
|
// SS2 - Single Shift 2
|
|
'N' => if (@hasDecl(T, "invokeCharset")) switch (action.intermediates.len) {
|
|
0 => try self.handler.invokeCharset(.GL, .G2, true),
|
|
else => {
|
|
log.warn("invalid single shift 2 command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented invokeCharset: {f}", .{action}),
|
|
|
|
// SS3 - Single Shift 3
|
|
'O' => if (@hasDecl(T, "invokeCharset")) switch (action.intermediates.len) {
|
|
0 => try self.handler.invokeCharset(.GL, .G3, true),
|
|
else => {
|
|
log.warn("invalid single shift 3 command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented invokeCharset: {f}", .{action}),
|
|
|
|
// SPA - Start of Guarded Area
|
|
'V' => switch (action.intermediates.len) {
|
|
0 => try self.handler.vt(.protected_mode_iso, {}),
|
|
else => log.warn("unimplemented ESC callback: {f}", .{action}),
|
|
},
|
|
|
|
// EPA - End of Guarded Area
|
|
'W' => switch (action.intermediates.len) {
|
|
0 => try self.handler.vt(.protected_mode_off, {}),
|
|
else => log.warn("unimplemented ESC callback: {f}", .{action}),
|
|
},
|
|
|
|
// DECID
|
|
'Z' => if (@hasDecl(T, "deviceAttributes") and action.intermediates.len == 0) {
|
|
try self.handler.deviceAttributes(.primary, &.{});
|
|
} else log.warn("unimplemented ESC callback: {f}", .{action}),
|
|
|
|
// RIS - Full Reset
|
|
'c' => if (@hasDecl(T, "fullReset")) switch (action.intermediates.len) {
|
|
0 => try self.handler.fullReset(),
|
|
else => {
|
|
log.warn("invalid full reset command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented ESC callback: {f}", .{action}),
|
|
|
|
// LS2 - Locking Shift 2
|
|
'n' => if (@hasDecl(T, "invokeCharset")) switch (action.intermediates.len) {
|
|
0 => try self.handler.invokeCharset(.GL, .G2, false),
|
|
else => {
|
|
log.warn("invalid single shift 2 command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented invokeCharset: {f}", .{action}),
|
|
|
|
// LS3 - Locking Shift 3
|
|
'o' => if (@hasDecl(T, "invokeCharset")) switch (action.intermediates.len) {
|
|
0 => try self.handler.invokeCharset(.GL, .G3, false),
|
|
else => {
|
|
log.warn("invalid single shift 3 command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented invokeCharset: {f}", .{action}),
|
|
|
|
// LS1R - Locking Shift 1 Right
|
|
'~' => if (@hasDecl(T, "invokeCharset")) switch (action.intermediates.len) {
|
|
0 => try self.handler.invokeCharset(.GR, .G1, false),
|
|
else => {
|
|
log.warn("invalid locking shift 1 right command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented invokeCharset: {f}", .{action}),
|
|
|
|
// LS2R - Locking Shift 2 Right
|
|
'}' => if (@hasDecl(T, "invokeCharset")) switch (action.intermediates.len) {
|
|
0 => try self.handler.invokeCharset(.GR, .G2, false),
|
|
else => {
|
|
log.warn("invalid locking shift 2 right command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented invokeCharset: {f}", .{action}),
|
|
|
|
// LS3R - Locking Shift 3 Right
|
|
'|' => if (@hasDecl(T, "invokeCharset")) switch (action.intermediates.len) {
|
|
0 => try self.handler.invokeCharset(.GR, .G3, false),
|
|
else => {
|
|
log.warn("invalid locking shift 3 right command: {f}", .{action});
|
|
return;
|
|
},
|
|
} else log.warn("unimplemented invokeCharset: {f}", .{action}),
|
|
|
|
// Set application keypad mode
|
|
'=' => if (@hasDecl(T, "setMode") and action.intermediates.len == 0) {
|
|
try self.handler.setMode(.keypad_keys, true);
|
|
} else log.warn("unimplemented setMode: {f}", .{action}),
|
|
|
|
// Reset application keypad mode
|
|
'>' => if (@hasDecl(T, "setMode") and action.intermediates.len == 0) {
|
|
try self.handler.setMode(.keypad_keys, false);
|
|
} else log.warn("unimplemented setMode: {f}", .{action}),
|
|
|
|
else => if (@hasDecl(T, "escUnimplemented"))
|
|
try self.handler.escUnimplemented(action)
|
|
else
|
|
log.warn("unimplemented ESC action: {f}", .{action}),
|
|
|
|
// Sets ST (string terminator). We don't have to do anything
|
|
// because our parser always accepts ST.
|
|
'\\' => {},
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
test Action {
|
|
// Forces the C type to be reified when the target is C, ensuring
|
|
// all our types are C ABI compatible.
|
|
_ = Action.C;
|
|
}
|
|
|
|
test "stream: print" {
|
|
const H = struct {
|
|
c: ?u21 = 0,
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
switch (action) {
|
|
.print => self.c = value.cp,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.next('x');
|
|
try testing.expectEqual(@as(u21, 'x'), s.handler.c.?);
|
|
}
|
|
|
|
test "simd: print invalid utf-8" {
|
|
const H = struct {
|
|
c: ?u21 = 0,
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
switch (action) {
|
|
.print => self.c = value.cp,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice(&.{0xFF});
|
|
try testing.expectEqual(@as(u21, 0xFFFD), s.handler.c.?);
|
|
}
|
|
|
|
test "simd: complete incomplete utf-8" {
|
|
const H = struct {
|
|
c: ?u21 = null,
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
switch (action) {
|
|
.print => self.c = value.cp,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice(&.{0xE0}); // 3 byte
|
|
try testing.expect(s.handler.c == null);
|
|
try s.nextSlice(&.{0xA0}); // still incomplete
|
|
try testing.expect(s.handler.c == null);
|
|
try s.nextSlice(&.{0x80});
|
|
try testing.expectEqual(@as(u21, 0x800), s.handler.c.?);
|
|
}
|
|
|
|
test "stream: cursor right (CUF)" {
|
|
const H = struct {
|
|
amount: u16 = 0,
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
switch (action) {
|
|
.cursor_right => self.amount = value.value,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1B[C");
|
|
try testing.expectEqual(@as(u16, 1), s.handler.amount);
|
|
|
|
try s.nextSlice("\x1B[5C");
|
|
try testing.expectEqual(@as(u16, 5), s.handler.amount);
|
|
|
|
s.handler.amount = 0;
|
|
try s.nextSlice("\x1B[5;4C");
|
|
try testing.expectEqual(@as(u16, 0), s.handler.amount);
|
|
|
|
s.handler.amount = 0;
|
|
try s.nextSlice("\x1b[?3C");
|
|
try testing.expectEqual(@as(u16, 0), s.handler.amount);
|
|
}
|
|
|
|
test "stream: dec set mode (SM) and reset mode (RM)" {
|
|
const H = struct {
|
|
mode: modes.Mode = @as(modes.Mode, @enumFromInt(1)),
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
switch (action) {
|
|
.set_mode => self.mode = value.mode,
|
|
.reset_mode => self.mode = @as(modes.Mode, @enumFromInt(1)),
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1B[?6h");
|
|
try testing.expectEqual(@as(modes.Mode, .origin), s.handler.mode);
|
|
|
|
try s.nextSlice("\x1B[?6l");
|
|
try testing.expectEqual(@as(modes.Mode, @enumFromInt(1)), s.handler.mode);
|
|
|
|
s.handler.mode = @as(modes.Mode, @enumFromInt(1));
|
|
try s.nextSlice("\x1B[6 h");
|
|
try testing.expectEqual(@as(modes.Mode, @enumFromInt(1)), s.handler.mode);
|
|
}
|
|
|
|
test "stream: ansi set mode (SM) and reset mode (RM)" {
|
|
const H = struct {
|
|
mode: ?modes.Mode = null,
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
switch (action) {
|
|
.set_mode => self.mode = value.mode,
|
|
.reset_mode => self.mode = null,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1B[4h");
|
|
try testing.expectEqual(@as(modes.Mode, .insert), s.handler.mode.?);
|
|
|
|
try s.nextSlice("\x1B[4l");
|
|
try testing.expect(s.handler.mode == null);
|
|
|
|
s.handler.mode = null;
|
|
try s.nextSlice("\x1B[>5h");
|
|
try testing.expect(s.handler.mode == null);
|
|
}
|
|
|
|
test "stream: ansi set mode (SM) and reset mode (RM) with unknown value" {
|
|
const H = struct {
|
|
mode: ?modes.Mode = null,
|
|
|
|
pub fn setMode(self: *@This(), mode: modes.Mode, v: bool) !void {
|
|
self.mode = null;
|
|
if (v) self.mode = mode;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1B[6h");
|
|
try testing.expect(s.handler.mode == null);
|
|
|
|
try s.nextSlice("\x1B[6l");
|
|
try testing.expect(s.handler.mode == null);
|
|
}
|
|
|
|
test "stream: restore mode" {
|
|
const H = struct {
|
|
const Self = @This();
|
|
called: bool = false,
|
|
|
|
pub fn vt(
|
|
self: *Self,
|
|
comptime action: Stream(Self).Action.Tag,
|
|
value: Stream(Self).Action.Value(action),
|
|
) !void {
|
|
_ = value;
|
|
switch (action) {
|
|
.top_and_bottom_margin => self.called = true,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
for ("\x1B[?42r") |c| try s.next(c);
|
|
try testing.expect(!s.handler.called);
|
|
}
|
|
|
|
test "stream: pop kitty keyboard with no params defaults to 1" {
|
|
const H = struct {
|
|
const Self = @This();
|
|
n: u16 = 0,
|
|
|
|
pub fn vt(
|
|
self: *Self,
|
|
comptime action: streampkg.Action.Tag,
|
|
value: streampkg.Action.Value(action),
|
|
) !void {
|
|
switch (action) {
|
|
.kitty_keyboard_pop => self.n = value,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
for ("\x1B[<u") |c| try s.next(c);
|
|
try testing.expectEqual(@as(u16, 1), s.handler.n);
|
|
}
|
|
|
|
test "stream: DECSCA" {
|
|
const H = struct {
|
|
const Self = @This();
|
|
v: ?ansi.ProtectedMode = null,
|
|
|
|
pub fn vt(
|
|
self: *Self,
|
|
comptime action: Stream(Self).Action.Tag,
|
|
value: Stream(Self).Action.Value(action),
|
|
) !void {
|
|
_ = value;
|
|
switch (action) {
|
|
.protected_mode_off => self.v = .off,
|
|
.protected_mode_iso => self.v = .iso,
|
|
.protected_mode_dec => self.v = .dec,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
{
|
|
for ("\x1B[\"q") |c| try s.next(c);
|
|
try testing.expectEqual(ansi.ProtectedMode.off, s.handler.v.?);
|
|
}
|
|
{
|
|
for ("\x1B[0\"q") |c| try s.next(c);
|
|
try testing.expectEqual(ansi.ProtectedMode.off, s.handler.v.?);
|
|
}
|
|
{
|
|
for ("\x1B[2\"q") |c| try s.next(c);
|
|
try testing.expectEqual(ansi.ProtectedMode.off, s.handler.v.?);
|
|
}
|
|
{
|
|
for ("\x1B[1\"q") |c| try s.next(c);
|
|
try testing.expectEqual(ansi.ProtectedMode.dec, s.handler.v.?);
|
|
}
|
|
}
|
|
|
|
test "stream: DECED, DECSED" {
|
|
const H = struct {
|
|
const Self = @This();
|
|
mode: ?csi.EraseDisplay = null,
|
|
protected: ?bool = null,
|
|
|
|
pub fn vt(
|
|
self: *Self,
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
switch (action) {
|
|
.erase_display_below => {
|
|
self.mode = .below;
|
|
self.protected = value;
|
|
},
|
|
.erase_display_above => {
|
|
self.mode = .above;
|
|
self.protected = value;
|
|
},
|
|
.erase_display_complete => {
|
|
self.mode = .complete;
|
|
self.protected = value;
|
|
},
|
|
.erase_display_scrollback => {
|
|
self.mode = .scrollback;
|
|
self.protected = value;
|
|
},
|
|
.erase_display_scroll_complete => {
|
|
self.mode = .scroll_complete;
|
|
self.protected = value;
|
|
},
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
{
|
|
for ("\x1B[?J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.below, s.handler.mode.?);
|
|
try testing.expect(s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[?0J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.below, s.handler.mode.?);
|
|
try testing.expect(s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[?1J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.above, s.handler.mode.?);
|
|
try testing.expect(s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[?2J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.complete, s.handler.mode.?);
|
|
try testing.expect(s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[?3J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.scrollback, s.handler.mode.?);
|
|
try testing.expect(s.handler.protected.?);
|
|
}
|
|
|
|
{
|
|
for ("\x1B[J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.below, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[0J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.below, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[1J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.above, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[2J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.complete, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[3J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.scrollback, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
{
|
|
// Invalid and ignored by the handler
|
|
for ("\x1B[>0J") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseDisplay.scrollback, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
}
|
|
|
|
test "stream: DECEL, DECSEL" {
|
|
const H = struct {
|
|
const Self = @This();
|
|
mode: ?csi.EraseLine = null,
|
|
protected: ?bool = null,
|
|
|
|
pub fn vt(
|
|
self: *Self,
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
switch (action) {
|
|
.erase_line_right => {
|
|
self.mode = .right;
|
|
self.protected = value;
|
|
},
|
|
.erase_line_left => {
|
|
self.mode = .left;
|
|
self.protected = value;
|
|
},
|
|
.erase_line_complete => {
|
|
self.mode = .complete;
|
|
self.protected = value;
|
|
},
|
|
.erase_line_right_unless_pending_wrap => {
|
|
self.mode = .right_unless_pending_wrap;
|
|
self.protected = value;
|
|
},
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
{
|
|
for ("\x1B[?K") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseLine.right, s.handler.mode.?);
|
|
try testing.expect(s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[?0K") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseLine.right, s.handler.mode.?);
|
|
try testing.expect(s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[?1K") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseLine.left, s.handler.mode.?);
|
|
try testing.expect(s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[?2K") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseLine.complete, s.handler.mode.?);
|
|
try testing.expect(s.handler.protected.?);
|
|
}
|
|
|
|
{
|
|
for ("\x1B[K") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseLine.right, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[0K") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseLine.right, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[1K") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseLine.left, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
{
|
|
for ("\x1B[2K") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseLine.complete, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
{
|
|
// Invalid and ignored by the handler
|
|
for ("\x1B[<1K") |c| try s.next(c);
|
|
try testing.expectEqual(csi.EraseLine.complete, s.handler.mode.?);
|
|
try testing.expect(!s.handler.protected.?);
|
|
}
|
|
}
|
|
|
|
test "stream: DECSCUSR" {
|
|
const H = struct {
|
|
style: ?ansi.CursorStyle = null,
|
|
|
|
pub fn setCursorStyle(self: *@This(), style: ansi.CursorStyle) !void {
|
|
self.style = style;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1B[ q");
|
|
try testing.expect(s.handler.style.? == .default);
|
|
|
|
try s.nextSlice("\x1B[1 q");
|
|
try testing.expect(s.handler.style.? == .blinking_block);
|
|
|
|
// Invalid and ignored by the handler
|
|
try s.nextSlice("\x1B[?0 q");
|
|
try testing.expect(s.handler.style.? == .blinking_block);
|
|
}
|
|
|
|
test "stream: DECSCUSR without space" {
|
|
const H = struct {
|
|
style: ?ansi.CursorStyle = null,
|
|
|
|
pub fn setCursorStyle(self: *@This(), style: ansi.CursorStyle) !void {
|
|
self.style = style;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1B[q");
|
|
try testing.expect(s.handler.style == null);
|
|
|
|
try s.nextSlice("\x1B[1q");
|
|
try testing.expect(s.handler.style == null);
|
|
}
|
|
|
|
test "stream: XTSHIFTESCAPE" {
|
|
const H = struct {
|
|
escape: ?bool = null,
|
|
|
|
pub fn setMouseShiftCapture(self: *@This(), v: bool) !void {
|
|
self.escape = v;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1B[>2s");
|
|
try testing.expect(s.handler.escape == null);
|
|
|
|
try s.nextSlice("\x1B[>s");
|
|
try testing.expect(s.handler.escape.? == false);
|
|
|
|
try s.nextSlice("\x1B[>0s");
|
|
try testing.expect(s.handler.escape.? == false);
|
|
|
|
try s.nextSlice("\x1B[>1s");
|
|
try testing.expect(s.handler.escape.? == true);
|
|
|
|
// Invalid and ignored by the handler
|
|
try s.nextSlice("\x1B[1 s");
|
|
try testing.expect(s.handler.escape.? == true);
|
|
}
|
|
|
|
test "stream: change window title with invalid utf-8" {
|
|
const H = struct {
|
|
seen: bool = false,
|
|
|
|
pub fn changeWindowTitle(self: *@This(), title: []const u8) !void {
|
|
_ = title;
|
|
|
|
self.seen = true;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
{
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1b]2;abc\x1b\\");
|
|
try testing.expect(s.handler.seen);
|
|
}
|
|
|
|
{
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1b]2;abc\xc0\x1b\\");
|
|
try testing.expect(!s.handler.seen);
|
|
}
|
|
}
|
|
|
|
test "stream: insert characters" {
|
|
const H = struct {
|
|
const Self = @This();
|
|
called: bool = false,
|
|
|
|
pub fn vt(
|
|
self: *Self,
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = value;
|
|
switch (action) {
|
|
.insert_blanks => self.called = true,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
for ("\x1B[42@") |c| try s.next(c);
|
|
try testing.expect(s.handler.called);
|
|
|
|
s.handler.called = false;
|
|
for ("\x1B[?42@") |c| try s.next(c);
|
|
try testing.expect(!s.handler.called);
|
|
}
|
|
|
|
test "stream: SCOSC" {
|
|
const H = struct {
|
|
const Self = @This();
|
|
called: bool = false,
|
|
|
|
pub fn vt(
|
|
self: *Self,
|
|
comptime action: Stream(Self).Action.Tag,
|
|
value: Stream(Self).Action.Value(action),
|
|
) !void {
|
|
_ = value;
|
|
switch (action) {
|
|
.left_and_right_margin => @panic("bad"),
|
|
.left_and_right_margin_ambiguous => self.called = true,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
for ("\x1B[s") |c| try s.next(c);
|
|
try testing.expect(s.handler.called);
|
|
}
|
|
|
|
test "stream: SCORC" {
|
|
const H = struct {
|
|
const Self = @This();
|
|
called: bool = false,
|
|
|
|
pub fn restoreCursor(self: *Self) !void {
|
|
self.called = true;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
for ("\x1B[u") |c| try s.next(c);
|
|
try testing.expect(s.handler.called);
|
|
}
|
|
|
|
test "stream: too many csi params" {
|
|
const H = struct {
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = value;
|
|
switch (action) {
|
|
.cursor_right => unreachable,
|
|
else => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1B[1;1;1;1;1;1;1;1;1;1;1;1;1;1;1;1;1C");
|
|
}
|
|
|
|
test "stream: csi param too long" {
|
|
const H = struct {
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
try s.nextSlice("\x1B[1111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111C");
|
|
}
|
|
|
|
test "stream: send report with CSI t" {
|
|
const H = struct {
|
|
style: ?csi.SizeReportStyle = null,
|
|
|
|
pub fn sendSizeReport(self: *@This(), style: csi.SizeReportStyle) void {
|
|
self.style = style;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[14t");
|
|
try testing.expectEqual(csi.SizeReportStyle.csi_14_t, s.handler.style);
|
|
|
|
try s.nextSlice("\x1b[16t");
|
|
try testing.expectEqual(csi.SizeReportStyle.csi_16_t, s.handler.style);
|
|
|
|
try s.nextSlice("\x1b[18t");
|
|
try testing.expectEqual(csi.SizeReportStyle.csi_18_t, s.handler.style);
|
|
|
|
try s.nextSlice("\x1b[21t");
|
|
try testing.expectEqual(csi.SizeReportStyle.csi_21_t, s.handler.style);
|
|
}
|
|
|
|
test "stream: invalid CSI t" {
|
|
const H = struct {
|
|
style: ?csi.SizeReportStyle = null,
|
|
|
|
pub fn sendSizeReport(self: *@This(), style: csi.SizeReportStyle) void {
|
|
self.style = style;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[19t");
|
|
try testing.expectEqual(null, s.handler.style);
|
|
}
|
|
|
|
test "stream: CSI t push title" {
|
|
const H = struct {
|
|
op: ?csi.TitlePushPop = null,
|
|
|
|
pub fn pushPopTitle(self: *@This(), op: csi.TitlePushPop) void {
|
|
self.op = op;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[22;0t");
|
|
try testing.expectEqual(csi.TitlePushPop{
|
|
.op = .push,
|
|
.index = 0,
|
|
}, s.handler.op.?);
|
|
}
|
|
|
|
test "stream: CSI t push title with explicit window" {
|
|
const H = struct {
|
|
op: ?csi.TitlePushPop = null,
|
|
|
|
pub fn pushPopTitle(self: *@This(), op: csi.TitlePushPop) void {
|
|
self.op = op;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[22;2t");
|
|
try testing.expectEqual(csi.TitlePushPop{
|
|
.op = .push,
|
|
.index = 0,
|
|
}, s.handler.op.?);
|
|
}
|
|
|
|
test "stream: CSI t push title with explicit icon" {
|
|
const H = struct {
|
|
op: ?csi.TitlePushPop = null,
|
|
|
|
pub fn pushPopTitle(self: *@This(), op: csi.TitlePushPop) void {
|
|
self.op = op;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[22;1t");
|
|
try testing.expectEqual(null, s.handler.op);
|
|
}
|
|
|
|
test "stream: CSI t push title with index" {
|
|
const H = struct {
|
|
op: ?csi.TitlePushPop = null,
|
|
|
|
pub fn pushPopTitle(self: *@This(), op: csi.TitlePushPop) void {
|
|
self.op = op;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[22;0;5t");
|
|
try testing.expectEqual(csi.TitlePushPop{
|
|
.op = .push,
|
|
.index = 5,
|
|
}, s.handler.op.?);
|
|
}
|
|
|
|
test "stream: CSI t pop title" {
|
|
const H = struct {
|
|
op: ?csi.TitlePushPop = null,
|
|
|
|
pub fn pushPopTitle(self: *@This(), op: csi.TitlePushPop) void {
|
|
self.op = op;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[23;0t");
|
|
try testing.expectEqual(csi.TitlePushPop{
|
|
.op = .pop,
|
|
.index = 0,
|
|
}, s.handler.op.?);
|
|
}
|
|
|
|
test "stream: CSI t pop title with explicit window" {
|
|
const H = struct {
|
|
op: ?csi.TitlePushPop = null,
|
|
|
|
pub fn pushPopTitle(self: *@This(), op: csi.TitlePushPop) void {
|
|
self.op = op;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[23;2t");
|
|
try testing.expectEqual(csi.TitlePushPop{
|
|
.op = .pop,
|
|
.index = 0,
|
|
}, s.handler.op.?);
|
|
}
|
|
|
|
test "stream: CSI t pop title with explicit icon" {
|
|
const H = struct {
|
|
op: ?csi.TitlePushPop = null,
|
|
|
|
pub fn pushPopTitle(self: *@This(), op: csi.TitlePushPop) void {
|
|
self.op = op;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[23;1t");
|
|
try testing.expectEqual(null, s.handler.op);
|
|
}
|
|
|
|
test "stream: CSI t pop title with index" {
|
|
const H = struct {
|
|
op: ?csi.TitlePushPop = null,
|
|
|
|
pub fn pushPopTitle(self: *@This(), op: csi.TitlePushPop) void {
|
|
self.op = op;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[23;0;5t");
|
|
try testing.expectEqual(csi.TitlePushPop{
|
|
.op = .pop,
|
|
.index = 5,
|
|
}, s.handler.op.?);
|
|
}
|
|
|
|
test "stream CSI W clear tab stops" {
|
|
const H = struct {
|
|
action: ?Action.Key = null,
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = value;
|
|
self.action = action;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[2W");
|
|
try testing.expectEqual(Action.Key.tab_clear_current, s.handler.action.?);
|
|
|
|
try s.nextSlice("\x1b[5W");
|
|
try testing.expectEqual(Action.Key.tab_clear_all, s.handler.action.?);
|
|
}
|
|
|
|
test "stream CSI W tab set" {
|
|
const H = struct {
|
|
action: ?Action.Key = null,
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = value;
|
|
self.action = action;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[W");
|
|
try testing.expectEqual(Action.Key.tab_set, s.handler.action.?);
|
|
|
|
s.handler.action = null;
|
|
try s.nextSlice("\x1b[0W");
|
|
try testing.expectEqual(Action.Key.tab_set, s.handler.action.?);
|
|
|
|
s.handler.action = null;
|
|
try s.nextSlice("\x1b[>W");
|
|
try testing.expect(s.handler.action == null);
|
|
|
|
s.handler.action = null;
|
|
try s.nextSlice("\x1b[99W");
|
|
try testing.expect(s.handler.action == null);
|
|
}
|
|
|
|
test "stream CSI ? W reset tab stops" {
|
|
const H = struct {
|
|
action: ?Action.Key = null,
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = value;
|
|
self.action = action;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
try s.nextSlice("\x1b[?2W");
|
|
try testing.expect(s.handler.action == null);
|
|
|
|
try s.nextSlice("\x1b[?5W");
|
|
try testing.expectEqual(Action.Key.tab_reset, s.handler.action.?);
|
|
|
|
// Invalid and ignored by the handler
|
|
s.handler.action = null;
|
|
try s.nextSlice("\x1b[?1;2;3W");
|
|
try testing.expect(s.handler.action == null);
|
|
}
|
|
|
|
test "stream: SGR with 17+ parameters for underline color" {
|
|
const H = struct {
|
|
attrs: ?sgr.Attribute = null,
|
|
called: bool = false,
|
|
|
|
pub fn setAttribute(self: *@This(), attr: sgr.Attribute) !void {
|
|
self.attrs = attr;
|
|
self.called = true;
|
|
}
|
|
|
|
pub fn vt(
|
|
self: *@This(),
|
|
comptime action: anytype,
|
|
value: anytype,
|
|
) !void {
|
|
_ = self;
|
|
_ = action;
|
|
_ = value;
|
|
}
|
|
};
|
|
|
|
var s: Stream(H) = .init(.{});
|
|
|
|
// Kakoune-style SGR with underline color as 17th parameter
|
|
// This tests the fix where param 17 was being dropped
|
|
try s.nextSlice("\x1b[4:3;38;2;51;51;51;48;2;170;170;170;58;2;255;97;136;0m");
|
|
try testing.expect(s.handler.called);
|
|
}
|