Skip to main content

tess/
keys.rs

1//! Custom keybindings loaded from `~/.config/tess/keys.toml`.
2//!
3//! Schema:
4//! ```toml
5//! [bindings]
6//! "j" = "scroll-down"
7//! "f1" = "!git status"
8//! ```
9
10use std::collections::HashMap;
11
12use crossterm::event::{KeyCode, KeyEvent, KeyModifiers};
13use serde::Deserialize;
14
15use crate::input::Command;
16
17#[derive(Debug, Clone)]
18pub enum BindingTarget {
19    Command(Command),
20    Shell(String),
21}
22
23#[derive(Debug, Clone)]
24pub struct KeyMap {
25    map: HashMap<KeyEvent, BindingTarget>,
26}
27
28impl KeyMap {
29    pub fn empty() -> Self {
30        Self { map: HashMap::new() }
31    }
32
33    /// Load keys from the global config dir (`/etc/tess/` or
34    /// `$TESS_GLOBAL_CONFIG_DIR`) and from `~/.config/tess/keys.toml`,
35    /// merging per individual binding key with local winning.
36    ///
37    /// Global parse errors warn on stderr and the global layer is treated as
38    /// empty; local parse errors fail startup.
39    pub fn load_layered() -> Result<Self, String> {
40        let mut bindings: HashMap<String, String> = HashMap::new();
41
42        // Global layer.
43        if let Some(dir) = crate::config_path::global_config_dir() {
44            let path = dir.join("keys.toml");
45            if path.exists() {
46                match std::fs::read_to_string(&path) {
47                    Ok(text) => {
48                        match toml::from_str::<KeysConfig>(&text) {
49                            Ok(cfg) => {
50                                for (k, v) in cfg.bindings {
51                                    bindings.insert(k, v);
52                                }
53                            }
54                            Err(e) => eprintln!(
55                                "tess: warning: keys.toml: {}: {e}; ignoring global config",
56                                path.display()
57                            ),
58                        }
59                    }
60                    Err(e) => eprintln!(
61                        "tess: warning: keys.toml: {}: {e}; ignoring global config",
62                        path.display()
63                    ),
64                }
65            }
66        }
67
68        // Local layer.
69        if let Some(dir) = crate::config_path::user_config_dir() {
70            let path = dir.join("keys.toml");
71            if path.exists() {
72                let text = std::fs::read_to_string(&path)
73                    .map_err(|e| format!("keys.toml: reading {}: {e}", path.display()))?;
74                let cfg: KeysConfig = toml::from_str(&text)
75                    .map_err(|e| format!("keys.toml: parsing {}: {e}", path.display()))?;
76                for (k, v) in cfg.bindings {
77                    bindings.insert(k, v);
78                }
79            }
80        }
81
82        // Build the final KeyMap from the merged bindings table.
83        let mut map = HashMap::with_capacity(bindings.len());
84        for (key_spec, action) in bindings {
85            let key = parse_key_spec(&key_spec)
86                .map_err(|e| format!("keys.toml: '{key_spec}': {e}"))?;
87            reject_forbidden_key(&key, &key_spec)
88                .map_err(|e| format!("keys.toml: {e}"))?;
89            let target = parse_action(&action)
90                .map_err(|e| format!("keys.toml: '{key_spec}': {e}"))?;
91            map.insert(key, target);
92        }
93        Ok(Self { map })
94    }
95
96    pub fn load_from_str(toml_text: &str) -> Result<Self, String> {
97        let cfg: KeysConfig = toml::from_str(toml_text)
98            .map_err(|e| format!("parsing: {e}"))?;
99        let mut map = HashMap::with_capacity(cfg.bindings.len());
100        for (key_spec, action) in cfg.bindings {
101            let key = parse_key_spec(&key_spec)
102                .map_err(|e| format!("'{key_spec}': {e}"))?;
103            reject_forbidden_key(&key, &key_spec)?;
104            let target = parse_action(&action)
105                .map_err(|e| format!("'{key_spec}': {e}"))?;
106            map.insert(key, target);
107        }
108        Ok(Self { map })
109    }
110
111    pub fn lookup(&self, key: &KeyEvent) -> Option<&BindingTarget> {
112        self.map.get(key)
113    }
114
115    pub fn is_empty(&self) -> bool {
116        self.map.is_empty()
117    }
118
119    /// Group user bindings by the kebab-case command name they target.
120    /// Shell-target bindings are excluded. Used by the help overlay to
121    /// replace default key displays with the user's chosen keys.
122    pub fn user_keys_by_command_name(&self) -> std::collections::HashMap<String, Vec<String>> {
123        let mut out: std::collections::HashMap<String, Vec<String>> =
124            std::collections::HashMap::new();
125        for (key, target) in &self.map {
126            let BindingTarget::Command(cmd) = target else { continue };
127            let Some(name) = command_to_kebab(cmd) else { continue };
128            out.entry(name.to_string())
129                .or_default()
130                .push(format_key_event(*key));
131        }
132        out
133    }
134}
135
136#[derive(Debug, Deserialize, Default)]
137struct KeysConfig {
138    #[serde(default)]
139    bindings: HashMap<String, String>,
140}
141
142/// Parse a key spec string into a `KeyEvent`.
143fn parse_key_spec(spec: &str) -> Result<KeyEvent, String> {
144    let lower = spec.to_lowercase();
145    let mut parts: Vec<&str> = lower.split('-').collect();
146    if parts.is_empty() {
147        return Err("empty key spec".to_string());
148    }
149    let key_part = parts.pop().unwrap();
150    let mut modifiers = KeyModifiers::NONE;
151    for m in &parts {
152        if m.is_empty() {
153            // Handle "ctrl--" (ctrl + dash). The trailing "-" became "".
154            continue;
155        }
156        match *m {
157            "ctrl" => modifiers |= KeyModifiers::CONTROL,
158            "alt" => modifiers |= KeyModifiers::ALT,
159            "shift" => modifiers |= KeyModifiers::SHIFT,
160            other => return Err(format!("unknown modifier '{other}'")),
161        }
162    }
163    let code = match key_part {
164        "esc" => KeyCode::Esc,
165        "enter" => KeyCode::Enter,
166        "tab" => KeyCode::Tab,
167        "backspace" => KeyCode::Backspace,
168        "space" => KeyCode::Char(' '),
169        "up" => KeyCode::Up,
170        "down" => KeyCode::Down,
171        "left" => KeyCode::Left,
172        "right" => KeyCode::Right,
173        "pgup" => KeyCode::PageUp,
174        "pgdn" => KeyCode::PageDown,
175        "home" => KeyCode::Home,
176        "end" => KeyCode::End,
177        "" => {
178            // The trailing piece was empty, meaning the key is literal "-"
179            // and the modifiers consumed everything else (e.g. "ctrl--").
180            KeyCode::Char('-')
181        }
182        s if s.starts_with('f') && s.len() > 1 => {
183            let n: u8 = s[1..].parse()
184                .map_err(|_| format!("unknown key '{s}'"))?;
185            KeyCode::F(n)
186        }
187        s if s.chars().count() == 1 => {
188            // For a BARE letter with no modifiers, an uppercase letter
189            // promotes to shift-prefix semantics ("J" == "shift-j").
190            // When modifiers are already present (e.g. "ctrl-J"), the
191            // user's intent is the literal letter — don't add SHIFT.
192            let original_char = spec.chars().last().unwrap();
193            if original_char.is_ascii_uppercase() && modifiers == KeyModifiers::NONE {
194                modifiers |= KeyModifiers::SHIFT;
195                KeyCode::Char(original_char.to_ascii_lowercase())
196            } else {
197                // Either lowercase letter, or uppercase with an explicit
198                // modifier — lowercase the char either way for consistency.
199                KeyCode::Char(original_char.to_ascii_lowercase())
200            }
201        }
202        other => return Err(format!("unknown key '{other}'")),
203    };
204    Ok(KeyEvent::new(code, modifiers))
205}
206
207fn reject_forbidden_key(key: &KeyEvent, original_spec: &str) -> Result<(), String> {
208    let forbidden = match (&key.code, key.modifiers) {
209        (KeyCode::Char('m'), KeyModifiers::NONE) => true,
210        (KeyCode::Char('\''), KeyModifiers::NONE) => true,
211        (KeyCode::Char('-'), KeyModifiers::NONE) => true,
212        (KeyCode::Char('x'), KeyModifiers::CONTROL) => true,
213        (KeyCode::Char(c), KeyModifiers::NONE) if c.is_ascii_digit() => true,
214        _ => false,
215    };
216    if forbidden {
217        return Err(format!(
218            "'{original_spec}' is part of a multi-key sequence and cannot be rebound"
219        ));
220    }
221    Ok(())
222}
223
224fn parse_action(action: &str) -> Result<BindingTarget, String> {
225    if let Some(shell_cmd) = action.strip_prefix('!') {
226        if shell_cmd.is_empty() {
227            return Err("shell binding requires a command after '!'".to_string());
228        }
229        return Ok(BindingTarget::Shell(shell_cmd.to_string()));
230    }
231    let cmd = command_from_kebab(action)
232        .ok_or_else(|| format!("unknown command '{action}'"))?;
233    Ok(BindingTarget::Command(cmd))
234}
235
236/// Map a kebab-case command name to the corresponding `Command` enum variant.
237fn command_from_kebab(name: &str) -> Option<Command> {
238    match name {
239        "scroll-down" => Some(Command::ScrollLines(1)),
240        "scroll-up" => Some(Command::ScrollLines(-1)),
241        "scroll-logical-down" => Some(Command::ScrollLogicalLines(1)),
242        "scroll-logical-up" => Some(Command::ScrollLogicalLines(-1)),
243        "page-down" => Some(Command::PageDown),
244        "page-up" => Some(Command::PageUp),
245        "half-page-down" => Some(Command::HalfPageDown),
246        "half-page-up" => Some(Command::HalfPageUp),
247        "quit" => Some(Command::Quit),
248        "refresh" => Some(Command::Refresh),
249        "reload" => Some(Command::Reload),
250        "toggle-line-numbers" => Some(Command::ToggleLineNumbers),
251        "toggle-chop" => Some(Command::ToggleChop),
252        "toggle-follow" => Some(Command::ToggleFollow),
253        "toggle-prettify" => Some(Command::TogglePrettify),
254        "search-forward" => Some(Command::SearchForward),
255        "search-backward" => Some(Command::SearchBackward),
256        "next-match" => Some(Command::NextMatch),
257        "previous-match" => Some(Command::PreviousMatch),
258        "option-prefix" => Some(Command::OptionPrefix),
259        "goto-line" => Some(Command::GotoLine),
260        "goto-record" => Some(Command::GotoRecord),
261        "goto-percent" => Some(Command::GotoPercent),
262        "mark-set" => Some(Command::MarkSet),
263        "mark-jump" => Some(Command::MarkJump),
264        "ctrl-x-prefix" => Some(Command::CtrlXPrefix),
265        "jump-previous" => Some(Command::JumpPrevious),
266        "shell-escape" => Some(Command::ShellEscape),
267        "cancel" => Some(Command::Cancel),
268        "hscroll-left" => Some(Command::HScrollLeft),
269        "hscroll-right" => Some(Command::HScrollRight),
270        "hscroll-left-step" => Some(Command::HScrollLeftStep),
271        "hscroll-right-step" => Some(Command::HScrollRightStep),
272        "clipboard-yank-line" => Some(Command::YankLine),
273        "anim-pause" => Some(Command::AnimPause),
274        "anim-step-forward" => Some(Command::AnimStepForward),
275        "anim-step-back" => Some(Command::AnimStepBack),
276        "anim-restart" => Some(Command::AnimRestart),
277        _ => None,
278    }
279}
280
281/// Inverse of `command_from_kebab` — covers the same command set.
282fn command_to_kebab(cmd: &Command) -> Option<&'static str> {
283    match cmd {
284        Command::ScrollLines(1) => Some("scroll-down"),
285        Command::ScrollLines(-1) => Some("scroll-up"),
286        Command::ScrollLogicalLines(1) => Some("scroll-logical-down"),
287        Command::ScrollLogicalLines(-1) => Some("scroll-logical-up"),
288        Command::PageDown => Some("page-down"),
289        Command::PageUp => Some("page-up"),
290        Command::HalfPageDown => Some("half-page-down"),
291        Command::HalfPageUp => Some("half-page-up"),
292        Command::Quit => Some("quit"),
293        Command::Refresh => Some("refresh"),
294        Command::Reload => Some("reload"),
295        Command::ToggleLineNumbers => Some("toggle-line-numbers"),
296        Command::ToggleChop => Some("toggle-chop"),
297        Command::ToggleFollow => Some("toggle-follow"),
298        Command::TogglePrettify => Some("toggle-prettify"),
299        Command::SearchForward => Some("search-forward"),
300        Command::SearchBackward => Some("search-backward"),
301        Command::NextMatch => Some("next-match"),
302        Command::PreviousMatch => Some("previous-match"),
303        Command::OptionPrefix => Some("option-prefix"),
304        Command::GotoLine => Some("goto-line"),
305        Command::GotoRecord => Some("goto-record"),
306        Command::GotoPercent => Some("goto-percent"),
307        Command::MarkSet => Some("mark-set"),
308        Command::MarkJump => Some("mark-jump"),
309        Command::CtrlXPrefix => Some("ctrl-x-prefix"),
310        Command::JumpPrevious => Some("jump-previous"),
311        Command::ShellEscape => Some("shell-escape"),
312        Command::Cancel => Some("cancel"),
313        Command::HScrollLeft => Some("hscroll-left"),
314        Command::HScrollRight => Some("hscroll-right"),
315        Command::HScrollLeftStep => Some("hscroll-left-step"),
316        Command::HScrollRightStep => Some("hscroll-right-step"),
317        Command::YankLine => Some("clipboard-yank-line"),
318        Command::AnimPause => Some("anim-pause"),
319        Command::AnimStepForward => Some("anim-step-forward"),
320        Command::AnimStepBack => Some("anim-step-back"),
321        Command::AnimRestart => Some("anim-restart"),
322        _ => None,
323    }
324}
325
326fn format_key_event(ke: KeyEvent) -> String {
327    let ctrl  = ke.modifiers.contains(KeyModifiers::CONTROL);
328    let alt   = ke.modifiers.contains(KeyModifiers::ALT);
329    let shift = ke.modifiers.contains(KeyModifiers::SHIFT);
330
331    // Convention: SHIFT-alone on an ASCII letter displays as the uppercase
332    // letter with no "Shift-" prefix (matches KEY_REGISTRY's `"J"` form).
333    if shift && !ctrl && !alt {
334        if let KeyCode::Char(c) = ke.code {
335            if c.is_ascii_alphabetic() {
336                return c.to_ascii_uppercase().to_string();
337            }
338        }
339    }
340
341    let mut parts: Vec<&'static str> = Vec::new();
342    if ctrl  { parts.push("Ctrl"); }
343    if alt   { parts.push("Alt"); }
344    if shift { parts.push("Shift"); }
345
346    let key = match ke.code {
347        KeyCode::Char(' ') => "Space".to_string(),
348        KeyCode::Char(c)   => c.to_string(),
349        KeyCode::F(n)      => format!("F{n}"),
350        KeyCode::Esc       => "Esc".into(),
351        KeyCode::Enter     => "Enter".into(),
352        KeyCode::Tab       => "Tab".into(),
353        KeyCode::Backspace => "Backspace".into(),
354        KeyCode::Up        => "\u{2191}".into(),
355        KeyCode::Down      => "\u{2193}".into(),
356        KeyCode::Left      => "\u{2190}".into(),
357        KeyCode::Right     => "\u{2192}".into(),
358        KeyCode::Home      => "Home".into(),
359        KeyCode::End       => "End".into(),
360        KeyCode::PageUp    => "PgUp".into(),
361        KeyCode::PageDown  => "PgDn".into(),
362        other              => format!("{other:?}"),
363    };
364    if parts.is_empty() { key } else { format!("{}-{}", parts.join("-"), key) }
365}
366
367#[cfg(test)]
368mod tests {
369    use super::*;
370    use std::sync::Mutex;
371
372    static HOME_LOCK: Mutex<()> = Mutex::new(());
373
374    #[test]
375    fn parse_empty_file_returns_empty_map() {
376        let m = KeyMap::load_from_str("").unwrap();
377        assert!(m.is_empty());
378    }
379
380    #[test]
381    fn parse_single_binding() {
382        let toml = r#"
383[bindings]
384"j" = "scroll-down"
385"#;
386        let m = KeyMap::load_from_str(toml).unwrap();
387        let key = KeyEvent::new(KeyCode::Char('j'), KeyModifiers::NONE);
388        assert!(matches!(m.lookup(&key), Some(BindingTarget::Command(Command::ScrollLines(1)))));
389    }
390
391    #[test]
392    fn parse_named_special_key() {
393        let toml = r#"
394[bindings]
395"f1" = "toggle-line-numbers"
396"esc" = "cancel"
397"#;
398        let m = KeyMap::load_from_str(toml).unwrap();
399        let f1 = KeyEvent::new(KeyCode::F(1), KeyModifiers::NONE);
400        let esc = KeyEvent::new(KeyCode::Esc, KeyModifiers::NONE);
401        assert!(matches!(m.lookup(&f1), Some(BindingTarget::Command(Command::ToggleLineNumbers))));
402        assert!(matches!(m.lookup(&esc), Some(BindingTarget::Command(Command::Cancel))));
403    }
404
405    #[test]
406    fn parse_modifier_combinations() {
407        let toml = r#"
408[bindings]
409"ctrl-r" = "reload"
410"shift-tab" = "scroll-logical-up"
411"#;
412        let m = KeyMap::load_from_str(toml).unwrap();
413        let ctrl_r = KeyEvent::new(KeyCode::Char('r'), KeyModifiers::CONTROL);
414        let shift_tab = KeyEvent::new(KeyCode::Tab, KeyModifiers::SHIFT);
415        assert!(matches!(m.lookup(&ctrl_r), Some(BindingTarget::Command(Command::Reload))));
416        assert!(matches!(m.lookup(&shift_tab), Some(BindingTarget::Command(Command::ScrollLogicalLines(-1)))));
417    }
418
419    #[test]
420    fn case_letter_resolves_to_shift_prefix() {
421        let toml = r#"
422[bindings]
423"J" = "scroll-logical-down"
424"#;
425        let m = KeyMap::load_from_str(toml).unwrap();
426        let shift_j = KeyEvent::new(KeyCode::Char('j'), KeyModifiers::SHIFT);
427        assert!(matches!(m.lookup(&shift_j), Some(BindingTarget::Command(Command::ScrollLogicalLines(1)))));
428    }
429
430    #[test]
431    fn forbidden_keys_error_at_parse() {
432        for key in &["m", "'", "-", "ctrl-x", "0", "5", "9"] {
433            let toml = format!(r#"
434[bindings]
435"{key}" = "quit"
436"#);
437            let err = KeyMap::load_from_str(&toml).unwrap_err();
438            assert!(err.contains("multi-key sequence"),
439                    "key '{key}' should be forbidden: {err}");
440        }
441    }
442
443    #[test]
444    fn unknown_command_name_errors() {
445        let toml = r#"
446[bindings]
447"j" = "definitely-not-a-real-command"
448"#;
449        let err = KeyMap::load_from_str(toml).unwrap_err();
450        assert!(err.contains("unknown command"));
451    }
452
453    #[test]
454    fn empty_shell_binding_errors() {
455        let toml = r#"
456[bindings]
457"f1" = "!"
458"#;
459        let err = KeyMap::load_from_str(toml).unwrap_err();
460        assert!(err.contains("requires a command"));
461    }
462
463    #[test]
464    fn parse_inline_shell_binding() {
465        let toml = r#"
466[bindings]
467"f2" = "!git status"
468"#;
469        let m = KeyMap::load_from_str(toml).unwrap();
470        let f2 = KeyEvent::new(KeyCode::F(2), KeyModifiers::NONE);
471        match m.lookup(&f2) {
472            Some(BindingTarget::Shell(cmd)) => assert_eq!(cmd, "git status"),
473            other => panic!("expected Shell, got {:?}", other),
474        }
475    }
476
477    #[test]
478    fn lookup_returns_none_for_unbound_key() {
479        let toml = r#"
480[bindings]
481"j" = "scroll-down"
482"#;
483        let m = KeyMap::load_from_str(toml).unwrap();
484        let other = KeyEvent::new(KeyCode::Char('k'), KeyModifiers::NONE);
485        assert!(m.lookup(&other).is_none());
486    }
487
488    #[test]
489    fn ctrl_uppercase_letter_does_not_add_shift() {
490        // "ctrl-J" should be Ctrl + 'j', NOT Ctrl + Shift + 'j'.
491        let toml = r#"
492[bindings]
493"ctrl-J" = "reload"
494"#;
495        let m = KeyMap::load_from_str(toml).unwrap();
496        let ctrl_j = KeyEvent::new(KeyCode::Char('j'), KeyModifiers::CONTROL);
497        assert!(matches!(m.lookup(&ctrl_j), Some(BindingTarget::Command(Command::Reload))),
498                "ctrl-J should resolve to Ctrl+j without Shift");
499        let ctrl_shift_j = KeyEvent::new(KeyCode::Char('j'), KeyModifiers::CONTROL | KeyModifiers::SHIFT);
500        assert!(m.lookup(&ctrl_shift_j).is_none(),
501                "ctrl-J should NOT also match Ctrl+Shift+j");
502    }
503
504    #[test]
505    fn user_remaps_by_command_name_groups_keys() {
506        let toml = r#"
507[bindings]
508"f3" = "scroll-down"
509"f4" = "scroll-down"
510"f5" = "quit"
511"#;
512        let m = KeyMap::load_from_str(toml).unwrap();
513        let groups = m.user_keys_by_command_name();
514        let mut down = groups.get("scroll-down").cloned().unwrap_or_default();
515        down.sort();
516        assert_eq!(down, vec!["F3".to_string(), "F4".to_string()]);
517        assert_eq!(groups.get("quit").cloned().unwrap_or_default(), vec!["F5".to_string()]);
518    }
519
520    #[test]
521    fn dash_with_modifier_is_a_real_key() {
522        // "ctrl--" should resolve to Ctrl + '-' (a valid bind).
523        // (Bare "-" is forbidden by reject_forbidden_key, but Ctrl-- isn't.)
524        let toml = r#"
525[bindings]
526"ctrl--" = "refresh"
527"#;
528        let m = KeyMap::load_from_str(toml).unwrap();
529        let ctrl_dash = KeyEvent::new(KeyCode::Char('-'), KeyModifiers::CONTROL);
530        assert!(matches!(m.lookup(&ctrl_dash), Some(BindingTarget::Command(Command::Refresh))));
531    }
532
533    #[test]
534    fn format_key_event_renders_modifier_combos() {
535        // Ctrl alone
536        assert_eq!(
537            format_key_event(KeyEvent::new(KeyCode::Char('r'), KeyModifiers::CONTROL)),
538            "Ctrl-r",
539        );
540        // Shift alone on a letter → uppercase, no prefix
541        assert_eq!(
542            format_key_event(KeyEvent::new(KeyCode::Char('j'), KeyModifiers::SHIFT)),
543            "J",
544        );
545        // Shift alone on a non-letter (e.g. Tab) → "Shift-Tab"
546        assert_eq!(
547            format_key_event(KeyEvent::new(KeyCode::Tab, KeyModifiers::SHIFT)),
548            "Shift-Tab",
549        );
550        // Plain lowercase letter → "j" (no uppercasing)
551        assert_eq!(
552            format_key_event(KeyEvent::new(KeyCode::Char('j'), KeyModifiers::NONE)),
553            "j",
554        );
555        // Function key
556        assert_eq!(
557            format_key_event(KeyEvent::new(KeyCode::F(3), KeyModifiers::NONE)),
558            "F3",
559        );
560        // Ctrl+Shift+letter — composed prefix, lowercase stored char
561        assert_eq!(
562            format_key_event(KeyEvent::new(KeyCode::Char('x'), KeyModifiers::CONTROL | KeyModifiers::SHIFT)),
563            "Ctrl-Shift-x",
564        );
565    }
566
567    #[test]
568    fn hscroll_names_resolve_to_commands() {
569        let toml = r#"
570[bindings]
571"f6" = "hscroll-right"
572"#;
573        let m = KeyMap::load_from_str(toml).unwrap();
574        let f6 = KeyEvent::new(KeyCode::F(6), KeyModifiers::NONE);
575        assert!(matches!(m.lookup(&f6), Some(BindingTarget::Command(Command::HScrollRight))));
576
577        assert!(matches!(command_from_kebab("hscroll-left"), Some(Command::HScrollLeft)));
578        assert!(matches!(command_from_kebab("hscroll-left-step"), Some(Command::HScrollLeftStep)));
579        assert!(matches!(command_from_kebab("hscroll-right-step"), Some(Command::HScrollRightStep)));
580    }
581
582    #[test]
583    fn command_kebab_round_trip() {
584        // Every name in command_from_kebab must round-trip through command_to_kebab.
585        let names = [
586            "scroll-down", "scroll-up", "scroll-logical-down", "scroll-logical-up",
587            "page-down", "page-up", "half-page-down", "half-page-up",
588            "quit", "refresh", "reload",
589            "toggle-line-numbers", "toggle-chop", "toggle-follow", "toggle-prettify",
590            "search-forward", "search-backward", "next-match", "previous-match",
591            "option-prefix", "goto-line", "goto-record", "goto-percent",
592            "mark-set", "mark-jump", "ctrl-x-prefix", "jump-previous",
593            "shell-escape", "cancel",
594            "hscroll-left", "hscroll-right", "hscroll-left-step", "hscroll-right-step",
595            "clipboard-yank-line",
596            "anim-pause", "anim-step-forward", "anim-step-back", "anim-restart",
597        ];
598        for name in &names {
599            let cmd = command_from_kebab(name).expect(&format!("from_kebab failed for {name}"));
600            let back = command_to_kebab(&cmd).expect(&format!("to_kebab failed for {name}"));
601            assert_eq!(back, *name, "round-trip mismatch for {name}");
602        }
603    }
604
605    #[test]
606    fn shell_bindings_are_excluded_from_user_keys() {
607        let toml = r#"
608[bindings]
609"f2" = "!git status"
610"f3" = "scroll-down"
611"#;
612        let m = KeyMap::load_from_str(toml).unwrap();
613        let groups = m.user_keys_by_command_name();
614        assert!(!groups.values().any(|v| v.contains(&"F2".to_string())),
615                "shell-bound F2 should not appear: {groups:?}");
616        assert_eq!(groups.get("scroll-down").cloned().unwrap_or_default(), vec!["F3".to_string()]);
617    }
618
619    #[test]
620    fn layered_keys_local_overrides_global_per_binding() {
621        let _guard = HOME_LOCK.lock().unwrap();
622        let prev_home = std::env::var_os("HOME");
623        let prev_global = std::env::var_os("TESS_GLOBAL_CONFIG_DIR");
624
625        let home = tempfile::tempdir().unwrap();
626        let global = tempfile::tempdir().unwrap();
627
628        std::env::set_var("HOME", home.path());
629        std::env::set_var("TESS_GLOBAL_CONFIG_DIR", global.path());
630
631        std::fs::write(
632            global.path().join("keys.toml"),
633            r#"
634[bindings]
635"j" = "scroll-down"
636"k" = "scroll-up"
637"#,
638        )
639        .unwrap();
640
641        let cfg_dir = home.path().join(".config").join("tess");
642        std::fs::create_dir_all(&cfg_dir).unwrap();
643        std::fs::write(
644            cfg_dir.join("keys.toml"),
645            r#"
646[bindings]
647"j" = "page-down"
648"#,
649        )
650        .unwrap();
651
652        let km = KeyMap::load_layered().unwrap();
653
654        // j: local wins (page-down)
655        let j = crossterm::event::KeyEvent::new(
656            crossterm::event::KeyCode::Char('j'),
657            crossterm::event::KeyModifiers::NONE,
658        );
659        match km.lookup(&j) {
660            Some(BindingTarget::Command(cmd)) => {
661                let dbg = format!("{cmd:?}");
662                assert!(dbg.to_lowercase().contains("page"), "got: {dbg}");
663            }
664            other => panic!("expected Command(PageDown), got {other:?}"),
665        }
666
667        // k: global survives (scroll-up)
668        let k = crossterm::event::KeyEvent::new(
669            crossterm::event::KeyCode::Char('k'),
670            crossterm::event::KeyModifiers::NONE,
671        );
672        match km.lookup(&k) {
673            Some(BindingTarget::Command(cmd)) => {
674                assert!(matches!(cmd, Command::ScrollLines(n) if *n < 0), "got: {cmd:?}");
675            }
676            other => panic!("expected Command(ScrollLines(-1)), got {other:?}"),
677        }
678
679        match prev_home {
680            Some(v) => std::env::set_var("HOME", v),
681            None => std::env::remove_var("HOME"),
682        }
683        match prev_global {
684            Some(v) => std::env::set_var("TESS_GLOBAL_CONFIG_DIR", v),
685            None => std::env::remove_var("TESS_GLOBAL_CONFIG_DIR"),
686        }
687    }
688
689    #[test]
690    fn layered_keys_warns_on_bad_global() {
691        let _guard = HOME_LOCK.lock().unwrap();
692        let prev_home = std::env::var_os("HOME");
693        let prev_global = std::env::var_os("TESS_GLOBAL_CONFIG_DIR");
694
695        let home = tempfile::tempdir().unwrap();
696        let global = tempfile::tempdir().unwrap();
697
698        std::env::set_var("HOME", home.path());
699        std::env::set_var("TESS_GLOBAL_CONFIG_DIR", global.path());
700
701        std::fs::write(
702            global.path().join("keys.toml"),
703            "= = not valid",
704        )
705        .unwrap();
706
707        // Local missing — should still succeed.
708        let km = KeyMap::load_layered().unwrap();
709        assert!(km.is_empty());
710
711        match prev_home {
712            Some(v) => std::env::set_var("HOME", v),
713            None => std::env::remove_var("HOME"),
714        }
715        match prev_global {
716            Some(v) => std::env::set_var("TESS_GLOBAL_CONFIG_DIR", v),
717            None => std::env::remove_var("TESS_GLOBAL_CONFIG_DIR"),
718        }
719    }
720}