103 lines
2.5 KiB
Rust
103 lines
2.5 KiB
Rust
|
|
use ratatui::buffer::Buffer;
|
||
|
|
use ratatui::layout::Rect;
|
||
|
|
use ratatui::text::Line;
|
||
|
|
use ratatui::widgets::Paragraph;
|
||
|
|
use ratatui::widgets::WidgetRef;
|
||
|
|
|
||
|
|
pub trait Renderable {
|
||
|
|
fn render(&self, area: Rect, buf: &mut Buffer);
|
||
|
|
fn desired_height(&self, width: u16) -> u16;
|
||
|
|
}
|
||
|
|
|
||
|
|
impl Renderable for () {
|
||
|
|
fn render(&self, _area: Rect, _buf: &mut Buffer) {}
|
||
|
|
fn desired_height(&self, _width: u16) -> u16 {
|
||
|
|
0
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
impl Renderable for &str {
|
||
|
|
fn render(&self, area: Rect, buf: &mut Buffer) {
|
||
|
|
self.render_ref(area, buf);
|
||
|
|
}
|
||
|
|
fn desired_height(&self, _width: u16) -> u16 {
|
||
|
|
1
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
impl Renderable for String {
|
||
|
|
fn render(&self, area: Rect, buf: &mut Buffer) {
|
||
|
|
self.render_ref(area, buf);
|
||
|
|
}
|
||
|
|
fn desired_height(&self, _width: u16) -> u16 {
|
||
|
|
1
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
impl<'a> Renderable for Line<'a> {
|
||
|
|
fn render(&self, area: Rect, buf: &mut Buffer) {
|
||
|
|
WidgetRef::render_ref(self, area, buf);
|
||
|
|
}
|
||
|
|
fn desired_height(&self, _width: u16) -> u16 {
|
||
|
|
1
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
impl<'a> Renderable for Paragraph<'a> {
|
||
|
|
fn render(&self, area: Rect, buf: &mut Buffer) {
|
||
|
|
self.render_ref(area, buf);
|
||
|
|
}
|
||
|
|
fn desired_height(&self, width: u16) -> u16 {
|
||
|
|
self.line_count(width) as u16
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
impl<R: Renderable> Renderable for Option<R> {
|
||
|
|
fn render(&self, area: Rect, buf: &mut Buffer) {
|
||
|
|
if let Some(renderable) = self {
|
||
|
|
renderable.render(area, buf);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
fn desired_height(&self, width: u16) -> u16 {
|
||
|
|
if let Some(renderable) = self {
|
||
|
|
renderable.desired_height(width)
|
||
|
|
} else {
|
||
|
|
0
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
pub struct ColumnRenderable {
|
||
|
|
children: Vec<Box<dyn Renderable>>,
|
||
|
|
}
|
||
|
|
|
||
|
|
impl Renderable for ColumnRenderable {
|
||
|
|
fn render(&self, area: Rect, buf: &mut Buffer) {
|
||
|
|
let mut y = area.y;
|
||
|
|
for child in &self.children {
|
||
|
|
let child_area = Rect::new(area.x, y, area.width, child.desired_height(area.width))
|
||
|
|
.intersection(area);
|
||
|
|
if !child_area.is_empty() {
|
||
|
|
child.render(child_area, buf);
|
||
|
|
}
|
||
|
|
y += child_area.height;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
fn desired_height(&self, width: u16) -> u16 {
|
||
|
|
self.children
|
||
|
|
.iter()
|
||
|
|
.map(|child| child.desired_height(width))
|
||
|
|
.sum()
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
impl ColumnRenderable {
|
||
|
|
pub fn new(children: impl IntoIterator<Item = Box<dyn Renderable>>) -> Self {
|
||
|
|
Self {
|
||
|
|
children: children.into_iter().collect(),
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|