1#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
5pub struct KeyEvent(pub KeyCode, pub Modifiers);
6
7impl KeyEvent {
8 pub(crate) const BACKSPACE: Self = Self(KeyCode::Backspace, Modifiers::NONE);
10 pub(crate) const ENTER: Self = Self(KeyCode::Enter, Modifiers::NONE);
12 pub(crate) const ESC: Self = Self(KeyCode::Esc, Modifiers::NONE);
14
15 #[must_use]
17 pub fn new(c: char, mut mods: Modifiers) -> Self {
18 use {KeyCode as K, KeyEvent as E, Modifiers as M};
19
20 if !c.is_control() {
21 if !mods.is_empty() {
22 mods.remove(M::SHIFT); }
25 return E(K::Char(c), mods);
26 }
27 match c {
28 '\x00' => E(K::Char('@'), mods | M::CTRL), '\x01' => E(K::Char('A'), mods | M::CTRL),
30 '\x02' => E(K::Char('B'), mods | M::CTRL),
31 '\x03' => E(K::Char('C'), mods | M::CTRL),
32 '\x04' => E(K::Char('D'), mods | M::CTRL),
33 '\x05' => E(K::Char('E'), mods | M::CTRL),
34 '\x06' => E(K::Char('F'), mods | M::CTRL),
35 '\x07' => E(K::Char('G'), mods | M::CTRL), #[cfg(unix)]
37 '\x08' => E(K::Backspace, mods), #[cfg(windows)]
39 '\x08' => E(K::Char('H'), mods | M::CTRL),
40 #[cfg(unix)]
41 '\x09' => {
42 if mods.contains(M::SHIFT) {
44 mods.remove(M::SHIFT);
45 E(K::BackTab, mods)
46 } else {
47 E(K::Tab, mods)
48 }
49 }
50 #[cfg(windows)]
51 '\x09' => E(K::Char('I'), mods | M::CTRL),
52 '\x0a' => E(K::Char('J'), mods | M::CTRL), '\x0b' => E(K::Char('K'), mods | M::CTRL),
54 '\x0c' => E(K::Char('L'), mods | M::CTRL),
55 #[cfg(unix)]
56 '\x0d' => E(K::Enter, mods), #[cfg(windows)]
58 '\x0d' => E(K::Char('M'), mods | M::CTRL),
59 '\x0e' => E(K::Char('N'), mods | M::CTRL),
60 '\x0f' => E(K::Char('O'), mods | M::CTRL),
61 '\x10' => E(K::Char('P'), mods | M::CTRL),
62 '\x11' => E(K::Char('Q'), mods | M::CTRL),
63 '\x12' => E(K::Char('R'), mods | M::CTRL),
64 '\x13' => E(K::Char('S'), mods | M::CTRL),
65 '\x14' => E(K::Char('T'), mods | M::CTRL),
66 '\x15' => E(K::Char('U'), mods | M::CTRL),
67 '\x16' => E(K::Char('V'), mods | M::CTRL),
68 '\x17' => E(K::Char('W'), mods | M::CTRL),
69 '\x18' => E(K::Char('X'), mods | M::CTRL),
70 '\x19' => E(K::Char('Y'), mods | M::CTRL),
71 '\x1a' => E(K::Char('Z'), mods | M::CTRL),
72 '\x1b' => E(K::Esc, mods), '\x1c' => E(K::Char('\\'), mods | M::CTRL),
74 '\x1d' => E(K::Char(']'), mods | M::CTRL),
75 '\x1e' => E(K::Char('^'), mods | M::CTRL),
76 '\x1f' => E(K::Char('_'), mods | M::CTRL),
77 '\x7f' => E(K::Backspace, mods), '\u{9b}' => E(K::Esc, mods | M::SHIFT),
79 _ => E(K::Null, mods),
80 }
81 }
82
83 #[must_use]
85 pub fn ctrl(c: char) -> Self {
86 Self::new(c, Modifiers::CTRL)
87 }
88
89 #[must_use]
91 pub fn alt(c: char) -> Self {
92 Self::new(c, Modifiers::ALT)
93 }
94
95 #[must_use]
99 pub fn normalize(e: Self) -> Self {
100 use {KeyCode as K, KeyEvent as E, Modifiers as M};
101
102 match e {
103 E(K::Char(c), m) if c.is_ascii_control() => Self::new(c, m),
104 E(K::Char(c), m) if c.is_ascii_lowercase() && m.contains(M::CTRL) => {
105 E(K::Char(c.to_ascii_uppercase()), m)
106 }
107 E(K::Char(c), m) if c.is_ascii_uppercase() && m.contains(M::SHIFT) => {
108 E(K::Char(c), m ^ M::SHIFT)
109 }
110 E(K::Tab, m) if m.contains(M::SHIFT) => E(K::BackTab, m ^ M::SHIFT),
111 _ => e,
112 }
113 }
114}
115
116impl From<char> for KeyEvent {
117 fn from(c: char) -> Self {
118 Self::new(c, Modifiers::NONE)
119 }
120}
121
122#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
124#[non_exhaustive]
125pub enum KeyCode {
126 UnknownEscSeq,
128 Backspace,
130 BackTab,
132 BracketedPasteStart,
134 BracketedPasteEnd,
136 Char(char),
138 Delete,
140 Down,
142 End,
144 Enter,
146 Esc,
148 F(u8),
150 Home,
152 Insert,
154 Left,
156 Null,
158 PageDown,
160 PageUp,
162 Right,
164 Tab,
166 Up,
168}
169
170bitflags::bitflags! {
171 #[derive(Clone, Copy, Debug, Eq, Hash, PartialEq)]
173 pub struct Modifiers: u8 {
174 const CTRL = 1<<3;
176 const ALT = 1<<2;
178 const SHIFT = 1<<1;
180
181 const NONE = 0;
183 const CTRL_SHIFT = Self::CTRL.bits() | Self::SHIFT.bits();
185 const ALT_SHIFT = Self::ALT.bits() | Self::SHIFT.bits();
187 const CTRL_ALT = Self::CTRL.bits() | Self::ALT.bits();
189 const CTRL_ALT_SHIFT = Self::CTRL.bits() | Self::ALT.bits() | Self::SHIFT.bits();
191 }
192}
193
194#[cfg(test)]
195mod tests {
196 use super::{KeyCode as K, KeyEvent as E, Modifiers as M};
197
198 #[test]
199 fn new() {
200 assert_eq!(E::ESC, E::new('\x1b', M::NONE));
201 }
202
203 #[test]
204 #[cfg(unix)]
205 fn from() {
206 assert_eq!(E(K::Tab, M::NONE), E::from('\t'));
207 }
208
209 #[test]
210 #[cfg(windows)]
211 fn from() {
212 assert_eq!(E(K::Char('I'), M::CTRL), E::from('\t'));
213 }
214
215 #[test]
216 fn normalize() {
217 assert_eq!(E::ctrl('A'), E::normalize(E(K::Char('\x01'), M::NONE)));
218 assert_eq!(E::ctrl('A'), E::normalize(E::ctrl('a')));
219 assert_eq!(E::from('A'), E::normalize(E(K::Char('A'), M::SHIFT)));
220 assert_eq!(E(K::BackTab, M::NONE), E::normalize(E(K::Tab, M::SHIFT)));
221 }
222}