From eabf97e6c9de14b53efb4184593dab7852bf1564 Mon Sep 17 00:00:00 2001 From: Chris Emerson Date: Sat, 4 Nov 2023 10:13:09 +0000 Subject: [PATCH 1/2] Handle the colspan=0 case. Although HTML5 forbids colspan=0, it's interpreted in some browsers as meaning "the rest of the columns", so handle this by replacing it with the expanded size before rendering. Fixes #90. --- src/lib.rs | 37 ++++++++++++++++++++++++--- src/tests.rs | 72 ++++++++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 106 insertions(+), 3 deletions(-) diff --git a/src/lib.rs b/src/lib.rs index a847a63..74504e6 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -182,7 +182,7 @@ impl RenderTableRow { /// Count the number of cells in the row. /// Takes into account colspan. pub fn num_cells(&self) -> usize { - self.cells.iter().map(|cell| cell.colspan).sum() + self.cells.iter().map(|cell| cell.colspan.max(1)).sum() } /// Return an iterator over (column, &cell)s, which /// takes into account colspan. @@ -644,7 +644,7 @@ fn tbody_to_render_tree<'a, 'b, T: Write>( _err_out: &'b mut T, ) -> TreeMapResult<'a, (), Handle, RenderNode> { pending(handle, |_, rowchildren| { - let rows = rowchildren + let mut rows = rowchildren .into_iter() .flat_map(|rownode| { if let RenderNodeInfo::TableRow(row, _) = rownode.info { @@ -654,7 +654,37 @@ fn tbody_to_render_tree<'a, 'b, T: Write>( None } }) - .collect(); + .collect::>(); + + // Handle colspan=0 by replacing it. + // Get a list of (has_zero_colspan, sum_colspan) + let num_columns = + rows.iter() + .map(|row| { + row.cells() + // Treat the column as having colspan 1 for initial counting. + .map(|cell| (cell.colspan == 0, cell.colspan.max(1))) + .fold((false, 0), |a, b| { + (a.0 || b.0, a.1 + b.1) + }) + }) + .collect::>(); + + let max_columns = num_columns.iter().map(|(_, span)| span).max().unwrap_or(&1); + + for (i, &(has_zero, num_cols)) in num_columns.iter().enumerate() { + // Note this won't be sensible if more than one column has colspan=0, + // but that's not very well defined anyway. + if has_zero { + for cell in rows[i].cells_mut() { + if cell.colspan == 0 { + // +1 because we said it had 1 to start with + cell.colspan = max_columns - num_cols + 1; + } + } + } + } + Some(RenderNode::new(RenderNodeInfo::TableBody(rows))) }) } @@ -1377,6 +1407,7 @@ fn render_table_tree( for cell in row.cells() { // FIXME: get_size_estimate is still recursive. let mut estimate = cell.get_size_estimate(); + // If the cell has a colspan>1, then spread its size between the // columns. estimate.size /= cell.colspan; diff --git a/src/tests.rs b/src/tests.rs index d36e045..d7f4411 100644 --- a/src/tests.rs +++ b/src/tests.rs @@ -132,6 +132,78 @@ fn test_colspan() { ); } +#[test] +fn test_colspan_zero() { + test_html( + br##" + + + + + + + + + + + + + + +
123
123
123
+"##, + r#"─┬─┬─ +1│2│3 +─┴─┼─ +12 │3 +─┬─┴─ +1│23 +─┴─── +"#, + 12, + ); +} + +#[test] +fn test_colspan_large() { + test_html( + br##" + + + + + + + + + + + + + + +
123
123
123
+"##, + r#"──────────── +1 +//////////// +2 +//////////// +3 +──────────── +12 +//////////// +3 +──────────── +1 +//////////// +23 +──────────── +"#, + 12, + ); +} + #[test] fn test_para() { assert_eq_str!(from_read(&b"

Hello

"[..], 10), "Hello\n"); From baf903d9c1d2668935fe658029b0cbada623e838 Mon Sep 17 00:00:00 2001 From: Chris Emerson Date: Sun, 5 Nov 2023 11:06:06 +0000 Subject: [PATCH 2/2] Avoid a potential division by a 0 width. --- src/lib.rs | 2 +- src/tests.rs | 62 ++++++++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 63 insertions(+), 1 deletion(-) diff --git a/src/lib.rs b/src/lib.rs index 74504e6..ca3f938 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1424,7 +1424,7 @@ fn render_table_tree( + col_sizes.len().saturating_sub(1); let width = renderer.width(); - let vert_row = min_size > width; + let vert_row = min_size > width || width == 0; let mut col_widths: Vec = if !vert_row { col_sizes diff --git a/src/tests.rs b/src/tests.rs index d7f4411..b6d9ec2 100644 --- a/src/tests.rs +++ b/src/tests.rs @@ -1334,6 +1334,68 @@ fn test_empty_cols() { ); } +#[test] +fn test_empty_table() { + test_html( + br##" +
+"##, + r#" +"#, + 12, + ); +} + +#[test] +fn test_table_empty_single_row() { + test_html( + br##" +
+"##, + r#" +"#, + 12, + ); +} + +#[test] +fn test_table_empty_single_row_empty_cell() { + test_html( + br##" +
+"##, + r#" +"#, + 12, + ); +} + +#[test] +fn test_renderer_zero_width() { + test_html( + br##"
  • x
+"##, +// Unfortunately the "x" ends up not being rendered as it doesn't fit. + r#"* + +"#, + 2, + ); +} + +#[test] +fn test_ul_tiny_table() { + test_html( + br##"
  • x
+"##, + r#"* ─ + x + ─ +"#, + 12, + ); +} + #[test] fn test_issue_54_oob() { test_html(