From f75e5c300c4694f918b34b7c1f24294fc8f7537c Mon Sep 17 00:00:00 2001 From: Matthieu Felix Date: Wed, 1 Nov 2023 21:30:03 -0400 Subject: [PATCH 1/2] Custom sequence bindings starting with non-modified keys Single-character bindings of keys pressed without a modifier are already allowed ([emacs](https://github.com/kkawakam/rustyline/blob/23cb8a16135dd885a356cc243d3a8cd0da0ef0e6/src/keymap.rs#L526), [vi command](https://github.com/kkawakam/rustyline/blob/23cb8a16135dd885a356cc243d3a8cd0da0ef0e6/src/keymap.rs#L697), [vi insert](https://github.com/kkawakam/rustyline/blob/23cb8a16135dd885a356cc243d3a8cd0da0ef0e6/src/keymap.rs#L874)), but these don't work for multi-character sequences whose first character is not modified. The current behavior for them just prints the first character. This change modifies `InputState::emacs` and `InputState::vi_insert` to check whether an unmodified key is the beginning of a custom sequence binding before inserting. If a sequence binding is started and fails to find a match, all printable keys pressed are emitted. --- src/keymap.rs | 65 +++++++++++++++++++++++++++++++++++++-------------- 1 file changed, 47 insertions(+), 18 deletions(-) diff --git a/src/keymap.rs b/src/keymap.rs index b7303df84..a062fe160 100644 --- a/src/keymap.rs +++ b/src/keymap.rs @@ -531,13 +531,6 @@ impl<'b> InputState<'b> { return Ok(cmd); } let cmd = match key { - E(K::Char(c), M::NONE) => { - if positive { - Cmd::SelfInsert(n, c) - } else { - Cmd::Unknown - } - } E(K::Char('A'), M::CTRL) => Cmd::Move(Movement::BeginningOfLine), E(K::Char('B'), M::CTRL) => Cmd::Move(if positive { Movement::BackwardChar(n) @@ -648,7 +641,20 @@ impl<'b> InputState<'b> { // TODO ESC-R (r): Undo all changes made to this line. E(K::Char('U' | 'u'), M::ALT) => Cmd::UpcaseWord, E(K::Char('Y' | 'y'), M::ALT) => Cmd::YankPop, - _ => self.common(rdr, wrt, evt, key, n, positive)?, + event => { + let cmd = self.common(rdr, wrt, evt, key, n, positive)?; + if cmd != Cmd::Unknown { + cmd + } else if let E(K::Char(c), M::NONE) = event { + if positive { + Cmd::SelfInsert(n, c) + } else { + Cmd::Unknown + } + } else { + Cmd::Unknown + } + } }; debug!(target: "rustyline", "Emacs command: {:?}", cmd); Ok(cmd) @@ -879,13 +885,6 @@ impl<'b> InputState<'b> { return Ok(cmd); } let cmd = match key { - E(K::Char(c), M::NONE) => { - if self.input_mode == InputMode::Replace { - Cmd::Overwrite(c) - } else { - Cmd::SelfInsert(1, c) - } - } E(K::Char('H'), M::CTRL) | E::BACKSPACE => Cmd::Kill(Movement::BackwardChar(1)), E(K::BackTab, M::NONE) => Cmd::CompleteBackward, E(K::Char('I'), M::CTRL) | E(K::Tab, M::NONE) => Cmd::Complete, @@ -904,7 +903,20 @@ impl<'b> InputState<'b> { wrt.done_inserting(); Cmd::Move(Movement::BackwardChar(1)) } - _ => self.common(rdr, wrt, evt, key, 1, true)?, + event => { + let cmd = self.common(rdr, wrt, evt, key, 1, true)?; + if cmd != Cmd::Unknown { + cmd + } else if let E(K::Char(c), M::NONE) = event { + if self.input_mode == InputMode::Replace { + Cmd::Overwrite(c) + } else { + Cmd::SelfInsert(1, c) + } + } else { + Cmd::Unknown + } + } }; debug!(target: "rustyline", "Vi insert: {:?}", cmd); if cmd.is_repeatable_change() { @@ -1163,8 +1175,8 @@ impl<'b> InputState<'b> { } else { break; } - let handler = subtrie.get(evt).unwrap(); - if let Some(handler) = handler { + let handler = subtrie.get(evt); + if let Ok(Some(handler)) = handler { let cmd = match handler { EventHandler::Simple(cmd) => Some(cmd.clone()), EventHandler::Conditional(handler) => { @@ -1175,12 +1187,29 @@ impl<'b> InputState<'b> { if cmd.is_some() { return Ok(cmd); } + } else { + return Ok(Some(Cmd::Insert(1, letter_sequence(evt)))); } } Ok(None) } } +fn letter_sequence(evt: &Event) -> String { + if let Event::KeySeq(key_seq) = evt { + key_seq + .iter() + .filter_map(|k| match k.0 { + K::Char(c) => Some(c), + K::Enter => Some('\n'), + _ => None, + }) + .collect() + } else { + "".to_string() + } +} + #[cfg(not(feature = "custom-bindings"))] impl<'b> InputState<'b> { fn custom_binding(&self, _: &dyn Refresher, _: &Event, _: RepeatCount, _: bool) -> Option { From b670ab476b0c47a42ddbb47a4c51488ed2aa9c5b Mon Sep 17 00:00:00 2001 From: Matthieu Felix Date: Wed, 15 Nov 2023 19:46:09 -0500 Subject: [PATCH 2/2] Add an example --- examples/custom_key_bindings.rs | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/examples/custom_key_bindings.rs b/examples/custom_key_bindings.rs index 78db874cd..da020a9e9 100644 --- a/examples/custom_key_bindings.rs +++ b/examples/custom_key_bindings.rs @@ -100,6 +100,10 @@ fn main() -> Result<()> { Event::KeySeq(vec![KeyEvent::ctrl('X'), KeyEvent::ctrl('E')]), EventHandler::Simple(Cmd::Suspend), // TODO external editor ); + rl.bind_sequence( + Event::KeySeq(vec!['\\'.into(), '8'.into()]), + EventHandler::Simple(Cmd::Insert(1, "∞".to_string())), + ); loop { let line = rl.readline("> ")?;