</head>
<body>
<table>
- <tbody>";
+ <tbody>
+";
const FOOTER: &str = " </tbody>
</table>
</body>
.collect()
}
+fn render_instance(entry: &Entry) -> String {
+ match &entry.instance {
+ None => String::from("✓ "),
+ Some(instance) => String::from(instance) + " ",
+ }
+}
+
+fn render_cell(col: &str, row: &RowInput) -> String {
+ // TODO: Escape HTML special characters
+ let entries: Vec<&Entry> = row.entries.iter().filter(|e| e.col == col).collect();
+ let class = if entries.is_empty() { "" } else { "yes" };
+ let all_empty = entries.iter().all(|e| e.instance.is_none());
+ let contents = if entries.is_empty() || (all_empty && entries.len() == 1) {
+ String::new()
+ } else if all_empty {
+ format!("{}", entries.len())
+ } else {
+ entries
+ .iter()
+ .map(|i| render_instance(i))
+ .collect::<String>()
+ };
+ format!("<td class=\"{class}\">{}</td>", contents.trim())
+}
+
+fn render_row(columns: &[String], row: &RowInput) -> String {
+ // This is O(n^2) & doesn't need to be
+ // TODO: Escape HTML special characters
+ format!(
+ "<tr><th>{}</th>{}</tr>\n",
+ row.label,
+ &columns
+ .iter()
+ .map(|col| render_cell(col, row))
+ .collect::<String>()
+ )
+}
+
+fn render_column_headers(columns: &[String]) -> String {
+ // TODO: Escape HTML special characters
+ String::from("<th></th>")
+ + &columns
+ .iter()
+ .map(|c| format!("<th>{c}</th>"))
+ .collect::<String>()
+ + "\n"
+}
+
/// # Errors
///
/// Will return `Err` if
/// * an indented line with no preceding non-indented line
pub fn tablify(input: impl std::io::Read) -> Result<String, std::io::Error> {
let rows = read_rows(input).collect::<Result<Vec<_>, _>>()?;
- let _columns = column_order(&rows);
- Ok(String::from(HEADER) + "Hello, world!" + FOOTER)
+ let columns = column_order(&rows);
+ Ok(String::from(HEADER)
+ + &render_column_headers(&columns)
+ + &rows
+ .into_iter()
+ .map(|r| render_row(&columns, &r))
+ .collect::<String>()
+ + FOOTER)
}
#[cfg(test)]
vec![(1, String::from("bar")), (2, String::from("baz"))]
);
}
+
+ #[test]
+ fn test_render_cell() {
+ assert_eq!(
+ render_cell(
+ "foo",
+ &RowInput {
+ label: String::from("nope"),
+ entries: vec![]
+ }
+ ),
+ String::from("<td class=\"\"></td>")
+ );
+ assert_eq!(
+ render_cell(
+ "foo",
+ &RowInput {
+ label: String::from("nope"),
+ entries: vec![Entry::from("bar")]
+ }
+ ),
+ String::from("<td class=\"\"></td>")
+ );
+ assert_eq!(
+ render_cell(
+ "foo",
+ &RowInput {
+ label: String::from("nope"),
+ entries: vec![Entry::from("foo")]
+ }
+ ),
+ String::from("<td class=\"yes\"></td>")
+ );
+ assert_eq!(
+ render_cell(
+ "foo",
+ &RowInput {
+ label: String::from("nope"),
+ entries: vec![Entry::from("foo"), Entry::from("foo")]
+ }
+ ),
+ String::from("<td class=\"yes\">2</td>")
+ );
+ assert_eq!(
+ render_cell(
+ "foo",
+ &RowInput {
+ label: String::from("nope"),
+ entries: vec![Entry::from("foo: 5"), Entry::from("foo: 10")]
+ }
+ ),
+ String::from("<td class=\"yes\">5 10</td>")
+ );
+ assert_eq!(
+ render_cell(
+ "foo",
+ &RowInput {
+ label: String::from("nope"),
+ entries: vec![Entry::from("foo: 5"), Entry::from("foo")]
+ }
+ ),
+ String::from("<td class=\"yes\">5 ✓</td>")
+ );
+ }
}