use crate::{
common::{is_windows_subsystem_for_linux, shell_modes, WSL},
env::{EnvStack, Environment as _},
fd_readable_set::{FdReadableSet, Timeout},
flog::{flog, FloggableDebug, FloggableDisplay},
key::{
self, alt, canonicalize_control_char, canonicalize_keyed_control_char, char_to_symbol,
function_key, shift, Key, Modifiers, ViewportPosition,
},
prelude::*,
reader::reader_test_and_clear_interrupted,
tty_handoff::{
maybe_set_kitty_keyboard_capability, maybe_set_scroll_content_up_capability,
SCROLL_CONTENT_UP_TERMINFO_CODE, TERMINAL_OS_NAME, XTGETTCAP_QUERY_OS_NAME, XTVERSION,
},
universal_notifier::default_notifier,
wutil::{fish_is_pua, fish_wcstol},
};
use fish_common::read_blocked;
use fish_feature_flags::{feature_test, FeatureFlag};
use fish_widestring::{bytes2wcstring, encode_byte_to_char, fish_reserved_codepoint};
use nix::sys::{select::FdSet, signal::SigSet, time::TimeSpec};
use std::{
cell::{RefCell, RefMut},
collections::VecDeque,
os::fd::{BorrowedFd, RawFd},
sync::atomic::{AtomicUsize, Ordering},
time::Duration,
};
pub const R_END_INPUT_FUNCTIONS: usize = (ReadlineCmd::ReverseRepeatJump as usize) + 1;
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
pub enum CharInputStyle {
Normal,
NotFirst,
}
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
#[repr(u8)]
pub enum ReadlineCmd {
BeginningOfLine,
EndOfLine,
ForwardChar,
BackwardChar,
BackwardCharPassive,
ForwardSingleChar,
ForwardCharPassive,
BackwardWord,
ForwardWordEmacs,
ForwardBigwordEmacs,
BackwardBigword,
ForwardWordEnd,
BackwardWordEnd,
ForwardBigwordEnd,
BackwardBigwordEnd,
ForwardWordVi,
ForwardBigwordVi,
ForwardPathComponent,
ForwardToken,
BackwardPathComponent,
BackwardToken,
NextdOrForwardWordEmacs,
PrevdOrBackwardWord,
HistoryDelete,
HistorySearchBackward,
HistorySearchForward,
HistoryPrefixSearchBackward,
HistoryPrefixSearchForward,
HistoryPager,
#[deprecated]
HistoryPagerDelete,
DeleteChar,
BackwardDeleteChar,
KillLine,
Yank,
YankPop,
Complete,
CompleteAndSearch,
PagerToggleSearch,
BeginningOfHistory,
EndOfHistory,
BackwardKillLine,
KillWholeLine,
KillInnerLine,
KillWordEmacs,
KillBigwordEmacs,
KillWordVi,
KillBigwordVi,
KillInnerWord,
KillInnerBigWord,
KillAWord,
KillABigWord,
KillPathComponent,
KillToken,
BackwardKillWord,
BackwardKillPathComponent,
BackwardKillBigword,
BackwardKillToken,
HistoryTokenSearchBackward,
HistoryTokenSearchForward,
HistoryLastTokenSearchBackward,
HistoryLastTokenSearchForward,
SelfInsert,
SelfInsertNotFirst,
GetKey,
TransposeChars,
TransposeWords,
UpcaseWord,
DowncaseWord,
CapitalizeWord,
TogglecaseChar,
UpcaseSelection,
DowncaseSelection,
TogglecaseSelection,
Execute,
BeginningOfBuffer,
EndOfBuffer,
RepaintMode,
Repaint,
ForceRepaint,
UpLine,
DownLine,
SuppressAutosuggestion,
AcceptAutosuggestion,
BeginSelection,
SwapSelectionStartStop,
EndSelection,
KillSelection,
InsertLineUnder,
InsertLineOver,
ForwardJump,
BackwardJump,
ForwardJumpTill,
BackwardJumpTill,
JumpToMatchingBracket,
JumpTillMatchingBracket,
FuncAnd,
FuncOr,
ExpandAbbr,
DeleteOrExit,
Exit,
ClearCommandline,
CancelCommandline,
Cancel,
Undo,
Redo,
BeginUndoGroup,
EndUndoGroup,
RepeatJump,
ClearScreenAndRepaint,
ScrollbackPush,
ReverseRepeatJump,
}
#[derive(Clone, Copy, Debug)]
pub struct KeyEvent {
pub key: Key,
pub shifted_codepoint: char,
pub base_layout_codepoint: char,
}
impl KeyEvent {
pub(crate) fn new(modifiers: Modifiers, codepoint: char) -> Self {
Self::from(Key::new(modifiers, codepoint))
}
pub(crate) fn new_with(
modifiers: Modifiers,
codepoint: char,
shifted_key: Option<char>,
base_layout_key: Option<char>,
) -> Self {
Self {
key: Key::new(modifiers, codepoint),
shifted_codepoint: shifted_key.unwrap_or_default(),
base_layout_codepoint: base_layout_key.unwrap_or_default(),
}
}
pub(crate) fn from_raw(codepoint: char) -> Self {
Self::from(Key::from_raw(codepoint))
}
pub fn from_single_byte(c: u8) -> Self {
Self::from(Key::from_single_byte(c))
}
pub(crate) fn codepoint_text(&self) -> Option<char> {
let mut modifiers = self.modifiers;
let mut c = self.codepoint;
if self.shifted_codepoint != '\0' && modifiers.shift {
modifiers.shift = false;
c = self.shifted_codepoint;
}
if modifiers.is_some() {
return None;
}
if c == key::SPACE {
return Some(' ');
}
if c == key::ENTER {
return Some('\n');
}
if c == key::TAB {
return Some('\t');
}
if fish_is_pua(c) || u32::from(c) <= 27 {
return None;
}
Some(c)
}
}
impl From<Key> for KeyEvent {
fn from(key: Key) -> Self {
Self::new_with(key.modifiers, key.codepoint, None, None)
}
}
impl std::ops::Deref for KeyEvent {
type Target = Key;
fn deref(&self) -> &Self::Target {
&self.key
}
}
impl std::ops::DerefMut for KeyEvent {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.key
}
}
fn apply_shift(mut key: Key, do_ascii: bool, shifted_codepoint: char) -> Option<Key> {
if !key.modifiers.shift {
return Some(key);
}
if shifted_codepoint != '\0' {
key.codepoint = shifted_codepoint;
} else if do_ascii && key.codepoint.is_ascii_lowercase() {
key.codepoint = key.codepoint.to_ascii_uppercase();
} else {
return None;
}
key.modifiers.shift = false;
Some(key)
}
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)]
pub(crate) enum KeyMatchQuality {
BaseLayoutModuloShift,
BaseLayout,
ModuloShift,
Exact,
}
impl FloggableDebug for KeyMatchQuality {}
pub(crate) fn match_key_event_to_key(event: &KeyEvent, key: &Key) -> Option<KeyMatchQuality> {
if &event.key == key {
return Some(KeyMatchQuality::Exact);
}
let shifted_evt = apply_shift(event.key, false, event.shifted_codepoint);
let shifted_key = apply_shift(*key, true, '\0');
if shifted_evt.is_some() && shifted_evt == shifted_key {
return Some(KeyMatchQuality::ModuloShift);
}
if event.base_layout_codepoint != '\0' {
let mut base_layout_key = event.key;
base_layout_key.codepoint = event.base_layout_codepoint;
if base_layout_key == *key {
return Some(KeyMatchQuality::BaseLayout);
}
let shifted_base_layout_key = apply_shift(base_layout_key, true, '\0');
if shifted_base_layout_key.is_some() && shifted_base_layout_key == shifted_key {
return Some(KeyMatchQuality::BaseLayoutModuloShift);
}
}
None
}
#[derive(Debug, Clone)]
pub enum CharEventType {
Char(KeyInputEvent),
Readline(ReadlineCmd),
Command(WString),
Eof,
CheckExit,
}
#[derive(Debug, Clone)]
pub struct ReadlineCmdEvent {
pub cmd: ReadlineCmd,
pub seq: WString,
}
#[derive(Debug, Clone)]
pub struct KeyInputEvent {
pub key: KeyEvent,
pub input_style: CharInputStyle,
pub seq: WString,
}
#[derive(Debug, Clone)]
pub enum ImplicitEvent {
Eof,
CheckExit,
FocusIn,
FocusOut,
MouseLeft(ViewportPosition),
NewColorTheme,
NewWindowHeight,
}
#[derive(Debug, Clone)]
pub enum QueryResponse {
PrimaryDeviceAttribute,
BackgroundColor(xterm_color::Color),
CursorPosition(ViewportPosition),
}
#[derive(Debug, Clone)]
pub enum QueryResultEvent {
Response(QueryResponse),
Timeout,
Interrupted,
}
#[derive(Debug, Clone)]
pub enum CharEvent {
Key(KeyInputEvent),
Readline(ReadlineCmdEvent),
Command(WString),
Implicit(ImplicitEvent),
QueryResult(QueryResultEvent),
}
impl FloggableDebug for CharEvent {}
impl CharEvent {
pub fn is_char(&self) -> bool {
matches!(self, CharEvent::Key(_))
}
pub fn is_readline(&self) -> bool {
matches!(self, CharEvent::Readline(_))
}
pub fn is_readline_or_command(&self) -> bool {
matches!(self, CharEvent::Readline(_) | CharEvent::Command(_))
}
pub fn get_char(&self) -> char {
let CharEvent::Key(kevt) = self else {
panic!("Not a char type");
};
kevt.key.codepoint
}
pub fn get_key(&self) -> Option<&KeyInputEvent> {
match self {
CharEvent::Key(kevt) => Some(kevt),
_ => None,
}
}
pub fn get_readline(&self) -> ReadlineCmd {
let CharEvent::Readline(c) = self else {
panic!("Not a readline type");
};
c.cmd
}
pub fn get_command(&self) -> Option<&wstr> {
match self {
CharEvent::Command(c) => Some(c),
_ => None,
}
}
pub fn from_char(c: char) -> CharEvent {
Self::from_key(KeyEvent::from_raw(c))
}
pub fn from_key(key: KeyEvent) -> CharEvent {
Self::from_key_seq(key, WString::new())
}
pub fn from_key_seq(key: KeyEvent, seq: WString) -> CharEvent {
CharEvent::Key(KeyInputEvent {
key,
input_style: CharInputStyle::Normal,
seq,
})
}
pub fn from_readline(cmd: ReadlineCmd) -> CharEvent {
Self::from_readline_seq(cmd, WString::new())
}
pub fn from_readline_seq(cmd: ReadlineCmd, seq: WString) -> CharEvent {
CharEvent::Readline(ReadlineCmdEvent { cmd, seq })
}
pub fn from_check_exit() -> CharEvent {
CharEvent::Implicit(ImplicitEvent::CheckExit)
}
}
const WAIT_ON_ESCAPE_DEFAULT: usize = 30;
static WAIT_ON_ESCAPE_MS: AtomicUsize = AtomicUsize::new(WAIT_ON_ESCAPE_DEFAULT);
const WAIT_ON_SEQUENCE_KEY_INFINITE: usize = usize::MAX;
static WAIT_ON_SEQUENCE_KEY_MS: AtomicUsize = AtomicUsize::new(WAIT_ON_SEQUENCE_KEY_INFINITE);
enum InputEventTrigger {
Byte(u8),
Eof,
Interrupted,
UvarNotified,
IOPortNotified,
TimeoutElapsed,
}
fn readb(in_fd: RawFd) -> Option<u8> {
assert!(in_fd >= 0, "Invalid in fd");
let mut arr: [u8; 1] = [0];
if read_blocked(in_fd, &mut arr) != Ok(1) {
return None;
}
let c = arr[0];
flog!(reader, "Read byte", char_to_symbol(char::from(c), true));
Some(c)
}
fn next_input_event(in_fd: RawFd, ioport_fd: RawFd, timeout: Timeout) -> InputEventTrigger {
let mut fdset = FdReadableSet::new();
loop {
fdset.clear();
fdset.add(in_fd);
fdset.add(ioport_fd);
let notifier = default_notifier();
let notifier_fd = notifier.notification_fd();
if let Some(notifier_fd) = notifier_fd {
fdset.add(notifier_fd);
}
let select_res = fdset.check_readable(timeout);
if select_res < 0 {
let err = errno::errno().0;
if err == libc::EINTR || err == libc::EAGAIN {
return InputEventTrigger::Interrupted;
} else {
return InputEventTrigger::Eof;
}
}
if select_res == 0 {
assert!(!matches!(timeout, Timeout::Forever));
return InputEventTrigger::TimeoutElapsed;
}
if let Some(notifier_fd) = notifier_fd {
if fdset.test(notifier_fd) && notifier.notification_fd_became_readable(notifier_fd) {
return InputEventTrigger::UvarNotified;
}
}
if fdset.test(in_fd) {
return readb(in_fd).map_or(InputEventTrigger::Eof, InputEventTrigger::Byte);
}
if fdset.test(ioport_fd) {
return InputEventTrigger::IOPortNotified;
}
}
}
pub fn check_fd_readable(in_fd: BorrowedFd, timeout: Duration) -> bool {
let mut readfds = {
let mut set = FdSet::new();
set.insert(in_fd);
set
};
let res = nix::sys::select::pselect(
None,
&mut readfds,
None,
None,
&TimeSpec::from_duration(timeout),
&SigSet::all(),
)
.unwrap();
if is_windows_subsystem_for_linux(WSL::V1) {
_ = SigSet::thread_get_mask().expect("Failed to get sigmask!");
}
res > 0
}
pub fn update_wait_on_escape_ms(vars: &EnvStack) {
let fish_escape_delay_ms = vars.get_unless_empty(L!("fish_escape_delay_ms"));
let Some(fish_escape_delay_ms) = fish_escape_delay_ms else {
WAIT_ON_ESCAPE_MS.store(WAIT_ON_ESCAPE_DEFAULT, Ordering::Relaxed);
return;
};
let fish_escape_delay_ms = fish_escape_delay_ms.as_string();
match fish_wcstol(&fish_escape_delay_ms) {
Ok(val) if (10..5000).contains(&val) => {
WAIT_ON_ESCAPE_MS.store(val.try_into().unwrap(), Ordering::Relaxed);
}
_ => {
eprintf!(
concat!(
"ignoring fish_escape_delay_ms: value '%s' ",
"is not an integer or is < 10 or >= 5000 ms\n"
),
fish_escape_delay_ms
);
}
}
}
pub fn update_wait_on_sequence_key_ms(vars: &EnvStack) {
let sequence_key_time_ms = vars.get_unless_empty(L!("fish_sequence_key_delay_ms"));
let Some(sequence_key_time_ms) = sequence_key_time_ms else {
WAIT_ON_SEQUENCE_KEY_MS.store(WAIT_ON_SEQUENCE_KEY_INFINITE, Ordering::Relaxed);
return;
};
let sequence_key_time_ms = sequence_key_time_ms.as_string();
match fish_wcstol(&sequence_key_time_ms) {
Ok(val) if (10..5000).contains(&val) => {
WAIT_ON_SEQUENCE_KEY_MS.store(val.try_into().unwrap(), Ordering::Relaxed);
}
_ => {
eprintf!(
concat!(
"ignoring fish_sequence_key_delay_ms: value '%s' ",
"is not an integer or is < 10 or >= 5000 ms\n"
),
sequence_key_time_ms
);
}
}
}
fn parse_mask(mask: u32) -> (Modifiers, bool) {
let modifiers = Modifiers {
ctrl: (mask & 4) != 0,
alt: (mask & 2) != 0,
shift: (mask & 1) != 0,
sup: (mask & 8) != 0,
};
let caps_lock = (mask & 64) != 0;
(modifiers, caps_lock)
}
#[derive(Default)]
pub struct InputData {
pub in_fd: RawFd,
pub queue: VecDeque<CharEvent>,
pub paste_buffer: Option<Vec<u8>>,
pub input_function_args: Vec<char>,
pub function_status: bool,
pub event_storage: Vec<CharEvent>,
pub blocking_query_timeout: Option<Duration>,
pub blocking_query: RefCell<Option<TerminalQuery>>,
}
impl InputData {
pub fn new(in_fd: RawFd, blocking_query_timeout: Option<Duration>) -> Self {
Self {
in_fd,
queue: VecDeque::new(),
paste_buffer: None,
input_function_args: Vec::new(),
function_status: false,
event_storage: Vec::new(),
blocking_query_timeout,
blocking_query: RefCell::new(None),
}
}
pub fn queue_char(&mut self, ch: CharEvent) {
self.queue.push_back(ch);
}
pub fn function_set_status(&mut self, status: bool) {
self.function_status = status;
}
}
#[derive(Clone, Debug, Default, Eq, PartialEq)]
pub struct BackgroundColorQuery {
pub result: Option<xterm_color::Color>,
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub enum CursorPositionQueryReason {
NewPrompt,
WindowHeightChange,
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct CursorPositionQuery {
pub reason: CursorPositionQueryReason,
pub result: Option<ViewportPosition>,
}
impl CursorPositionQuery {
pub fn new(reason: CursorPositionQueryReason) -> Self {
Self {
reason,
result: None,
}
}
}
#[derive(Clone, Debug, Default, Eq, PartialEq)]
pub struct RecurrentQuery {
pub background_color: Option<BackgroundColorQuery>,
pub cursor_position: Option<CursorPositionQuery>,
}
#[derive(Clone, Eq, PartialEq)]
pub enum TerminalQuery {
Initial,
Recurrent(RecurrentQuery),
}
pub const LONG_READ_TIMEOUT: Duration = Duration::from_secs(10);
pub trait InputEventQueuer {
fn try_pop(&mut self) -> Option<CharEvent> {
if self.is_blocked_querying() {
use ImplicitEvent::*;
match self.get_input_data().queue.front()? {
CharEvent::QueryResult(_) | CharEvent::Implicit(CheckExit | Eof) => {}
CharEvent::Key(_)
| CharEvent::Readline(_)
| CharEvent::Command(_)
| CharEvent::Implicit(_) => {
return None; }
}
}
self.get_input_data_mut().queue.pop_front()
}
fn readch(&mut self) -> CharEvent {
loop {
if let Some(mevt) = self.try_pop() {
return mevt;
}
self.prepare_to_select();
if let Some(mevt) = self.try_pop() {
return mevt;
}
match next_input_event(
self.get_in_fd(),
self.get_ioport_fd(),
if self.is_blocked_querying() {
Timeout::Duration(self.get_input_data().blocking_query_timeout.unwrap())
} else {
Timeout::Forever
},
) {
InputEventTrigger::Eof => {
return CharEvent::Implicit(ImplicitEvent::Eof);
}
InputEventTrigger::Interrupted => {
self.select_interrupted();
}
InputEventTrigger::UvarNotified => {
self.uvar_change_notified();
}
InputEventTrigger::IOPortNotified => {
self.ioport_notified();
}
InputEventTrigger::Byte(read_byte) => {
let mut have_escape_prefix = false;
let mut buffer = vec![read_byte];
let mut key = if read_byte == 0x1b {
self.parse_escape_sequence(&mut buffer, &mut have_escape_prefix)
} else {
canonicalize_control_char(read_byte).map(KeyEvent::from)
};
if self.paste_is_buffering() {
if read_byte != 0x1b {
self.paste_push_char(read_byte);
}
continue;
}
let mut seq = WString::new();
if key.is_some_and(|key| key.key == Key::from_raw(key::INVALID)) {
continue;
}
assert!(key.is_none_or(|key| key.codepoint != key::INVALID));
let ok = loop {
match decode_utf8(&mut seq, InvalidPolicy::Error, &buffer) {
DecodeState::Incomplete => {
buffer.push(
match next_input_event(
self.get_in_fd(),
self.get_ioport_fd(),
Timeout::Forever,
) {
InputEventTrigger::Byte(b) => b,
_ => 0,
},
);
}
DecodeState::Complete => {
if have_escape_prefix {
let c = seq.as_char_slice().last().unwrap();
key = Some(KeyEvent::from(alt(*c)));
}
break true;
}
DecodeState::Error => {
self.push_front(CharEvent::from_check_exit());
break false;
}
}
};
if !ok {
continue;
}
let (key_evt, extra) = if let Some(key) = key {
(CharEvent::from_key_seq(key, seq), None)
} else {
let Some(c) = seq.chars().next() else {
continue;
};
(
CharEvent::from_key_seq(KeyEvent::from_raw(c), seq.clone()),
Some(seq.chars().skip(1).map(CharEvent::from_char)),
)
};
if self.is_blocked_querying() {
flog!(
reader,
"Still blocked on response from terminal, deferring key event",
key_evt
);
self.push_back(key_evt);
extra.map(|extra| {
for evt in extra {
self.push_back(evt);
}
});
let vintr = shell_modes().control_chars[libc::VINTR];
if vintr != 0
&& key.is_some_and(|key| {
match_key_event_to_key(&key, &Key::from_single_byte(vintr))
.is_some()
})
{
flog!(
reader,
"Received interrupt key, giving up waiting for response from terminal"
);
let ok = stop_query(self.blocking_query());
assert!(ok);
self.get_input_data_mut().queue.clear();
self.push_front(CharEvent::QueryResult(QueryResultEvent::Interrupted));
}
continue;
}
extra.map(|extra| self.insert_front(extra));
return key_evt;
}
InputEventTrigger::TimeoutElapsed => {
return CharEvent::QueryResult(QueryResultEvent::Timeout);
}
}
}
}
fn read_sequence_byte(&mut self, buffer: &mut Vec<u8>) -> Option<u8> {
let fd = self.get_in_fd();
let strict = feature_test(FeatureFlag::OmitTermWorkarounds);
let historical_millis = |ms| {
if strict {
LONG_READ_TIMEOUT
} else {
Duration::from_millis(ms)
}
};
if !check_fd_readable(
unsafe { BorrowedFd::borrow_raw(fd) },
if self.paste_is_buffering() || self.is_blocked_querying() {
historical_millis(300)
} else if buffer == b"\x1b" {
Duration::from_millis(1) } else {
historical_millis(30)
},
) {
flog!(
reader,
format!("Incomplete escape sequence: {}", DisplayBytes(buffer))
);
if buffer != b"\x1b" && strict {
flog!(
error,
format!(
"Incomplete escape sequence seen (logging because omit-term-workarounds is on): {}",
DisplayBytes(buffer)
)
);
}
return None;
}
let next = readb(fd)?;
buffer.push(next);
Some(next)
}
fn parse_escape_sequence(
&mut self,
buffer: &mut Vec<u8>,
have_escape_prefix: &mut bool,
) -> Option<KeyEvent> {
assert!(buffer.len() <= 2);
let recursive_invocation = buffer.len() == 2;
let Some(next) = self.read_sequence_byte(buffer) else {
return Some(KeyEvent::from_raw(key::ESCAPE));
};
let invalid = KeyEvent::from_raw(key::INVALID);
if recursive_invocation && next == b'\x1b' {
return Some(
match self.parse_escape_sequence(buffer, have_escape_prefix) {
Some(mut nested_sequence) => {
if nested_sequence.key == invalid.key {
return Some(KeyEvent::from_raw(key::ESCAPE));
}
nested_sequence.modifiers.alt = true;
nested_sequence
}
_ => invalid,
},
);
}
if next == b'[' {
return Some(self.parse_csi(buffer).unwrap_or(invalid));
}
if next == b'O' {
return Some(self.parse_ss3(buffer).unwrap_or(invalid));
}
if !recursive_invocation {
if next == b']' {
self.parse_osc(buffer);
return Some(invalid);
}
if next == b'P' {
return Some(self.parse_dcs(buffer).unwrap_or(invalid));
}
}
match canonicalize_control_char(next) {
Some(mut key) => {
key.modifiers.alt = true;
Some(KeyEvent::from(key))
}
None => {
*have_escape_prefix = true;
None
}
}
}
fn parse_csi(&mut self, buffer: &mut Vec<u8>) -> Option<KeyEvent> {
let mut params = [[0_u32; 4]; 16];
let Some(mut c) = self.read_sequence_byte(buffer) else {
return Some(KeyEvent::from(alt('[')));
};
let mut next_char = |zelf: &mut Self| zelf.read_sequence_byte(buffer).unwrap_or(0xff);
let private_mode;
if matches!(c, b'?' | b'<' | b'=' | b'>') {
private_mode = Some(c);
c = next_char(self);
} else {
private_mode = None;
}
let mut count = 0;
let mut subcount = 0;
while count < 16 && (0x30..=0x3f).contains(&c) {
if c.is_ascii_digit() {
match params[count][subcount]
.checked_mul(10)
.and_then(|result| result.checked_add(u32::from(c - b'0')))
{
Some(c) => params[count][subcount] = c,
None => return invalid_sequence(buffer),
}
} else if c == b':' && subcount < 3 {
subcount += 1;
} else if c == b';' {
count += 1;
subcount = 0;
} else {
return None;
}
c = next_char(self);
}
if c != b'$' && !(0x40..=0x7e).contains(&c) {
return None;
}
let kitty_key = |key: char, shifted_key: Option<char>, base_layout_key: Option<char>| {
let mask = params[1][0].saturating_sub(1);
let (mut modifiers, caps_lock) = parse_mask(mask);
if caps_lock && modifiers == Modifiers::SHIFT && !key.to_uppercase().eq(Some(key)) {
modifiers.shift = false;
}
KeyEvent::new_with(modifiers, key, shifted_key, base_layout_key)
};
let masked_key = |key: char| kitty_key(key, None, None);
let key = match c {
b'$' => {
if next_char(self) == b'y' {
return None;
}
match params[0][0] {
23 | 24 => KeyEvent::from(shift(
char::from_u32(u32::from(function_key(11)) + params[0][0] - 23).unwrap(), )),
_ => return None,
}
}
b'A' => masked_key(key::UP),
b'B' => masked_key(key::DOWN),
b'C' => masked_key(key::RIGHT),
b'D' => masked_key(key::LEFT),
b'E' => masked_key('5'), b'F' => masked_key(key::END), b'H' => masked_key(key::HOME), b'M' | b'm' => {
flog!(reader, "mouse event");
let sgr = private_mode == Some(b'<');
if !sgr && c == b'm' {
return None;
}
let Some(button) = (if sgr {
Some(params[0][0])
} else {
u32::from(next_char(self)).checked_sub(32)
}) else {
return invalid_sequence(buffer);
};
let mut convert = |param| {
(if sgr {
Some(param)
} else {
u32::from(next_char(self)).checked_sub(32)
})
.and_then(|coord| coord.checked_sub(1))
.and_then(|coord| usize::try_from(coord).ok())
};
let Some(x) = convert(params[1][0]) else {
return invalid_sequence(buffer);
};
let Some(y) = convert(params[2][0]) else {
return invalid_sequence(buffer);
};
let position = ViewportPosition { x, y };
let (modifiers, _caps_lock) = parse_mask((button >> 2) & 0x07);
let code = button & 0x43;
if code != 0 || c != b'M' || modifiers.is_some() {
return None;
}
self.push_front(CharEvent::Implicit(ImplicitEvent::MouseLeft(position)));
return None;
}
b't' => {
flog!(reader, "mouse event");
let _ = next_char(self);
let _ = next_char(self);
return None;
}
b'T' => {
flog!(reader, "mouse event");
for _ in 0..6 {
let _ = next_char(self);
}
return None;
}
b'P' => masked_key(function_key(1)),
b'Q' => masked_key(function_key(2)),
b'R' => {
let Some(y) = params[0][0]
.checked_sub(1)
.and_then(|y| usize::try_from(y).ok())
else {
return invalid_sequence(buffer);
};
let Some(x) = params[1][0]
.checked_sub(1)
.and_then(|x| usize::try_from(x).ok())
else {
return invalid_sequence(buffer);
};
flog!(reader, "Received cursor position report y:", y, "x:", x);
let cursor_pos = ViewportPosition { x, y };
self.push_query_response(QueryResponse::CursorPosition(cursor_pos));
return None;
}
b'S' => masked_key(function_key(4)),
b'~' => match params[0][0] {
1 => masked_key(key::HOME), 2 => masked_key(key::INSERT),
3 => masked_key(key::DELETE),
4 => masked_key(key::END), 5 => masked_key(key::PAGE_UP),
6 => masked_key(key::PAGE_DOWN),
7 => masked_key(key::HOME), 8 => masked_key(key::END), 11..=15 => masked_key(
char::from_u32(u32::from(function_key(1)) + params[0][0] - 11).unwrap(),
),
17..=21 => masked_key(
char::from_u32(u32::from(function_key(6)) + params[0][0] - 17).unwrap(),
),
23 | 24 => masked_key(
char::from_u32(u32::from(function_key(11)) + params[0][0] - 23).unwrap(),
),
25 | 26 => KeyEvent::from(shift(
char::from_u32(u32::from(function_key(3)) + params[0][0] - 25).unwrap(),
)), 27 => {
let Some(key) = char::from_u32(params[2][0]) else {
return invalid_sequence(buffer);
};
masked_key(canonicalize_keyed_control_char(key))
}
28 | 29 => KeyEvent::from(shift(
char::from_u32(u32::from(function_key(5)) + params[0][0] - 28).unwrap(),
)), 31 | 32 => KeyEvent::from(shift(
char::from_u32(u32::from(function_key(7)) + params[0][0] - 31).unwrap(),
)), 33 | 34 => KeyEvent::from(shift(
char::from_u32(u32::from(function_key(9)) + params[0][0] - 33).unwrap(),
)), 200 => {
self.paste_start_buffering();
return None;
}
201 => {
self.paste_commit();
return None;
}
_ => return None,
},
b'c' if private_mode == Some(b'?') => {
flog!(reader, "Received Primary Device Attribute response");
self.push_query_response(QueryResponse::PrimaryDeviceAttribute);
return None;
}
b'n' if private_mode == Some(b'?') && params[0] == [997, 0, 0, 0] => {
match params[1] {
[1, 0, 0, 0] | [2, 0, 0, 0] => (),
_ => return None,
}
flog!(reader, "Received color theme change");
self.push_front(CharEvent::Implicit(ImplicitEvent::NewColorTheme));
return None;
}
b'u' => {
if private_mode == Some(b'?') {
maybe_set_kitty_keyboard_capability();
return None;
}
let key = match params[0][0] {
57361 => key::PRINT_SCREEN,
57363 => key::MENU,
57399 => '0',
57400 => '1',
57401 => '2',
57402 => '3',
57403 => '4',
57404 => '5',
57405 => '6',
57406 => '7',
57407 => '8',
57408 => '9',
57409 => '.',
57410 => '/',
57411 => '*',
57412 => '-',
57413 => '+',
57414 => key::ENTER,
57415 => '=',
57417 => key::LEFT,
57418 => key::RIGHT,
57419 => key::UP,
57420 => key::DOWN,
57421 => key::PAGE_UP,
57422 => key::PAGE_DOWN,
57423 => key::HOME,
57424 => key::END,
57425 => key::INSERT,
57426 => key::DELETE,
cp => {
let Some(key) = char::from_u32(cp) else {
return invalid_sequence(buffer);
};
canonicalize_keyed_control_char(key)
}
};
let Some(shifted_key) = char::from_u32(params[0][1]) else {
return invalid_sequence(buffer);
};
let Some(base_layout_key) = char::from_u32(params[0][2]) else {
return invalid_sequence(buffer);
};
kitty_key(
key,
Some(canonicalize_keyed_control_char(shifted_key)),
Some(base_layout_key),
)
}
b'Z' => KeyEvent::from(shift(key::TAB)),
b'I' => {
self.push_front(CharEvent::Implicit(ImplicitEvent::FocusIn));
return None;
}
b'O' => {
self.push_front(CharEvent::Implicit(ImplicitEvent::FocusOut));
return None;
}
_ => return None,
};
Some(key)
}
fn parse_ss3(&mut self, buffer: &mut Vec<u8>) -> Option<KeyEvent> {
let mut raw_mask = 0;
let Some(mut code) = self.read_sequence_byte(buffer) else {
return Some(KeyEvent::from(alt('O')));
};
while code.is_ascii_digit() {
raw_mask = raw_mask * 10 + u32::from(code - b'0');
code = self.read_sequence_byte(buffer).unwrap_or(0xff);
}
let (modifiers, _caps_lock) = parse_mask(raw_mask.saturating_sub(1));
#[rustfmt::skip]
let key = match code {
b' ' => KeyEvent::new(modifiers, key::SPACE),
b'A' => KeyEvent::new(modifiers, key::UP),
b'B' => KeyEvent::new(modifiers, key::DOWN),
b'C' => KeyEvent::new(modifiers, key::RIGHT),
b'D' => KeyEvent::new(modifiers, key::LEFT),
b'F' => KeyEvent::new(modifiers, key::END),
b'H' => KeyEvent::new(modifiers, key::HOME),
b'I' => KeyEvent::new(modifiers, key::TAB),
b'M' => KeyEvent::new(modifiers, key::ENTER),
b'P' => KeyEvent::new(modifiers, function_key(1)),
b'Q' => KeyEvent::new(modifiers, function_key(2)),
b'R' => KeyEvent::new(modifiers, function_key(3)),
b'S' => KeyEvent::new(modifiers, function_key(4)),
b'X' => KeyEvent::new(modifiers, '='),
b'j' => KeyEvent::new(modifiers, '*'),
b'k' => KeyEvent::new(modifiers, '+'),
b'l' => KeyEvent::new(modifiers, ','),
b'm' => KeyEvent::new(modifiers, '-'),
b'n' => KeyEvent::new(modifiers, '.'),
b'o' => KeyEvent::new(modifiers, '/'),
b'p' => KeyEvent::new(modifiers, '0'),
b'q' => KeyEvent::new(modifiers, '1'),
b'r' => KeyEvent::new(modifiers, '2'),
b's' => KeyEvent::new(modifiers, '3'),
b't' => KeyEvent::new(modifiers, '4'),
b'u' => KeyEvent::new(modifiers, '5'),
b'v' => KeyEvent::new(modifiers, '6'),
b'w' => KeyEvent::new(modifiers, '7'),
b'x' => KeyEvent::new(modifiers, '8'),
b'y' => KeyEvent::new(modifiers, '9'),
_ => return None,
};
Some(key)
}
fn read_until_sequence_terminator(
&mut self,
buffer: &mut Vec<u8>,
allow_bel: bool,
) -> Option<()> {
let mut escape = false;
loop {
let b = self.read_sequence_byte(buffer)?;
if allow_bel && b == b'\x07' {
buffer.pop();
return Some(());
}
if escape && b == b'\\' {
buffer.pop();
buffer.pop();
return Some(());
}
escape = b == b'\x1b';
}
}
fn parse_xtversion(&mut self, buffer: &mut Vec<u8>) -> Option<()> {
assert_eq!(buffer, b"\x1bP>");
self.read_until_sequence_terminator(buffer, false)?;
if buffer.get(3)? != &b'|' {
return None;
}
XTVERSION.get_or_init(|| {
let xtversion = bytes2wcstring(&buffer[4..buffer.len()]);
flog!(
reader,
format!("Received XTVERSION response: {}", xtversion)
);
xtversion
});
None
}
fn parse_osc(&mut self, buffer: &mut Vec<u8>) -> Option<()> {
let osc_prefix = b"\x1b]";
assert_eq!(buffer, osc_prefix);
self.read_until_sequence_terminator(buffer, true)?;
let buffer = &buffer[osc_prefix.len()..];
let buffer = buffer.strip_prefix(b"11;")?;
let c = xterm_color::Color::parse(buffer).ok()?;
flog!(reader, format!("Received background color {c:?}"));
self.push_query_response(QueryResponse::BackgroundColor(c));
None
}
fn parse_dcs(&mut self, buffer: &mut Vec<u8>) -> Option<KeyEvent> {
assert_eq!(buffer, b"\x1bP");
let Some(success) = self.read_sequence_byte(buffer) else {
return Some(KeyEvent::from(alt('P')));
};
let success = match success {
b'0' => false,
b'1' => true,
b'>' => {
self.parse_xtversion(buffer);
return None;
}
_ => return None,
};
if self.read_sequence_byte(buffer)? != b'+' {
return None;
}
if self.read_sequence_byte(buffer)? != b'r' {
return None;
}
self.read_until_sequence_terminator(buffer, false)?;
let buffer = &buffer[5..];
if !success {
flog!(
reader,
format!(
"Received XTGETTCAP failure response: {}",
bytes2wcstring(&parse_hex(buffer)?),
)
);
return None;
}
let mut buffer = buffer.splitn(2, |&c| c == b'=');
let key = buffer.next().unwrap();
let key = parse_hex(key)?;
let value = if let Some(value) = buffer.next() {
let value = parse_hex(value)?;
flog!(
reader,
format!(
"Received XTGETTCAP response: {}={:?}",
bytes2wcstring(&key),
bytes2wcstring(&value)
)
);
Some(value)
} else {
flog!(
reader,
format!("Received XTGETTCAP response: {}", bytes2wcstring(&key))
);
None
};
if key == SCROLL_CONTENT_UP_TERMINFO_CODE.as_bytes() {
maybe_set_scroll_content_up_capability();
} else if key == XTGETTCAP_QUERY_OS_NAME.as_bytes() {
if let Some(value) = value {
TERMINAL_OS_NAME.get_or_init(|| Some(bytes2wcstring(&value)));
}
}
None
}
fn readch_timed_esc(&mut self) -> Option<CharEvent> {
self.readch_timed(WAIT_ON_ESCAPE_MS.load(Ordering::Relaxed))
}
fn readch_timed_sequence_key(&mut self) -> Option<CharEvent> {
let wait_on_sequence_key_ms = WAIT_ON_SEQUENCE_KEY_MS.load(Ordering::Relaxed);
if wait_on_sequence_key_ms == WAIT_ON_SEQUENCE_KEY_INFINITE {
return Some(self.readch());
}
self.readch_timed(wait_on_sequence_key_ms)
}
fn readch_timed(&mut self, wait_time_ms: usize) -> Option<CharEvent> {
if let Some(evt) = self.try_pop() {
return Some(evt);
}
check_fd_readable(
unsafe { BorrowedFd::borrow_raw(self.get_in_fd()) },
Duration::from_millis(u64::try_from(wait_time_ms).unwrap()),
)
.then(|| self.readch())
}
fn get_in_fd(&self) -> RawFd {
self.get_input_data().in_fd
}
fn get_ioport_fd(&self) -> RawFd {
-1
}
fn get_input_data(&self) -> &InputData;
fn get_input_data_mut(&mut self) -> &mut InputData;
fn paste_start_buffering(&mut self) {
self.get_input_data_mut().paste_buffer = Some(Vec::new());
}
fn paste_is_buffering(&self) -> bool {
self.get_input_data().paste_buffer.is_some()
}
fn paste_push_char(&mut self, b: u8) {
self.get_input_data_mut()
.paste_buffer
.as_mut()
.unwrap()
.push(b);
}
fn paste_commit(&mut self) {
self.get_input_data_mut().paste_buffer = None;
}
fn push_back(&mut self, ch: CharEvent) {
self.get_input_data_mut().queue.push_back(ch);
}
fn push_front(&mut self, ch: CharEvent) {
self.get_input_data_mut().queue.push_front(ch);
}
fn push_query_response(&mut self, response: QueryResponse) {
self.push_front(CharEvent::QueryResult(QueryResultEvent::Response(response)));
}
fn promote_interruptions_to_front(&mut self) {
let queue = &mut self.get_input_data_mut().queue;
let is_char = |evt: &CharEvent| {
evt.is_char() || matches!(evt, CharEvent::Implicit(ImplicitEvent::Eof))
};
let Some(first): Option<usize> = queue.iter().position(|e| !is_char(e)) else {
return;
};
let last = queue
.range(first..)
.position(is_char)
.map_or(queue.len(), |x| x + first);
let elems: Vec<CharEvent> = queue.drain(first..last).collect();
for elem in elems.into_iter().rev() {
queue.push_front(elem);
}
}
fn insert_front<I>(&mut self, evts: I)
where
I: IntoIterator<Item = CharEvent>,
I::IntoIter: DoubleEndedIterator,
{
let queue = &mut self.get_input_data_mut().queue;
let iter = evts.into_iter().rev();
queue.reserve(iter.size_hint().0);
for evt in iter {
queue.push_front(evt);
}
}
fn drop_leading_readline_events(&mut self) {
let queue = &mut self.get_input_data_mut().queue;
while let Some(evt) = queue.front() {
if evt.is_readline_or_command() {
queue.pop_front();
} else {
break;
}
}
}
fn blocking_query(&self) -> RefMut<'_, Option<TerminalQuery>> {
self.get_input_data().blocking_query.borrow_mut()
}
fn is_blocked_querying(&self) -> bool {
self.blocking_query().is_some()
}
fn prepare_to_select(&mut self) {}
fn select_interrupted(&mut self) {}
fn enqueue_interrupt_key(&mut self) {
let vintr = shell_modes().control_chars[libc::VINTR];
if vintr != 0 {
let interrupt_evt = CharEvent::from_key(KeyEvent::from_single_byte(vintr));
if stop_query(self.blocking_query()) {
flog!(
reader,
"Received interrupt, giving up on waiting for terminal response"
);
self.get_input_data_mut().queue.clear();
self.push_front(CharEvent::QueryResult(QueryResultEvent::Interrupted));
} else {
self.push_front(interrupt_evt);
}
}
}
fn uvar_change_notified(&mut self) {}
fn ioport_notified(&mut self) {}
fn get_function_status(&self) -> bool {
self.get_input_data().function_status
}
fn has_lookahead(&self) -> bool {
!self.get_input_data().queue.is_empty()
}
}
pub(crate) enum DecodeState {
Incomplete,
Complete,
Error,
}
#[derive(Eq, PartialEq)]
pub(crate) enum InvalidPolicy {
Error,
Passthrough,
}
pub(crate) fn decode_utf8(
out_seq: &mut WString,
invalid_policy: InvalidPolicy,
buffer: &[u8],
) -> DecodeState {
use DecodeState::*;
match std::str::from_utf8(buffer) {
Ok(parsed_str) => {
for c in parsed_str.chars() {
if !fish_reserved_codepoint(c) {
out_seq.push(c);
}
}
Complete
}
Err(e) => match e.error_len() {
Some(_) => match invalid_policy {
InvalidPolicy::Error => {
flog!(reader, "Illegal input encoding");
Error
}
InvalidPolicy::Passthrough => {
for &b in buffer {
out_seq.push(encode_byte_to_char(b));
}
Complete
}
},
None => Incomplete,
},
}
}
pub(crate) fn stop_query(mut query: RefMut<'_, Option<TerminalQuery>>) -> bool {
query.take().is_some()
}
fn invalid_sequence(buffer: &[u8]) -> Option<KeyEvent> {
flog!(
reader,
"Error: invalid escape sequence: ",
DisplayBytes(buffer)
);
None
}
struct DisplayBytes<'a>(&'a [u8]);
impl<'a> std::fmt::Display for DisplayBytes<'a> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
for (i, &c) in self.0.iter().enumerate() {
if i != 0 {
write!(f, " ")?;
}
write!(f, "{}", char_to_symbol(char::from(c), i == 0))?;
}
Ok(())
}
}
impl<'a> FloggableDisplay for DisplayBytes<'a> {}
pub struct InputEventQueue {
data: InputData,
}
impl InputEventQueue {
pub fn new(in_fd: RawFd, blocking_query_timeout: Option<Duration>) -> Self {
Self {
data: InputData::new(in_fd, blocking_query_timeout),
}
}
}
impl InputEventQueuer for InputEventQueue {
fn get_input_data(&self) -> &InputData {
&self.data
}
fn get_input_data_mut(&mut self) -> &mut InputData {
&mut self.data
}
fn select_interrupted(&mut self) {
if reader_test_and_clear_interrupted() != 0 {
self.enqueue_interrupt_key();
}
}
}
fn parse_hex(hex: &[u8]) -> Option<Vec<u8>> {
if hex.len() % 2 != 0 {
return None;
}
let mut result = vec![0; hex.len() / 2];
parse_hex_into(&mut result, hex)?;
Some(result)
}
fn parse_hex_into(out: &mut [u8], hex: &[u8]) -> Option<()> {
assert_eq!(out.len() * 2, hex.len());
let mut i = 0;
while i < hex.len() {
let d1 = char::from(hex[i]).to_digit(16)?;
let d2 = char::from(hex[i + 1]).to_digit(16)?;
let decoded = u8::try_from(16 * d1 + d2).unwrap();
out[i / 2] = decoded;
i += 2;
}
Some(())
}
#[cfg(test)]
mod tests {
use super::{
match_key_event_to_key, parse_hex, CharEvent, InputEventQueue, InputEventQueuer as _,
KeyEvent, KeyMatchQuality, ReadlineCmd,
};
use crate::key::{Key, Modifiers};
#[test]
fn test_match_key_event_to_key() {
macro_rules! validate {
($evt:expr, $key:expr, $expected:expr) => {
assert_eq!(match_key_event_to_key(&$evt, &$key), $expected);
};
}
let none = Modifiers::default();
let shift = Modifiers::SHIFT;
let ctrl = Modifiers::CTRL;
let ctrl_shift = Modifiers {
ctrl: true,
shift: true,
..Default::default()
};
let exact = KeyMatchQuality::Exact;
let modulo_shift = KeyMatchQuality::ModuloShift;
let base_layout = KeyMatchQuality::BaseLayout;
let base_layout_modulo_shift = KeyMatchQuality::BaseLayoutModuloShift;
validate!(KeyEvent::new(none, 'a'), Key::new(none, 'a'), Some(exact));
validate!(KeyEvent::new(none, 'a'), Key::new(none, 'A'), None);
validate!(KeyEvent::new(shift, 'a'), Key::new(shift, 'a'), Some(exact));
validate!(KeyEvent::new(shift, 'a'), Key::new(none, 'A'), None);
validate!(KeyEvent::new(shift, 'ä'), Key::new(none, 'Ä'), None);
validate!(
KeyEvent::new(none, 'A'),
Key::new(shift, 'a'),
Some(modulo_shift)
);
validate!(KeyEvent::new(none, 'A'), Key::new(shift, 'A'), None);
validate!(KeyEvent::new(none, 'Ä'), Key::new(none, 'Ä'), Some(exact));
validate!(KeyEvent::new(none, 'Ä'), Key::new(shift, 'ä'), None);
let ctrl_shift_equals = KeyEvent::new_with(ctrl_shift, '=', Some('+'), None);
validate!(ctrl_shift_equals, Key::new(ctrl_shift, '='), Some(exact));
validate!(ctrl_shift_equals, Key::new(ctrl, '+'), Some(modulo_shift)); validate!(ctrl_shift_equals, Key::new(ctrl_shift, '+'), None);
validate!(ctrl_shift_equals, Key::new(ctrl, '='), None);
let caps_ctrl_shift_ä = KeyEvent::new(ctrl_shift, 'ä');
validate!(caps_ctrl_shift_ä, Key::new(ctrl_shift, 'ä'), Some(exact)); validate!(caps_ctrl_shift_ä, Key::new(ctrl, 'ä'), None);
validate!(caps_ctrl_shift_ä, Key::new(ctrl, 'Ä'), None); validate!(caps_ctrl_shift_ä, Key::new(ctrl_shift, 'Ä'), None);
let caps_ctrl_shift_ä = KeyEvent::new_with(ctrl_shift, 'ä', Some('Ä'), None);
validate!(caps_ctrl_shift_ä, Key::new(ctrl_shift, 'ä'), Some(exact)); validate!(caps_ctrl_shift_ä, Key::new(ctrl, 'ä'), None);
validate!(caps_ctrl_shift_ä, Key::new(ctrl, 'Ä'), Some(modulo_shift)); validate!(caps_ctrl_shift_ä, Key::new(ctrl_shift, 'Ä'), None);
let ctrl_ц = KeyEvent::new_with(ctrl, 'ц', None, Some('w'));
let ctrl_shift_ц = KeyEvent::new_with(ctrl_shift, 'ц', Some('Ц'), Some('w'));
validate!(ctrl_ц, Key::new(ctrl, 'ц'), Some(exact));
validate!(ctrl_ц, Key::new(ctrl, 'w'), Some(base_layout));
validate!(ctrl_ц, Key::new(ctrl_shift, 'ц'), None);
validate!(ctrl_ц, Key::new(ctrl_shift, 'w'), None);
validate!(
ctrl_shift_ц,
Key::new(ctrl, 'W'),
Some(base_layout_modulo_shift)
);
validate!(ctrl_shift_ц, Key::new(ctrl, 'w'), None);
validate!(ctrl_shift_ц, Key::new(ctrl, 'Ц'), Some(modulo_shift));
validate!(ctrl_shift_ц, Key::new(ctrl_shift, 'w'), Some(base_layout));
}
#[test]
fn test_push_front_back() {
let mut queue = InputEventQueue::new(0, None);
queue.push_front(CharEvent::from_char('a'));
queue.push_front(CharEvent::from_char('b'));
queue.push_back(CharEvent::from_char('c'));
queue.push_back(CharEvent::from_char('d'));
assert_eq!(queue.try_pop().unwrap().get_char(), 'b');
assert_eq!(queue.try_pop().unwrap().get_char(), 'a');
assert_eq!(queue.try_pop().unwrap().get_char(), 'c');
assert_eq!(queue.try_pop().unwrap().get_char(), 'd');
assert!(queue.try_pop().is_none());
}
#[test]
fn test_promote_interruptions_to_front() {
let mut queue = InputEventQueue::new(0, None);
queue.push_back(CharEvent::from_char('a'));
queue.push_back(CharEvent::from_char('b'));
queue.push_back(CharEvent::from_readline(ReadlineCmd::Undo));
queue.push_back(CharEvent::from_readline(ReadlineCmd::Redo));
queue.push_back(CharEvent::from_char('c'));
queue.push_back(CharEvent::from_char('d'));
queue.promote_interruptions_to_front();
assert_eq!(queue.try_pop().unwrap().get_readline(), ReadlineCmd::Undo);
assert_eq!(queue.try_pop().unwrap().get_readline(), ReadlineCmd::Redo);
assert_eq!(queue.try_pop().unwrap().get_char(), 'a');
assert_eq!(queue.try_pop().unwrap().get_char(), 'b');
assert_eq!(queue.try_pop().unwrap().get_char(), 'c');
assert_eq!(queue.try_pop().unwrap().get_char(), 'd');
assert!(!queue.has_lookahead());
queue.push_back(CharEvent::from_char('e'));
queue.promote_interruptions_to_front();
assert_eq!(queue.try_pop().unwrap().get_char(), 'e');
assert!(!queue.has_lookahead());
}
#[test]
fn test_insert_front() {
let mut queue = InputEventQueue::new(0, None);
queue.push_back(CharEvent::from_char('a'));
queue.push_back(CharEvent::from_char('b'));
let events = vec![
CharEvent::from_char('A'),
CharEvent::from_char('B'),
CharEvent::from_char('C'),
];
queue.insert_front(events);
assert_eq!(queue.try_pop().unwrap().get_char(), 'A');
assert_eq!(queue.try_pop().unwrap().get_char(), 'B');
assert_eq!(queue.try_pop().unwrap().get_char(), 'C');
assert_eq!(queue.try_pop().unwrap().get_char(), 'a');
assert_eq!(queue.try_pop().unwrap().get_char(), 'b');
}
#[test]
fn test_parse_hex() {
assert_eq!(parse_hex(b"3d"), Some(vec![61]));
}
}