aboutsummaryrefslogtreecommitdiff
path: root/helix-view/src/info.rs
blob: 70e934cd65e435bbb7867419b3def6bb9ba9cfb1 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
use crate::input::KeyEvent;
use helix_core::unicode::width::UnicodeWidthStr;
use std::fmt::Write;

#[derive(Debug)]
/// Info box used in editor. Rendering logic will be in other crate.
pub struct Info {
    /// Title shown at top.
    pub title: String,
    /// Text body, should contain newlines.
    pub text: String,
    /// Body width.
    pub width: u16,
    /// Body height.
    pub height: u16,
}

impl Info {
    // body is a BTreeMap instead of a HashMap because keymaps are represented
    // with nested hashmaps with no ordering, and each invocation of infobox would
    // show different orders of items
    pub fn key(title: &str, body: Vec<(&str, Vec<KeyEvent>)>) -> Info {
        let (lpad, mpad, rpad) = (1, 2, 1);
        let keymaps_width: u16 = body
            .iter()
            .map(|r| r.1.iter().map(|e| e.width() as u16 + 2).sum::<u16>() - 2)
            .max()
            .unwrap();
        let mut text = String::new();
        let mut width = 0;
        let height = body.len() as u16;
        for (desc, keyevents) in body {
            let keyevent = keyevents[0];
            let mut left = keymaps_width - keyevent.width() as u16;
            for _ in 0..lpad {
                text.push(' ');
            }
            write!(text, "{}", keyevent).ok();
            for keyevent in &keyevents[1..] {
                write!(text, ", {}", keyevent).ok();
                left -= 2 + keyevent.width() as u16;
            }
            for _ in 0..left + mpad {
                text.push(' ');
            }
            let desc = desc.trim();
            let w = lpad + keymaps_width + mpad + (desc.width() as u16) + rpad;
            if w > width {
                width = w;
            }
            writeln!(text, "{}", desc).ok();
        }
        Info {
            title: title.to_string(),
            text,
            width,
            height,
        }
    }
}