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
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
|
use std::iter::repeat;
use tui::{
buffer::Buffer,
layout::Rect,
widgets::{Block, Paragraph, Text, Widget},
};
#[derive(Default)]
pub struct ListState {
/// The index at which the list last started. Used for scrolling
pub start_index: usize,
}
impl ListState {
pub fn update(&mut self, selected: Option<usize>, height: usize) -> &mut Self {
self.start_index = match selected {
Some(pos) => match height as usize {
h if self.start_index + h - 1 < pos => pos - h + 1,
_ if self.start_index > pos => pos,
_ => self.start_index,
},
None => 0,
};
self
}
}
pub fn fill_background_to_right(mut s: String, entire_width: u16) -> String {
match (s.len(), entire_width as usize) {
(x, y) if x >= y => s,
(x, y) => {
s.extend(repeat(' ').take(y - x));
s
}
}
}
pub struct List<'b, 't, I>
where
I: Iterator<Item = Vec<Text<'t>>>,
{
pub block: Option<Block<'b>>,
pub items: I,
}
impl<'b, 't, I> Widget for List<'b, 't, I>
where
I: Iterator<Item = Vec<Text<'t>>>,
{
fn draw(&mut self, area: Rect, buf: &mut Buffer) {
let list_area = match self.block {
Some(ref mut b) => {
b.draw(area, buf);
b.inner(area)
}
None => area,
};
if list_area.width < 1 || list_area.height < 1 {
return;
}
for (i, text_iterator) in self
.items
.by_ref()
.enumerate()
.take(list_area.height as usize)
{
let (x, y) = (list_area.left(), list_area.top() + i as u16);
Paragraph::new(text_iterator.iter()).draw(
Rect {
x,
y,
width: list_area.width,
height: 1,
},
buf,
);
}
}
}
|