1#![doc = include_str!("../readme.md")]
2
3mod cellselection;
4pub mod edit;
5mod noselection;
6mod rowselection;
7mod rowsetselection;
8mod table;
9pub mod textdata;
10mod util;
11
12use crate::textdata::Row;
13use ratatui::buffer::Buffer;
14use ratatui::layout::{Constraint, Rect};
15use ratatui::style::Style;
16
17#[derive(Debug)]
19pub struct TableContext {
20 pub focus: bool,
22
23 pub selected_cell: bool,
25 pub selected_row: bool,
27 pub selected_column: bool,
29
30 pub style: Style,
32 pub row_style: Option<Style>,
34 pub select_style: Option<Style>,
36
37 pub space_area: Rect,
40 pub row_area: Rect,
42
43 pub non_exhaustive: NonExhaustive,
45}
46
47pub trait TableData<'a> {
53 fn rows(&self) -> usize;
55
56 fn header(&self) -> Option<Row<'a>> {
59 None
60 }
61
62 fn footer(&self) -> Option<Row<'a>> {
65 None
66 }
67
68 #[allow(unused_variables)]
70 fn row_height(&self, row: usize) -> u16 {
71 1
72 }
73
74 #[allow(unused_variables)]
76 fn row_style(&self, row: usize) -> Option<Style> {
77 None
78 }
79
80 fn widths(&self) -> Vec<Constraint> {
82 Vec::default()
83 }
84
85 fn render_cell(
88 &self,
89 ctx: &TableContext,
90 column: usize,
91 row: usize,
92 area: Rect,
93 buf: &mut Buffer,
94 );
95}
96
97impl<'a> TableData<'a> for Box<dyn TableData<'a> + 'a> {
98 fn rows(&self) -> usize {
99 (**self).rows()
100 }
101
102 fn header(&self) -> Option<Row<'a>> {
103 (**self).header()
104 }
105
106 fn footer(&self) -> Option<Row<'a>> {
107 (**self).footer()
108 }
109
110 fn row_height(&self, row: usize) -> u16 {
111 (**self).row_height(row)
112 }
113
114 fn row_style(&self, row: usize) -> Option<Style> {
115 (**self).row_style(row)
116 }
117
118 fn widths(&self) -> Vec<Constraint> {
119 (**self).widths()
120 }
121
122 fn render_cell(
123 &self,
124 ctx: &TableContext,
125 column: usize,
126 row: usize,
127 area: Rect,
128 buf: &mut Buffer,
129 ) {
130 (**self).render_cell(ctx, column, row, area, buf)
131 }
132}
133
134pub trait TableDataIter<'a> {
138 fn cloned(&self) -> Option<Box<dyn TableDataIter<'a> + 'a>> {
142 None
143 }
144
145 fn rows(&self) -> Option<usize>;
153
154 fn header(&self) -> Option<Row<'a>> {
157 None
158 }
159
160 fn footer(&self) -> Option<Row<'a>> {
163 None
164 }
165
166 fn nth(&mut self, n: usize) -> bool;
169
170 fn row_height(&self) -> u16 {
172 1
173 }
174
175 fn row_style(&self) -> Option<Style> {
177 None
178 }
179
180 fn widths(&self) -> Vec<Constraint> {
182 Vec::default()
183 }
184
185 fn render_cell(&self, ctx: &TableContext, column: usize, area: Rect, buf: &mut Buffer);
188}
189
190pub trait TableSelection {
192 fn count(&self) -> usize;
194
195 fn is_selected_row(&self, row: usize) -> bool;
197
198 fn is_selected_column(&self, column: usize) -> bool;
200
201 fn is_selected_cell(&self, column: usize, row: usize) -> bool;
203
204 fn lead_selection(&self) -> Option<(usize, usize)>;
206}
207
208use crate::_private::NonExhaustive;
209
210pub use table::{handle_doubleclick_events, Table, TableState, TableStyle};
211
212pub mod selection {
214 pub use crate::cellselection::CellSelection;
215 pub mod cellselection {
216 pub use crate::cellselection::{handle_events, handle_mouse_events};
217 }
218 pub use crate::noselection::NoSelection;
219 pub mod noselection {
220 pub use crate::noselection::{handle_events, handle_mouse_events};
221 }
222 pub use crate::rowselection::RowSelection;
223 pub mod rowselection {
224 pub use crate::rowselection::{handle_events, handle_mouse_events};
225 }
226 pub use crate::rowsetselection::RowSetSelection;
227 pub mod rowsetselection {
228 pub use crate::rowsetselection::{handle_events, handle_mouse_events};
229 }
230}
231
232pub mod event {
234 pub use rat_event::*;
235
236 #[derive(Debug, Clone, Copy, PartialEq, Eq)]
238 #[non_exhaustive]
239 pub enum TableOutcome {
240 Continue,
242 Unchanged,
244 Changed,
246 Selected,
248 }
249
250 impl ConsumedEvent for TableOutcome {
251 fn is_consumed(&self) -> bool {
252 *self != TableOutcome::Continue
253 }
254 }
255
256 impl From<TableOutcome> for Outcome {
257 fn from(value: TableOutcome) -> Self {
258 match value {
259 TableOutcome::Continue => Outcome::Continue,
260 TableOutcome::Unchanged => Outcome::Unchanged,
261 TableOutcome::Changed => Outcome::Changed,
262 TableOutcome::Selected => Outcome::Changed,
263 }
264 }
265 }
266
267 #[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
269 pub enum DoubleClickOutcome {
270 Continue,
272 Unchanged,
275 Changed,
280 ClickClick(usize, usize),
282 }
283
284 impl From<DoubleClickOutcome> for Outcome {
285 fn from(value: DoubleClickOutcome) -> Self {
286 match value {
287 DoubleClickOutcome::Continue => Outcome::Continue,
288 DoubleClickOutcome::Unchanged => Outcome::Unchanged,
289 DoubleClickOutcome::Changed => Outcome::Changed,
290 DoubleClickOutcome::ClickClick(_, _) => Outcome::Changed,
291 }
292 }
293 }
294
295 impl From<Outcome> for DoubleClickOutcome {
296 fn from(value: Outcome) -> Self {
297 match value {
298 Outcome::Continue => DoubleClickOutcome::Continue,
299 Outcome::Unchanged => DoubleClickOutcome::Unchanged,
300 Outcome::Changed => DoubleClickOutcome::Changed,
301 }
302 }
303 }
304
305 impl ConsumedEvent for DoubleClickOutcome {
306 fn is_consumed(&self) -> bool {
307 !matches!(self, DoubleClickOutcome::Continue)
308 }
309 }
310
311 #[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
313 pub enum EditOutcome {
314 Continue,
316 Unchanged,
319 Changed,
324 Cancel,
326 Commit,
328 CommitAndAppend,
330 CommitAndEdit,
332 Insert,
334 Remove,
336 Edit,
338 Append,
341 }
342
343 impl From<Outcome> for EditOutcome {
344 fn from(value: Outcome) -> Self {
345 match value {
346 Outcome::Continue => EditOutcome::Continue,
347 Outcome::Unchanged => EditOutcome::Unchanged,
348 Outcome::Changed => EditOutcome::Changed,
349 }
350 }
351 }
352
353 impl From<EditOutcome> for Outcome {
354 fn from(value: EditOutcome) -> Self {
355 match value {
356 EditOutcome::Continue => Outcome::Continue,
357 EditOutcome::Unchanged => Outcome::Unchanged,
358 EditOutcome::Changed => Outcome::Changed,
359 EditOutcome::Insert => Outcome::Unchanged,
360 EditOutcome::Remove => Outcome::Unchanged,
361 EditOutcome::Edit => Outcome::Unchanged,
362 EditOutcome::Append => Outcome::Unchanged,
363 EditOutcome::Cancel => Outcome::Unchanged,
364 EditOutcome::Commit => Outcome::Unchanged,
365 EditOutcome::CommitAndAppend => Outcome::Unchanged,
366 EditOutcome::CommitAndEdit => Outcome::Unchanged,
367 }
368 }
369 }
370
371 impl ConsumedEvent for EditOutcome {
372 fn is_consumed(&self) -> bool {
373 !matches!(self, EditOutcome::Continue)
374 }
375 }
376}
377
378mod _private {
379 #[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
380 pub struct NonExhaustive;
381}