Refactor infobox rendering and parsing (#579)

This commit is contained in:
Gokul Soumya 2021-08-17 05:55:48 +05:30 committed by GitHub
parent 27616153bc
commit 14c08e855f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 35 additions and 47 deletions

View File

@ -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)
} }
} }

View File

@ -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);
}
} }
} }

View File

@ -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,
} }
} }
} }