use ui_model::{Attrs, UiModel}; pub struct ModelLayout { pub model: UiModel, rows_filled: usize, } impl ModelLayout { const ROWS_STEP: usize = 10; pub fn new(columns: u64) -> Self { ModelLayout { model: UiModel::new(ModelLayout::ROWS_STEP as u64, columns), rows_filled: 0, } } pub fn layout_append(&mut self, lines: &Vec, Vec)>>) -> (usize, usize) { let rows_filled = self.rows_filled; self.layout_replace(rows_filled, lines) } pub fn layout(&mut self, lines: &Vec, Vec)>>, ) -> (usize, usize) { self.layout_replace(0, lines) } fn check_model_size(&mut self, rows: usize) { if rows > self.model.rows { let model_cols = self.model.columns; let model_rows = ((rows / ModelLayout::ROWS_STEP) + 1) * ModelLayout::ROWS_STEP; let mut model = UiModel::new(model_rows as u64, model_cols as u64); self.model.copy_rows(&mut model, self.rows_filled); } } /// Wrap all lines into model /// /// returns actual width fn layout_replace( &mut self, row_offset: usize, lines: &Vec, Vec)>>, ) -> (usize, usize) { let rows = ModelLayout::count_lines(&lines, self.model.columns); self.check_model_size(rows + row_offset); self.rows_filled = rows + row_offset; let mut max_col_idx = 0; let mut col_idx = 0; let mut row_idx = row_offset; for content in lines { for &(ref attr, ref ch_list) in content { for ch in ch_list { if col_idx >= self.model.columns { col_idx = 0; row_idx += 1; } else { col_idx += 1; } if max_col_idx < col_idx { max_col_idx = col_idx; } self.model.set_cursor(row_idx, col_idx as usize); self.model.put(*ch, false, attr.as_ref()); } } row_idx += 1; } (max_col_idx + 1, rows) } fn count_lines(lines: &Vec, Vec)>>, max_columns: usize) -> usize { let mut row_count = 0; for line in lines { let len: usize = line.iter().map(|c| c.1.len()).sum(); row_count += len / (max_columns + 1) + 1; } row_count } } #[cfg(test)] mod tests { use super::*; #[test] fn test_count_lines() { let lines = vec![vec![(None, vec!['a'; 5])]]; let rows = ModelLayout::count_lines(&lines, 4); assert_eq!(2, rows); } }