mirror of
https://github.com/helix-editor/helix.git
synced 2025-01-19 21:47:07 +04:00
Refactor infobox rendering and parsing (#579)
This commit is contained in:
parent
27616153bc
commit
14c08e855f
@ -145,7 +145,7 @@ fn from(node: KeyTrieNode) -> Self {
|
|||||||
.map(|(desc, keys)| (desc.strip_prefix(&prefix).unwrap(), keys))
|
.map(|(desc, keys)| (desc.strip_prefix(&prefix).unwrap(), keys))
|
||||||
.collect();
|
.collect();
|
||||||
}
|
}
|
||||||
Info::key(node.name(), body)
|
Info::new(node.name(), body)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
use crate::compositor::{Component, Context};
|
use crate::compositor::{Component, Context};
|
||||||
use helix_view::graphics::Rect;
|
use helix_view::graphics::{Margin, Rect};
|
||||||
use helix_view::info::Info;
|
use helix_view::info::Info;
|
||||||
use tui::buffer::Buffer as Surface;
|
use tui::buffer::Buffer as Surface;
|
||||||
use tui::widgets::{Block, Borders, Widget};
|
use tui::widgets::{Block, Borders, Paragraph, Widget};
|
||||||
|
|
||||||
impl Component for Info {
|
impl Component for Info {
|
||||||
fn render(&mut self, viewport: Rect, surface: &mut Surface, cx: &mut Context) {
|
fn render(&mut self, viewport: Rect, surface: &mut Surface, cx: &mut Context) {
|
||||||
@ -11,10 +11,11 @@ fn render(&mut self, viewport: Rect, surface: &mut Surface, cx: &mut Context) {
|
|||||||
// Calculate the area of the terminal to modify. Because we want to
|
// Calculate the area of the terminal to modify. Because we want to
|
||||||
// render at the bottom right, we use the viewport's width and height
|
// render at the bottom right, we use the viewport's width and height
|
||||||
// which evaluate to the most bottom right coordinate.
|
// which evaluate to the most bottom right coordinate.
|
||||||
let (width, height) = (self.width + 2, self.height + 2);
|
let width = self.width + 2 + 2; // +2 for border, +2 for margin
|
||||||
|
let height = self.height + 2; // +2 for border
|
||||||
let area = viewport.intersection(Rect::new(
|
let area = viewport.intersection(Rect::new(
|
||||||
viewport.width.saturating_sub(width),
|
viewport.width.saturating_sub(width),
|
||||||
viewport.height.saturating_sub(height + 2),
|
viewport.height.saturating_sub(height + 2), // +2 for statusline
|
||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
));
|
));
|
||||||
@ -24,15 +25,14 @@ fn render(&mut self, viewport: Rect, surface: &mut Surface, cx: &mut Context) {
|
|||||||
.title(self.title.as_str())
|
.title(self.title.as_str())
|
||||||
.borders(Borders::ALL)
|
.borders(Borders::ALL)
|
||||||
.border_style(style);
|
.border_style(style);
|
||||||
let inner = block.inner(area);
|
|
||||||
|
let margin = Margin {
|
||||||
|
vertical: 0,
|
||||||
|
horizontal: 1,
|
||||||
|
};
|
||||||
|
let inner = block.inner(area).inner(&margin);
|
||||||
block.render(area, surface);
|
block.render(area, surface);
|
||||||
|
|
||||||
// Only write as many lines as there are rows available.
|
Paragraph::new(self.text.as_str()).render(inner, surface);
|
||||||
for (y, line) in (inner.y..)
|
|
||||||
.zip(self.text.lines())
|
|
||||||
.take(inner.height as usize)
|
|
||||||
{
|
|
||||||
surface.set_string(inner.x, y, line, style);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,45 +16,33 @@ pub struct Info {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Info {
|
impl Info {
|
||||||
// body is a BTreeMap instead of a HashMap because keymaps are represented
|
pub fn new(title: &str, body: Vec<(&str, Vec<KeyEvent>)>) -> Info {
|
||||||
// with nested hashmaps with no ordering, and each invocation of infobox would
|
let body = body
|
||||||
// show different orders of items
|
.into_iter()
|
||||||
pub fn key(title: &str, body: Vec<(&str, Vec<KeyEvent>)>) -> Info {
|
.map(|(desc, events)| {
|
||||||
let (lpad, mpad, rpad) = (1, 2, 1);
|
let events = events.iter().map(ToString::to_string).collect::<Vec<_>>();
|
||||||
let keymaps_width: u16 = body
|
(desc, events.join(", "))
|
||||||
.iter()
|
})
|
||||||
.map(|r| r.1.iter().map(|e| e.width() as u16 + 2).sum::<u16>() - 2)
|
.collect::<Vec<_>>();
|
||||||
.max()
|
|
||||||
.unwrap();
|
let keymaps_width = body.iter().map(|r| r.1.len()).max().unwrap();
|
||||||
let mut text = String::new();
|
let mut text = String::new();
|
||||||
let mut width = 0;
|
|
||||||
let height = body.len() as u16;
|
for (desc, keyevents) in &body {
|
||||||
for (desc, keyevents) in body {
|
let _ = writeln!(
|
||||||
let keyevent = keyevents[0];
|
text,
|
||||||
let mut left = keymaps_width - keyevent.width() as u16;
|
"{:width$} {}",
|
||||||
for _ in 0..lpad {
|
keyevents,
|
||||||
text.push(' ');
|
desc,
|
||||||
}
|
width = keymaps_width
|
||||||
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 {
|
Info {
|
||||||
title: title.to_string(),
|
title: title.to_string(),
|
||||||
|
width: text.lines().map(|l| l.width()).max().unwrap() as u16,
|
||||||
|
height: body.len() as u16,
|
||||||
text,
|
text,
|
||||||
width,
|
|
||||||
height,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user