From a78feb46aefaf8e8950e9b029984e9ff98fe69b0 Mon Sep 17 00:00:00 2001
From: Ben Bridle <ben@derelict.engineering>
Date: Mon, 6 Jan 2025 12:21:06 +1300
Subject: Rewrite the library a second time

---
 src/table.rs | 85 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 file changed, 85 insertions(+)
 create mode 100644 src/table.rs

(limited to 'src/table.rs')

diff --git a/src/table.rs b/src/table.rs
new file mode 100644
index 0000000..071bd1a
--- /dev/null
+++ b/src/table.rs
@@ -0,0 +1,85 @@
+use crate::*;
+
+pub struct Table {
+    /// A [Line] is the content of a cell, a group of cells forms a table row,
+    /// a group of rows forms a separated section of the table, and a group of
+    /// sections forms the table itself.
+    /// Each row in the table has the same number of columns as the table header.
+    pub sections: Vec<Vec<Vec<Line>>>,
+    pub columns: Vec<Column>,
+}
+
+impl Table {
+    pub fn from_strs(lines: &[&str]) -> Option<Self> {
+        let mut lines = lines.into_iter();
+        let columns: Vec<Column> = {
+            let names = split_cells(lines.next()?)?;
+            let alignments = parse_alignments(lines.next()?)?;
+            if names.len() != alignments.len() { return None }
+            let make_column = |(n, a)| Column { name: n, alignment: a };
+            std::iter::zip(names, alignments).map(make_column).collect()
+        };
+        let mut sections = Vec::new();
+        let mut rows = Vec::new();
+
+        for line in lines {
+            if let Some(alignments) = parse_alignments(line) {
+                if alignments.len() != columns.len() { return None }
+                sections.push(std::mem::take(&mut rows))
+            } else {
+                let row: Vec<Line> = split_cells(line)?;
+                if row.len() != columns.len() { return None }
+                rows.push(row);
+            }
+        }
+        if !rows.is_empty() {
+            sections.push(std::mem::take(&mut rows));
+        }
+        return Some( Self { columns, sections } );
+    }
+}
+
+pub struct Column {
+    pub name: Line,
+    pub alignment: Alignment,
+}
+
+pub enum Alignment {
+    Left,
+    Center,
+    Right,
+}
+
+impl Alignment {
+    pub fn from_str(cell: &str) -> Option<Self> {
+        if !cell.chars().all(|c| c == ':' || c == '-') {
+            return None }
+        match (cell.starts_with(':'), cell.ends_with(':')) {
+            (false, false) => Some(Alignment::Left),
+            (false, true ) => Some(Alignment::Right),
+            (true,  false) => Some(Alignment::Left),
+            (true,  true ) => Some(Alignment::Center),
+        }
+    }
+}
+
+fn split_columns(line: &str) -> Option<Vec<&str>> {
+    if let Some(("", tail)) = line.split_once('|') {
+        if let Some((head, "")) = tail.rsplit_once('|') {
+            return Some(head.split('|').map(str::trim).collect());
+        }
+    }
+    return None;
+}
+
+fn split_cells(line: &str) -> Option<Vec<Line>> {
+    Some(split_columns(line)?.into_iter().map(Line::from_str).collect())
+}
+
+fn parse_alignments(line: &str) -> Option<Vec<Alignment>> {
+    let mut alignments = Vec::new();
+    for cell in split_columns(line)? {
+        alignments.push(Alignment::from_str(cell)?);
+    }
+    Some(alignments)
+}
-- 
cgit v1.2.3-70-g09d2