//! //! Example for [TableDataIter] used with [StatefulWidget] //! use crate::mini_salsa::theme::THEME; use crate::mini_salsa::{run_ui, setup_logging, MiniSalsaState}; use format_num_pattern::NumberFormat; use rat_event::{HandleEvent, Regular}; use rat_focus::{Focus, FocusBuilder, FocusFlag}; use rat_ftable::event::Outcome; use rat_ftable::selection::RowSelection; use rat_ftable::textdata::{Cell, Row}; use rat_ftable::{Table, TableContext, TableDataIter, TableState}; use rat_scrolled::Scroll; use ratatui::buffer::Buffer; use ratatui::layout::{Constraint, Flex, Layout, Rect}; use ratatui::text::Span; use ratatui::widgets::{block, Block, StatefulWidget, Widget}; use ratatui::Frame; use std::cmp::max; use std::iter::Enumerate; use std::slice::Iter; mod data; mod mini_salsa; fn main() -> Result<(), anyhow::Error> { setup_logging()?; let mut data = Data { table_data: data::DATA .iter() .map(|v| Sample { text: *v, num1: rand::random(), num2: rand::random(), check: rand::random(), }) .take(100_000) .collect(), }; let mut state = State { table: Default::default(), }; run_ui("iter", handle_table, repaint_table, &mut data, &mut state) } struct Sample { text: &'static str, num1: f32, num2: f32, check: bool, } struct Data { table_data: Vec, } struct State { table: TableState, } fn repaint_table( frame: &mut Frame<'_>, area: Rect, data: &mut Data, _istate: &mut MiniSalsaState, state: &mut State, ) -> Result<(), anyhow::Error> { let l0 = Layout::horizontal([Constraint::Percentage(61)]) .flex(Flex::Center) .split(area); struct DataIter<'a> { size: usize, iter: Enumerate>, item: Option<(usize, &'a Sample)>, } impl<'a> TableDataIter<'a> for DataIter<'a> { fn rows(&self) -> Option { Some(self.size) } fn nth(&mut self, n: usize) -> bool { self.item = self.iter.nth(n); self.item.is_some() } fn render_cell(&self, _ctx: &TableContext, column: usize, area: Rect, buf: &mut Buffer) { let row = self.item.expect("data"); match column { 0 => { let row_fmt = NumberFormat::new("000000").expect("fmt"); let span = Span::from(row_fmt.fmt_u(row.0)); span.render(area, buf); } 1 => { let span = Span::from(row.1.text); span.render(area, buf); } 2 => { let num1_fmt = NumberFormat::new("####0.00").expect("fmt"); let span = Span::from(num1_fmt.fmt_u(row.1.num1)); span.render(area, buf); } 3 => { let num2_fmt = NumberFormat::new("####0.00").expect("fmt"); let span = Span::from(num2_fmt.fmt_u(row.1.num2)); span.render(area, buf); } 4 => { let cc = if row.1.check { "\u{2622}" } else { "\u{2623}" }; let span = Span::from(cc); span.render(area, buf); } _ => {} } } } Table::default() .iter(DataIter { size: data.table_data.len(), iter: data.table_data.iter().enumerate(), item: None, }) .widths([ Constraint::Length(6), Constraint::Length(20), Constraint::Length(15), Constraint::Length(15), Constraint::Length(3), ]) .column_spacing(1) .header(Row::new([ Cell::from("Nr"), Cell::from("Text"), Cell::from("Val1"), Cell::from("Val2"), Cell::from("State"), ])) .footer(Row::new(["a", "b", "c", "d", "e"])) .block( Block::bordered() .border_type(block::BorderType::Rounded) .border_style(THEME.block()) .title_style(THEME.block_title()) .title("tabledata-iter (no row-count)"), ) .vscroll(Scroll::new()) .styles(THEME.table_style()) .render(l0[0], frame.buffer_mut(), &mut state.table); Ok(()) } fn focus(state: &mut State) -> Focus { let mut fb = FocusBuilder::new(None); fb.widget(&state.table); fb.widget(&FocusFlag::new()); fb.build() } fn handle_table( event: &crossterm::event::Event, _data: &mut Data, _istate: &mut MiniSalsaState, state: &mut State, ) -> Result { let f = focus(state).handle(event, Regular); let r = state.table.handle(event, Regular); Ok(max(f, r)) }