1use core::cmp::Ordering;
6use std::mem;
7use std::ops::Range;
8
9use app_units::Au;
10use atomic_refcell::AtomicRef;
11use log::warn;
12use rayon::iter::{IndexedParallelIterator, IntoParallelRefIterator, ParallelIterator};
13use servo_arc::Arc;
14use strum::{EnumIter, IntoEnumIterator};
15use style::Zero;
16use style::computed_values::border_collapse::T as BorderCollapse;
17use style::computed_values::box_sizing::T as BoxSizing;
18use style::computed_values::caption_side::T as CaptionSide;
19use style::computed_values::empty_cells::T as EmptyCells;
20use style::computed_values::position::T as Position;
21use style::computed_values::table_layout::T as TableLayoutMode;
22use style::computed_values::visibility::T as Visibility;
23use style::properties::ComputedValues;
24use style::values::computed::{
25 AlignmentBaseline, BaselineShift, BorderStyle, LengthPercentage as ComputedLengthPercentage,
26 Percentage,
27};
28use style::values::generics::box_::BaselineShiftKeyword;
29
30use super::{
31 ArcRefCell, CollapsedBorder, CollapsedBorderLine, SpecificTableGridInfo, Table, TableCaption,
32 TableLayoutStyle, TableSlot, TableSlotCell, TableSlotCoordinates, TableTrack, TableTrackGroup,
33};
34use crate::context::LayoutContext;
35use crate::dom::WeakLayoutBox;
36use crate::formatting_contexts::Baselines;
37use crate::fragment_tree::{
38 BoxFragment, CollapsedBlockMargins, ExtraBackground, Fragment, FragmentFlags,
39 PositioningFragment, SpecificLayoutInfo,
40};
41use crate::geom::{
42 LogicalRect, LogicalSides, LogicalSides1D, LogicalVec2, PhysicalPoint, PhysicalRect,
43 PhysicalSides, PhysicalVec, ToLogical, ToLogicalWithContainingBlock,
44};
45use crate::layout_box_base::IndependentFormattingContextLayoutResult;
46use crate::positioned::{PositioningContext, PositioningContextLength, relative_adjustement};
47use crate::sizing::{
48 ComputeInlineContentSizes, ContentSizes, InlineContentSizesResult, LazySize, Size,
49 SizeConstraint,
50};
51use crate::style_ext::{
52 BorderStyleColor, Clamp, ComputedValuesExt, LayoutStyle, PaddingBorderMargin,
53};
54use crate::table::WeakTableLevelBox;
55use crate::{
56 ConstraintSpace, ContainingBlock, ContainingBlockSize, IndefiniteContainingBlock, WritingMode,
57};
58
59#[derive(PartialEq)]
60enum CellContentAlignment {
61 Top,
62 Bottom,
63 Middle,
64 Baseline,
65}
66
67struct CellLayout {
71 layout: IndependentFormattingContextLayoutResult,
72 padding: LogicalSides<Au>,
73 border: LogicalSides<Au>,
74 positioning_context: PositioningContext,
75}
76
77impl CellLayout {
78 fn ascent(&self) -> Au {
79 self.layout
80 .baselines
81 .first
82 .unwrap_or(self.layout.content_block_size)
83 }
84
85 fn outer_block_size(&self) -> Au {
87 self.layout.content_block_size + self.border.block_sum() + self.padding.block_sum()
88 }
89
90 fn is_empty(&self) -> bool {
93 self.layout.fragments.is_empty()
94 }
95
96 fn is_empty_for_empty_cells(&self) -> bool {
98 self.layout
99 .fragments
100 .iter()
101 .all(|fragment| matches!(fragment, Fragment::AbsoluteOrFixedPositioned(_)))
102 }
103}
104
105#[derive(Clone, Debug, Default)]
107struct RowLayout {
108 constrained: bool,
109 has_cell_with_span_greater_than_one: bool,
110 percent: Percentage,
111}
112
113#[derive(Clone, Debug, Default)]
115struct ColumnLayout {
116 constrained: bool,
117 has_originating_cells: bool,
118 content_sizes: ContentSizes,
119 percentage: Option<Percentage>,
120}
121
122fn max_two_optional_percentages(
123 a: Option<Percentage>,
124 b: Option<Percentage>,
125) -> Option<Percentage> {
126 match (a, b) {
127 (Some(a), Some(b)) => Some(Percentage(a.0.max(b.0))),
128 _ => a.or(b),
129 }
130}
131
132impl ColumnLayout {
133 fn incorporate_cell_measure(&mut self, cell_measure: &CellOrTrackMeasure) {
134 self.content_sizes.max_assign(cell_measure.content_sizes);
135 self.percentage = max_two_optional_percentages(self.percentage, cell_measure.percentage);
136 }
137}
138
139impl CollapsedBorder {
140 fn new(style_color: BorderStyleColor, width: Au) -> Self {
141 Self { style_color, width }
142 }
143
144 fn from_layout_style(
145 layout_style: &LayoutStyle,
146 writing_mode: WritingMode,
147 ) -> LogicalSides<Self> {
148 let border_style_color = layout_style.style().border_style_color(writing_mode);
149 let border_width = layout_style.border_width(writing_mode);
150 LogicalSides {
151 inline_start: Self::new(border_style_color.inline_start, border_width.inline_start),
152 inline_end: Self::new(border_style_color.inline_end, border_width.inline_end),
153 block_start: Self::new(border_style_color.block_start, border_width.block_start),
154 block_end: Self::new(border_style_color.block_end, border_width.block_end),
155 }
156 }
157
158 fn max_assign(&mut self, other: &Self) {
159 if *self < *other {
160 *self = other.clone();
161 }
162 }
163
164 fn max_assign_to_slice(&self, slice: &mut [CollapsedBorder]) {
165 for collapsed_border in slice {
166 collapsed_border.max_assign(self)
167 }
168 }
169
170 fn hide(&mut self) {
171 self.style_color = BorderStyleColor::hidden();
172 self.width = Au::zero();
173 }
174}
175
176impl PartialOrd for CollapsedBorder {
183 fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
184 let is_hidden = |border: &Self| border.style_color.style == BorderStyle::Hidden;
185 let candidate = (is_hidden(self).cmp(&is_hidden(other)))
186 .then_with(|| self.width.cmp(&other.width))
187 .then_with(|| self.style_color.style.cmp(&other.style_color.style));
188 if !candidate.is_eq() || self.style_color.color == other.style_color.color {
189 Some(candidate)
190 } else {
191 None
192 }
193 }
194}
195
196impl Eq for CollapsedBorder {}
197
198type CollapsedBorders = LogicalVec2<Vec<CollapsedBorderLine>>;
199
200pub(crate) struct TableLayout<'a> {
204 table: &'a Table,
205 pbm: PaddingBorderMargin,
206 rows: Vec<RowLayout>,
207 columns: Vec<ColumnLayout>,
208 cell_measures: Vec<Vec<LogicalVec2<CellOrTrackMeasure>>>,
209 table_width: Au,
212 assignable_width: Au,
215 final_table_height: Au,
216 distributed_column_widths: Vec<Au>,
217 row_sizes: Vec<Au>,
218 row_baselines: Vec<Au>,
220 cells_laid_out: Vec<Vec<Option<CellLayout>>>,
221 basis_for_cell_padding_percentage: Au,
222 collapsed_borders: Option<CollapsedBorders>,
224 is_in_fixed_mode: bool,
225}
226
227#[derive(Clone, Debug)]
228struct CellOrTrackMeasure {
229 content_sizes: ContentSizes,
230 percentage: Option<Percentage>,
231}
232
233impl Zero for CellOrTrackMeasure {
234 fn zero() -> Self {
235 Self {
236 content_sizes: ContentSizes::zero(),
237 percentage: None,
238 }
239 }
240
241 fn is_zero(&self) -> bool {
242 self.content_sizes.is_zero() && self.percentage.is_none()
243 }
244}
245
246impl<'a> TableLayout<'a> {
247 fn new(table: &'a Table) -> TableLayout<'a> {
248 let style = &table.style;
251 let is_in_fixed_mode = style.get_table().table_layout == TableLayoutMode::Fixed &&
252 !matches!(
253 style.box_size(style.writing_mode).inline,
254 Size::Initial | Size::MaxContent
255 );
256 Self {
257 table,
258 pbm: PaddingBorderMargin::zero(),
259 rows: Vec::new(),
260 columns: Vec::new(),
261 cell_measures: Vec::new(),
262 table_width: Au::zero(),
263 assignable_width: Au::zero(),
264 final_table_height: Au::zero(),
265 distributed_column_widths: Vec::new(),
266 row_sizes: Vec::new(),
267 row_baselines: Vec::new(),
268 cells_laid_out: Vec::new(),
269 basis_for_cell_padding_percentage: Au::zero(),
270 collapsed_borders: None,
271 is_in_fixed_mode,
272 }
273 }
274
275 pub(crate) fn compute_cell_measures(
278 &mut self,
279 layout_context: &LayoutContext,
280 writing_mode: WritingMode,
281 ) {
282 let row_measures = vec![LogicalVec2::zero(); self.table.size.width];
283 self.cell_measures = vec![row_measures; self.table.size.height];
284
285 for row_index in 0..self.table.size.height {
286 for column_index in 0..self.table.size.width {
287 let cell = match self.table.slots[row_index][column_index] {
288 TableSlot::Cell(ref cell) => cell,
289 _ => continue,
290 }
291 .borrow();
292
293 let layout_style = cell.context.layout_style();
294 let padding = layout_style
295 .padding(writing_mode)
296 .percentages_relative_to(Au::zero());
297 let border = self
298 .get_collapsed_border_widths_for_area(LogicalSides {
299 inline_start: column_index,
300 inline_end: column_index + cell.colspan,
301 block_start: row_index,
302 block_end: row_index + cell.rowspan,
303 })
304 .unwrap_or_else(|| layout_style.border_width(writing_mode));
305
306 let padding_border_sums = LogicalVec2 {
307 inline: padding.inline_sum() + border.inline_sum(),
308 block: padding.block_sum() + border.block_sum(),
309 };
310
311 let CellOrColumnOuterSizes {
312 preferred: preferred_size,
313 min: min_size,
314 max: max_size,
315 percentage: percentage_size,
316 } = CellOrColumnOuterSizes::new(
317 &cell.context.base.style,
318 writing_mode,
319 &padding_border_sums,
320 self.is_in_fixed_mode,
321 );
322
323 let inline_measure = if self.is_in_fixed_mode {
328 if row_index > 0 {
329 CellOrTrackMeasure::zero()
330 } else {
331 CellOrTrackMeasure {
332 content_sizes: preferred_size.inline.into(),
333 percentage: percentage_size.inline,
334 }
335 }
336 } else {
337 let constraint_space = ConstraintSpace::new(
338 SizeConstraint::default(),
339 &cell.context.base.style,
340 cell.context.preferred_aspect_ratio(&padding_border_sums),
341 );
342 let inline_content_sizes = cell
343 .context
344 .inline_content_sizes(layout_context, &constraint_space)
345 .sizes +
346 padding_border_sums.inline.into();
347 assert!(
348 inline_content_sizes.max_content >= inline_content_sizes.min_content,
349 "the max-content size should never be smaller than the min-content size"
350 );
351
352 let outer_min_content_width = inline_content_sizes
354 .min_content
355 .clamp_between_extremums(min_size.inline, max_size.inline);
356 let outer_max_content_width = if self.columns[column_index].constrained {
357 inline_content_sizes
358 .min_content
359 .max(preferred_size.inline)
360 .clamp_between_extremums(min_size.inline, max_size.inline)
361 } else {
362 inline_content_sizes
363 .max_content
364 .max(preferred_size.inline)
365 .clamp_between_extremums(min_size.inline, max_size.inline)
366 };
367 assert!(outer_min_content_width <= outer_max_content_width);
368
369 CellOrTrackMeasure {
370 content_sizes: ContentSizes {
371 min_content: outer_min_content_width,
372 max_content: outer_max_content_width,
373 },
374 percentage: percentage_size.inline,
375 }
376 };
377
378 let block_measure = CellOrTrackMeasure {
382 content_sizes: preferred_size.block.into(),
383 percentage: percentage_size.block,
384 };
385
386 self.cell_measures[row_index][column_index] = LogicalVec2 {
387 inline: inline_measure,
388 block: block_measure,
389 };
390 }
391 }
392 }
393
394 fn compute_track_constrainedness_and_has_originating_cells(
400 &mut self,
401 writing_mode: WritingMode,
402 ) {
403 self.rows = vec![RowLayout::default(); self.table.size.height];
404 self.columns = vec![ColumnLayout::default(); self.table.size.width];
405
406 let is_length = |size: &Size<ComputedLengthPercentage>| {
407 size.to_numeric().is_some_and(|size| !size.has_percentage())
408 };
409
410 for column_index in 0..self.table.size.width {
411 if let Some(column) = self.table.columns.get(column_index) {
412 let column = column.borrow();
413 if is_length(&column.base.style.box_size(writing_mode).inline) {
414 self.columns[column_index].constrained = true;
415 continue;
416 }
417 if let Some(column_group_index) = column.group_index {
418 let column_group = self.table.column_groups[column_group_index].borrow();
419 if is_length(&column_group.base.style.box_size(writing_mode).inline) {
420 self.columns[column_index].constrained = true;
421 continue;
422 }
423 }
424 }
425 }
426
427 for row_index in 0..self.table.size.height {
428 if let Some(row) = self.table.rows.get(row_index) {
429 let row = row.borrow();
430 if is_length(&row.base.style.box_size(writing_mode).block) {
431 self.rows[row_index].constrained = true;
432 continue;
433 }
434 if let Some(row_group_index) = row.group_index {
435 let row_group = self.table.row_groups[row_group_index].borrow();
436 if is_length(&row_group.base.style.box_size(writing_mode).block) {
437 self.rows[row_index].constrained = true;
438 continue;
439 }
440 }
441 }
442 }
443
444 for column_index in 0..self.table.size.width {
445 for row_index in 0..self.table.size.height {
446 let coords = TableSlotCoordinates::new(column_index, row_index);
447 let cell_constrained = match self.table.resolve_first_cell(coords) {
448 Some(cell) if cell.colspan == 1 => cell
449 .context
450 .base
451 .style
452 .box_size(writing_mode)
453 .map(is_length),
454 _ => LogicalVec2::default(),
455 };
456
457 let rowspan_greater_than_1 = match self.table.slots[row_index][column_index] {
458 TableSlot::Cell(ref cell) => cell.borrow().rowspan > 1,
459 _ => false,
460 };
461
462 self.rows[row_index].has_cell_with_span_greater_than_one |= rowspan_greater_than_1;
463 self.rows[row_index].constrained |= cell_constrained.block;
464
465 let has_originating_cell =
466 matches!(self.table.get_slot(coords), Some(TableSlot::Cell(_)));
467 self.columns[column_index].has_originating_cells |= has_originating_cell;
468 self.columns[column_index].constrained |= cell_constrained.inline;
469 }
470 }
471 }
472
473 fn compute_column_measures(&mut self, writing_mode: WritingMode) {
476 let mut colspan_cell_constraints = Vec::new();
507 for column_index in 0..self.table.size.width {
508 let column = &mut self.columns[column_index];
509
510 let column_measure = self.table.get_column_measure_for_column_at_index(
511 writing_mode,
512 column_index,
513 self.is_in_fixed_mode,
514 );
515 column.content_sizes = column_measure.content_sizes;
516 column.percentage = column_measure.percentage;
517
518 for row_index in 0..self.table.size.height {
519 let coords = TableSlotCoordinates::new(column_index, row_index);
520 let cell_measure = &self.cell_measures[row_index][column_index].inline;
521
522 let cell = match self.table.get_slot(coords) {
523 Some(TableSlot::Cell(cell)) => cell,
524 _ => continue,
525 }
526 .borrow();
527
528 if cell.colspan != 1 {
529 colspan_cell_constraints.push(ColspanToDistribute {
530 starting_column: column_index,
531 span: cell.colspan,
532 content_sizes: cell_measure.content_sizes,
533 percentage: cell_measure.percentage,
534 });
535 continue;
536 }
537
538 column.incorporate_cell_measure(cell_measure);
541 }
542 }
543
544 colspan_cell_constraints.sort_by(ColspanToDistribute::comparison_for_sort);
546
547 self.distribute_colspanned_cells_to_columns(colspan_cell_constraints);
549
550 let mut total_intrinsic_percentage_width = 0.;
557 for column in self.columns.iter_mut() {
558 if let Some(ref mut percentage) = column.percentage {
559 let final_intrinsic_percentage_width =
560 percentage.0.min(1. - total_intrinsic_percentage_width);
561 total_intrinsic_percentage_width += final_intrinsic_percentage_width;
562 *percentage = Percentage(final_intrinsic_percentage_width);
563 }
564 }
565 }
566
567 fn distribute_colspanned_cells_to_columns(
568 &mut self,
569 colspan_cell_constraints: Vec<ColspanToDistribute>,
570 ) {
571 for colspan_cell_constraints in colspan_cell_constraints {
572 self.distribute_colspanned_cell_to_columns(colspan_cell_constraints);
573 }
574 }
575
576 fn distribute_colspanned_cell_to_columns(
581 &mut self,
582 colspan_cell_constraints: ColspanToDistribute,
583 ) {
584 let border_spacing = self.table.border_spacing().inline;
585 let column_range = colspan_cell_constraints.range();
586 let column_count = column_range.len();
587 let total_border_spacing =
588 border_spacing.scale_by((colspan_cell_constraints.span - 1) as f32);
589
590 let mut percent_columns_count = 0;
591 let mut columns_percent_sum = 0.;
592 let mut columns_non_percent_max_inline_size_sum = Au::zero();
593 for column in self.columns[column_range.clone()].iter() {
594 if let Some(percentage) = column.percentage {
595 percent_columns_count += 1;
596 columns_percent_sum += percentage.0;
597 } else {
598 columns_non_percent_max_inline_size_sum += column.content_sizes.max_content;
599 }
600 }
601
602 let colspan_percentage = colspan_cell_constraints.percentage.unwrap_or_default();
603 let surplus_percent = colspan_percentage.0 - columns_percent_sum;
604 if surplus_percent > 0. && column_count > percent_columns_count {
605 for column in self.columns[column_range.clone()].iter_mut() {
606 if column.percentage.is_some() {
607 continue;
608 }
609
610 let ratio = if columns_non_percent_max_inline_size_sum.is_zero() {
611 1. / ((column_count - percent_columns_count) as f32)
612 } else {
613 column.content_sizes.max_content.to_f32_px() /
614 columns_non_percent_max_inline_size_sum.to_f32_px()
615 };
616 column.percentage = Some(Percentage(surplus_percent * ratio));
617 }
618 }
619
620 let colspan_cell_min_size = (colspan_cell_constraints.content_sizes.min_content -
621 total_border_spacing)
622 .max(Au::zero());
623 let distributed_minimum =
624 Self::distribute_width_to_columns(colspan_cell_min_size, &self.columns[column_range]);
625 {
626 let column_span = &mut self.columns[colspan_cell_constraints.range()];
627 for (column, minimum_size) in column_span.iter_mut().zip(distributed_minimum) {
628 column.content_sizes.min_content.max_assign(minimum_size);
629 }
630 }
631
632 let colspan_cell_max_size = (colspan_cell_constraints.content_sizes.max_content -
633 total_border_spacing)
634 .max(Au::zero());
635 let distributed_maximum = Self::distribute_width_to_columns(
636 colspan_cell_max_size,
637 &self.columns[colspan_cell_constraints.range()],
638 );
639 {
640 let column_span = &mut self.columns[colspan_cell_constraints.range()];
641 for (column, maximum_size) in column_span.iter_mut().zip(distributed_maximum) {
642 column
643 .content_sizes
644 .max_content
645 .max_assign(maximum_size.max(column.content_sizes.min_content));
646 }
647 }
648 }
649
650 fn compute_measures(&mut self, layout_context: &LayoutContext, writing_mode: WritingMode) {
651 self.compute_track_constrainedness_and_has_originating_cells(writing_mode);
652 self.compute_cell_measures(layout_context, writing_mode);
653 self.compute_column_measures(writing_mode);
654 }
655
656 fn compute_grid_min_max(&self) -> ContentSizes {
658 let mut largest_percentage_column_max_size = Au::zero();
671 let mut percent_sum = 0.;
672 let mut non_percent_columns_max_sum = Au::zero();
673 let mut grid_min_max = ContentSizes::zero();
674 for column in self.columns.iter() {
675 match column.percentage {
676 Some(percentage) if !percentage.is_zero() => {
677 largest_percentage_column_max_size.max_assign(
678 column
679 .content_sizes
680 .max_content
681 .scale_by(1.0 / percentage.0),
682 );
683 percent_sum += percentage.0;
684 },
685 _ => {
686 non_percent_columns_max_sum += column.content_sizes.max_content;
687 },
688 }
689
690 grid_min_max += column.content_sizes;
691 }
692
693 grid_min_max
694 .max_content
695 .max_assign(largest_percentage_column_max_size);
696
697 if !percent_sum.is_zero() &&
701 self.table
702 .percentage_columns_allowed_for_inline_content_sizes
703 {
704 let total_inline_size =
705 non_percent_columns_max_sum.scale_by(1.0 / (1.0 - percent_sum.min(1.0)));
706 grid_min_max.max_content.max_assign(total_inline_size);
707 }
708
709 assert!(
710 grid_min_max.min_content <= grid_min_max.max_content,
711 "GRIDMAX should never be smaller than GRIDMIN {:?}",
712 grid_min_max
713 );
714
715 let inline_border_spacing = self.table.total_border_spacing().inline;
716 grid_min_max.min_content += inline_border_spacing;
717 grid_min_max.max_content += inline_border_spacing;
718 grid_min_max
719 }
720
721 fn compute_caption_minimum_inline_size(&self, layout_context: &LayoutContext) -> Au {
723 let containing_block = IndefiniteContainingBlock {
724 size: LogicalVec2::default(),
725 style: &self.table.style,
726 };
727 self.table
728 .captions
729 .iter()
730 .map(|caption| {
731 caption
732 .borrow()
733 .context
734 .outer_inline_content_sizes(
735 layout_context,
736 &containing_block,
737 &LogicalVec2::zero(),
738 false, )
740 .sizes
741 .min_content
742 })
743 .max()
744 .unwrap_or_default()
745 }
746
747 fn compute_table_width(&mut self, containing_block_for_children: &ContainingBlock) {
748 self.table_width = containing_block_for_children.size.inline;
753
754 self.assignable_width = self.table_width - self.table.total_border_spacing().inline;
758
759 self.basis_for_cell_padding_percentage =
762 self.table_width - self.table.border_spacing().inline * 2;
763 }
764
765 fn distribute_width_to_columns(target_inline_size: Au, columns: &[ColumnLayout]) -> Vec<Au> {
768 if columns.is_empty() {
771 return Vec::new();
772 }
773
774 let mut min_content_sizing_guesses = Vec::new();
803 let mut min_content_percentage_sizing_guesses = Vec::new();
804 let mut min_content_specified_sizing_guesses = Vec::new();
805 let mut max_content_sizing_guesses = Vec::new();
806
807 for column in columns {
808 let min_content_width = column.content_sizes.min_content;
809 let max_content_width = column.content_sizes.max_content;
810 let constrained = column.constrained;
811
812 let (
813 min_content_percentage_sizing_guess,
814 min_content_specified_sizing_guess,
815 max_content_sizing_guess,
816 ) = if let Some(percentage) = column.percentage {
817 let resolved = target_inline_size.scale_by(percentage.0);
818 let percent_guess = min_content_width.max(resolved);
819 (percent_guess, percent_guess, percent_guess)
820 } else if constrained {
821 (min_content_width, max_content_width, max_content_width)
822 } else {
823 (min_content_width, min_content_width, max_content_width)
824 };
825
826 min_content_sizing_guesses.push(min_content_width);
827 min_content_percentage_sizing_guesses.push(min_content_percentage_sizing_guess);
828 min_content_specified_sizing_guesses.push(min_content_specified_sizing_guess);
829 max_content_sizing_guesses.push(max_content_sizing_guess);
830 }
831
832 let max_content_sizing_sum = max_content_sizing_guesses.iter().sum();
840 if target_inline_size >= max_content_sizing_sum {
841 Self::distribute_extra_width_to_columns(
842 columns,
843 &mut max_content_sizing_guesses,
844 max_content_sizing_sum,
845 target_inline_size,
846 );
847 return max_content_sizing_guesses;
848 }
849 let min_content_specified_sizing_sum = min_content_specified_sizing_guesses.iter().sum();
850 if target_inline_size == min_content_specified_sizing_sum {
851 return min_content_specified_sizing_guesses;
852 }
853 let min_content_percentage_sizing_sum = min_content_percentage_sizing_guesses.iter().sum();
854 if target_inline_size == min_content_percentage_sizing_sum {
855 return min_content_percentage_sizing_guesses;
856 }
857 let min_content_sizes_sum = min_content_sizing_guesses.iter().sum();
858 if target_inline_size <= min_content_sizes_sum {
859 return min_content_sizing_guesses;
860 }
861
862 let bounds = |sum_a, sum_b| target_inline_size > sum_a && target_inline_size < sum_b;
863
864 let blend = |a: &[Au], sum_a: Au, b: &[Au], sum_b: Au| {
865 let weight_a = (target_inline_size - sum_b).to_f32_px() / (sum_a - sum_b).to_f32_px();
867 let weight_b = 1.0 - weight_a;
868
869 let mut remaining_assignable_width = target_inline_size;
870 let mut widths: Vec<Au> = a
871 .iter()
872 .zip(b.iter())
873 .map(|(guess_a, guess_b)| {
874 let column_width = guess_a.scale_by(weight_a) + guess_b.scale_by(weight_b);
875 let column_width = column_width.min(remaining_assignable_width);
878 remaining_assignable_width -= column_width;
879 column_width
880 })
881 .collect();
882
883 if !remaining_assignable_width.is_zero() {
884 debug_assert!(
888 remaining_assignable_width >= Au::zero(),
889 "Sum of columns shouldn't exceed the assignable table width"
890 );
891 debug_assert!(
892 remaining_assignable_width <= Au::new(widths.len() as i32),
893 "A deviation of more than one Au per column is unlikely to be caused by float imprecision"
894 );
895
896 widths[0] += remaining_assignable_width;
899 }
900
901 debug_assert!(widths.iter().sum::<Au>() == target_inline_size);
902
903 widths
904 };
905
906 if bounds(min_content_sizes_sum, min_content_percentage_sizing_sum) {
907 return blend(
908 &min_content_sizing_guesses,
909 min_content_sizes_sum,
910 &min_content_percentage_sizing_guesses,
911 min_content_percentage_sizing_sum,
912 );
913 }
914
915 if bounds(
916 min_content_percentage_sizing_sum,
917 min_content_specified_sizing_sum,
918 ) {
919 return blend(
920 &min_content_percentage_sizing_guesses,
921 min_content_percentage_sizing_sum,
922 &min_content_specified_sizing_guesses,
923 min_content_specified_sizing_sum,
924 );
925 }
926
927 assert!(bounds(
928 min_content_specified_sizing_sum,
929 max_content_sizing_sum
930 ));
931 blend(
932 &min_content_specified_sizing_guesses,
933 min_content_specified_sizing_sum,
934 &max_content_sizing_guesses,
935 max_content_sizing_sum,
936 )
937 }
938
939 fn distribute_extra_width_to_columns(
942 columns: &[ColumnLayout],
943 column_sizes: &mut [Au],
944 column_sizes_sum: Au,
945 assignable_width: Au,
946 ) {
947 let all_columns = 0..columns.len();
948 let extra_inline_size = assignable_width - column_sizes_sum;
949
950 let has_originating_cells =
951 |column_index: &usize| columns[*column_index].has_originating_cells;
952 let is_constrained = |column_index: &usize| columns[*column_index].constrained;
953 let is_unconstrained = |column_index: &usize| !is_constrained(column_index);
954 let has_percent_greater_than_zero = |column_index: &usize| {
955 columns[*column_index]
956 .percentage
957 .is_some_and(|percentage| percentage.0 > 0.)
958 };
959 let has_percent_zero = |column_index: &usize| !has_percent_greater_than_zero(column_index);
960 let has_max_content =
961 |column_index: &usize| !columns[*column_index].content_sizes.max_content.is_zero();
962
963 let max_content_sum = |column_index: usize| columns[column_index].content_sizes.max_content;
964
965 let unconstrained_max_content_columns = all_columns
971 .clone()
972 .filter(is_unconstrained)
973 .filter(has_originating_cells)
974 .filter(has_percent_zero)
975 .filter(has_max_content);
976 let total_max_content_width: Au = unconstrained_max_content_columns
977 .clone()
978 .map(max_content_sum)
979 .sum();
980 if !total_max_content_width.is_zero() {
981 for column_index in unconstrained_max_content_columns {
982 column_sizes[column_index] += extra_inline_size.scale_by(
983 columns[column_index].content_sizes.max_content.to_f32_px() /
984 total_max_content_width.to_f32_px(),
985 );
986 }
987 return;
988 }
989
990 let unconstrained_no_percent_columns = all_columns
996 .clone()
997 .filter(is_unconstrained)
998 .filter(has_originating_cells)
999 .filter(has_percent_zero);
1000 let total_unconstrained_no_percent = unconstrained_no_percent_columns.clone().count();
1001 if total_unconstrained_no_percent > 0 {
1002 let extra_space_per_column =
1003 extra_inline_size.scale_by(1.0 / total_unconstrained_no_percent as f32);
1004 for column_index in unconstrained_no_percent_columns {
1005 column_sizes[column_index] += extra_space_per_column;
1006 }
1007 return;
1008 }
1009
1010 let constrained_max_content_columns = all_columns
1016 .clone()
1017 .filter(is_constrained)
1018 .filter(has_originating_cells)
1019 .filter(has_percent_zero)
1020 .filter(has_max_content);
1021 let total_max_content_width: Au = constrained_max_content_columns
1022 .clone()
1023 .map(max_content_sum)
1024 .sum();
1025 if !total_max_content_width.is_zero() {
1026 for column_index in constrained_max_content_columns {
1027 column_sizes[column_index] += extra_inline_size.scale_by(
1028 columns[column_index].content_sizes.max_content.to_f32_px() /
1029 total_max_content_width.to_f32_px(),
1030 );
1031 }
1032 return;
1033 }
1034
1035 let columns_with_percentage = all_columns.clone().filter(has_percent_greater_than_zero);
1041 let total_percent = columns_with_percentage
1042 .clone()
1043 .map(|column_index| columns[column_index].percentage.unwrap_or_default().0)
1044 .sum::<f32>();
1045 if total_percent > 0. {
1046 for column_index in columns_with_percentage {
1047 let column_percentage = columns[column_index].percentage.unwrap_or_default();
1048 column_sizes[column_index] +=
1049 extra_inline_size.scale_by(column_percentage.0 / total_percent);
1050 }
1051 return;
1052 }
1053
1054 let has_originating_cells_columns = all_columns.filter(has_originating_cells);
1058 let total_has_originating_cells = has_originating_cells_columns.clone().count();
1059 if total_has_originating_cells > 0 {
1060 let extra_space_per_column =
1061 extra_inline_size.scale_by(1.0 / total_has_originating_cells as f32);
1062 for column_index in has_originating_cells_columns {
1063 column_sizes[column_index] += extra_space_per_column;
1064 }
1065 return;
1066 }
1067
1068 let extra_space_for_all_columns = extra_inline_size.scale_by(1.0 / columns.len() as f32);
1071 for guess in column_sizes.iter_mut() {
1072 *guess += extra_space_for_all_columns;
1073 }
1074 }
1075
1076 fn layout_cells_in_row(
1079 &mut self,
1080 layout_context: &LayoutContext,
1081 containing_block_for_table: &ContainingBlock,
1082 ) {
1083 let layout_table_slot = |coordinate: TableSlotCoordinates, slot: &TableSlot| {
1084 let TableSlot::Cell(cell) = slot else {
1085 return None;
1086 };
1087
1088 let cell = cell.borrow();
1089 let area = LogicalSides {
1090 inline_start: coordinate.x,
1091 inline_end: coordinate.x + cell.colspan,
1092 block_start: coordinate.y,
1093 block_end: coordinate.y + cell.rowspan,
1094 };
1095 let layout_style = cell.context.layout_style();
1096 let border = self
1097 .get_collapsed_border_widths_for_area(area)
1098 .unwrap_or_else(|| {
1099 layout_style.border_width(containing_block_for_table.style.writing_mode)
1100 });
1101 let padding: LogicalSides<Au> = layout_style
1102 .padding(containing_block_for_table.style.writing_mode)
1103 .percentages_relative_to(self.basis_for_cell_padding_percentage);
1104 let padding_border_sums = LogicalVec2 {
1105 inline: padding.inline_sum() + border.inline_sum(),
1106 block: padding.block_sum() + border.block_sum(),
1107 };
1108 let border_spacing_spanned =
1109 self.table.border_spacing().inline * (cell.colspan - 1) as i32;
1110
1111 let mut total_cell_width = (coordinate.x..coordinate.x + cell.colspan)
1112 .map(|column_index| self.distributed_column_widths[column_index])
1113 .sum::<Au>() -
1114 padding_border_sums.inline +
1115 border_spacing_spanned;
1116 total_cell_width = total_cell_width.max(Au::zero());
1117
1118 let preferred_aspect_ratio = cell.context.preferred_aspect_ratio(&padding_border_sums);
1119 let containing_block_for_children = ContainingBlock {
1120 size: ContainingBlockSize {
1121 inline: total_cell_width,
1122 block: SizeConstraint::default(),
1123 },
1124 style: &cell.context.base.style,
1125 };
1126
1127 let mut positioning_context = PositioningContext::default();
1128 let layout = cell.context.layout(
1129 layout_context,
1130 &mut positioning_context,
1131 &containing_block_for_children,
1132 containing_block_for_table,
1133 preferred_aspect_ratio,
1134 &LazySize::intrinsic(),
1135 );
1136
1137 Some(CellLayout {
1138 layout,
1139 padding,
1140 border,
1141 positioning_context,
1142 })
1143 };
1144
1145 self.cells_laid_out = if layout_context.use_rayon {
1146 self.table
1147 .slots
1148 .par_iter()
1149 .enumerate()
1150 .map(|(row_index, row_slots)| {
1151 row_slots
1152 .par_iter()
1153 .enumerate()
1154 .map(|(column_index, slot)| {
1155 layout_table_slot(
1156 TableSlotCoordinates::new(column_index, row_index),
1157 slot,
1158 )
1159 })
1160 .collect()
1161 })
1162 .collect()
1163 } else {
1164 self.table
1165 .slots
1166 .iter()
1167 .enumerate()
1168 .map(|(row_index, row_slots)| {
1169 row_slots
1170 .iter()
1171 .enumerate()
1172 .map(|(column_index, slot)| {
1173 layout_table_slot(
1174 TableSlotCoordinates::new(column_index, row_index),
1175 slot,
1176 )
1177 })
1178 .collect()
1179 })
1180 .collect()
1181 };
1182
1183 for row_index in 0..self.table.size.height {
1186 for column_index in 0..self.table.size.width {
1187 let Some(layout) = &self.cells_laid_out[row_index][column_index] else {
1188 continue;
1189 };
1190
1191 self.cell_measures[row_index][column_index]
1192 .block
1193 .content_sizes
1194 .max_assign(layout.outer_block_size().into());
1195 }
1196 }
1197 }
1198
1199 fn do_first_row_layout(&mut self, writing_mode: WritingMode) -> Vec<Au> {
1202 let mut row_sizes = (0..self.table.size.height)
1203 .map(|row_index| {
1204 let (mut max_ascent, mut max_descent, mut max_row_height) =
1205 (Au::zero(), Au::zero(), Au::zero());
1206
1207 for column_index in 0..self.table.size.width {
1208 let cell = match self.table.slots[row_index][column_index] {
1209 TableSlot::Cell(ref cell) => cell,
1210 _ => continue,
1211 };
1212
1213 let layout = match self.cells_laid_out[row_index][column_index] {
1214 Some(ref layout) => layout,
1215 None => {
1216 warn!(
1217 "Did not find a layout at a slot index with an originating cell."
1218 );
1219 continue;
1220 },
1221 };
1222
1223 let cell = cell.borrow();
1224 let outer_block_size = layout.outer_block_size();
1225 if cell.rowspan == 1 {
1226 max_row_height.max_assign(outer_block_size);
1227 }
1228
1229 if cell.content_alignment() == CellContentAlignment::Baseline {
1230 let ascent = layout.ascent();
1231 let border_padding_start =
1232 layout.border.block_start + layout.padding.block_start;
1233 let border_padding_end = layout.border.block_end + layout.padding.block_end;
1234 max_ascent.max_assign(ascent + border_padding_start);
1235
1236 if cell.rowspan == 1 {
1240 max_descent.max_assign(
1241 layout.layout.content_block_size - ascent + border_padding_end,
1242 );
1243 }
1244 }
1245 }
1246 self.row_baselines.push(max_ascent);
1247 max_row_height.max(max_ascent + max_descent)
1248 })
1249 .collect();
1250 self.calculate_row_sizes_after_first_layout(&mut row_sizes, writing_mode);
1251 row_sizes
1252 }
1253
1254 #[allow(clippy::ptr_arg)] fn calculate_row_sizes_after_first_layout(
1259 &mut self,
1260 row_sizes: &mut Vec<Au>,
1261 writing_mode: WritingMode,
1262 ) {
1263 let mut cells_to_distribute = Vec::new();
1264 let mut total_percentage = 0.;
1265 #[allow(clippy::needless_range_loop)] for row_index in 0..self.table.size.height {
1267 let row_measure = self
1268 .table
1269 .get_row_measure_for_row_at_index(writing_mode, row_index);
1270 row_sizes[row_index].max_assign(row_measure.content_sizes.min_content);
1271
1272 let mut percentage = row_measure.percentage.unwrap_or_default().0;
1273 for column_index in 0..self.table.size.width {
1274 let cell_percentage = self.cell_measures[row_index][column_index]
1275 .block
1276 .percentage
1277 .unwrap_or_default()
1278 .0;
1279 percentage = percentage.max(cell_percentage);
1280
1281 let cell_measure = &self.cell_measures[row_index][column_index].block;
1282 let cell = match self.table.slots[row_index][column_index] {
1283 TableSlot::Cell(ref cell) if cell.borrow().rowspan > 1 => cell,
1284 TableSlot::Cell(_) => {
1285 row_sizes[row_index].max_assign(cell_measure.content_sizes.max_content);
1288 continue;
1289 },
1290 _ => continue,
1291 };
1292
1293 cells_to_distribute.push(RowspanToDistribute {
1294 coordinates: TableSlotCoordinates::new(column_index, row_index),
1295 cell: cell.borrow(),
1296 measure: cell_measure,
1297 });
1298 }
1299
1300 self.rows[row_index].percent = Percentage(percentage.min(1. - total_percentage));
1301 total_percentage += self.rows[row_index].percent.0;
1302 }
1303
1304 cells_to_distribute.sort_by(|a, b| {
1305 if a.range() == b.range() {
1306 return a
1307 .measure
1308 .content_sizes
1309 .min_content
1310 .cmp(&b.measure.content_sizes.min_content);
1311 }
1312 if a.fully_encloses(b) {
1313 return std::cmp::Ordering::Greater;
1314 }
1315 if b.fully_encloses(a) {
1316 return std::cmp::Ordering::Less;
1317 }
1318 a.coordinates.y.cmp(&b.coordinates.y)
1319 });
1320
1321 for rowspan_to_distribute in cells_to_distribute {
1322 let rows_spanned = rowspan_to_distribute.range();
1323 let current_rows_size = rows_spanned.clone().map(|index| row_sizes[index]).sum();
1324 let border_spacing_spanned =
1325 self.table.border_spacing().block * (rows_spanned.len() - 1) as i32;
1326 let excess_size = (rowspan_to_distribute.measure.content_sizes.min_content -
1327 current_rows_size -
1328 border_spacing_spanned)
1329 .max(Au::zero());
1330
1331 self.distribute_extra_size_to_rows(
1332 excess_size,
1333 rows_spanned,
1334 row_sizes,
1335 None,
1336 true, );
1338 }
1339 }
1340
1341 fn distribute_extra_size_to_rows(
1344 &self,
1345 mut excess_size: Au,
1346 track_range: Range<usize>,
1347 track_sizes: &mut [Au],
1348 percentage_resolution_size: Option<Au>,
1349 rowspan_distribution: bool,
1350 ) {
1351 if excess_size.is_zero() {
1352 return;
1353 }
1354
1355 let is_constrained = |track_index: &usize| self.rows[*track_index].constrained;
1356 let is_unconstrained = |track_index: &usize| !is_constrained(track_index);
1357 let is_empty: Vec<bool> = track_sizes.iter().map(|size| size.is_zero()).collect();
1358 let is_not_empty = |track_index: &usize| !is_empty[*track_index];
1359 let other_row_that_starts_a_rowspan = |track_index: &usize| {
1360 *track_index != track_range.start &&
1361 self.rows[*track_index].has_cell_with_span_greater_than_one
1362 };
1363
1364 if let Some(percentage_resolution_size) = percentage_resolution_size {
1368 let get_percent_block_size_deficit = |row_index: usize, track_size: Au| {
1369 let size_needed_for_percent =
1370 percentage_resolution_size.scale_by(self.rows[row_index].percent.0);
1371 (size_needed_for_percent - track_size).max(Au::zero())
1372 };
1373 let percent_block_size_deficit: Au = track_range
1374 .clone()
1375 .map(|index| get_percent_block_size_deficit(index, track_sizes[index]))
1376 .sum();
1377 let percent_distributable_block_size = percent_block_size_deficit.min(excess_size);
1378 if percent_distributable_block_size > Au::zero() {
1379 for track_index in track_range.clone() {
1380 let row_deficit =
1381 get_percent_block_size_deficit(track_index, track_sizes[track_index]);
1382 if row_deficit > Au::zero() {
1383 let ratio =
1384 row_deficit.to_f32_px() / percent_block_size_deficit.to_f32_px();
1385 let size = percent_distributable_block_size.scale_by(ratio);
1386 track_sizes[track_index] += size;
1387 excess_size -= size;
1388 }
1389 }
1390 }
1391 }
1392
1393 if rowspan_distribution {
1396 let rows_that_start_rowspan: Vec<usize> = track_range
1397 .clone()
1398 .filter(other_row_that_starts_a_rowspan)
1399 .collect();
1400 if !rows_that_start_rowspan.is_empty() {
1401 let scale = 1.0 / rows_that_start_rowspan.len() as f32;
1402 for track_index in rows_that_start_rowspan.iter() {
1403 track_sizes[*track_index] += excess_size.scale_by(scale);
1404 }
1405 return;
1406 }
1407 }
1408
1409 let unconstrained_non_empty_rows: Vec<usize> = track_range
1411 .clone()
1412 .filter(is_unconstrained)
1413 .filter(is_not_empty)
1414 .collect();
1415 if !unconstrained_non_empty_rows.is_empty() {
1416 let total_size: Au = unconstrained_non_empty_rows
1417 .iter()
1418 .map(|index| track_sizes[*index])
1419 .sum();
1420 for track_index in unconstrained_non_empty_rows.iter() {
1421 let scale = track_sizes[*track_index].to_f32_px() / total_size.to_f32_px();
1422 track_sizes[*track_index] += excess_size.scale_by(scale);
1423 }
1424 return;
1425 }
1426
1427 let (non_empty_rows, empty_rows): (Vec<usize>, Vec<usize>) =
1428 track_range.clone().partition(is_not_empty);
1429 let only_have_empty_rows = empty_rows.len() == track_range.len();
1430 if !empty_rows.is_empty() {
1431 if rowspan_distribution && only_have_empty_rows {
1434 track_sizes[*empty_rows.last().unwrap()] += excess_size;
1435 return;
1436 }
1437
1438 let non_empty_rows_all_constrained = !non_empty_rows.iter().any(is_unconstrained);
1441 if only_have_empty_rows || non_empty_rows_all_constrained {
1442 let mut rows_to_grow = &empty_rows;
1445 let unconstrained_empty_rows: Vec<usize> = rows_to_grow
1446 .iter()
1447 .copied()
1448 .filter(is_unconstrained)
1449 .collect();
1450 if !unconstrained_empty_rows.is_empty() {
1451 rows_to_grow = &unconstrained_empty_rows;
1452 }
1453
1454 let scale = 1.0 / rows_to_grow.len() as f32;
1456 for track_index in rows_to_grow.iter() {
1457 track_sizes[*track_index] += excess_size.scale_by(scale);
1458 }
1459 return;
1460 }
1461 }
1462
1463 if !non_empty_rows.is_empty() {
1466 let total_size: Au = non_empty_rows.iter().map(|index| track_sizes[*index]).sum();
1467 for track_index in non_empty_rows.iter() {
1468 let scale = track_sizes[*track_index].to_f32_px() / total_size.to_f32_px();
1469 track_sizes[*track_index] += excess_size.scale_by(scale);
1470 }
1471 }
1472 }
1473
1474 fn compute_table_height_and_final_row_heights(
1477 &mut self,
1478 mut row_sizes: Vec<Au>,
1479 containing_block_for_children: &ContainingBlock,
1480 ) {
1481 let table_height_from_style = containing_block_for_children.size.block.definite_or_min();
1488
1489 let block_border_spacing = self.table.total_border_spacing().block;
1490 let table_height_from_rows = row_sizes.iter().sum::<Au>() + block_border_spacing;
1491 self.final_table_height = table_height_from_rows.max(table_height_from_style);
1492
1493 if self.final_table_height == table_height_from_rows {
1496 self.row_sizes = row_sizes;
1497 return;
1498 }
1499
1500 self.distribute_extra_size_to_rows(
1505 self.final_table_height - table_height_from_rows,
1506 0..self.table.size.height,
1507 &mut row_sizes,
1508 Some(self.final_table_height),
1509 false, );
1511 self.row_sizes = row_sizes;
1512 }
1513
1514 fn layout_caption(
1515 &self,
1516 caption: &TableCaption,
1517 layout_context: &LayoutContext,
1518 parent_positioning_context: &mut PositioningContext,
1519 ) -> BoxFragment {
1520 let containing_block = &ContainingBlock {
1521 size: ContainingBlockSize {
1522 inline: self.table_width + self.pbm.padding_border_sums.inline,
1523 block: SizeConstraint::default(),
1524 },
1525 style: &self.table.style,
1526 };
1527
1528 let ignore_block_margins_for_stretch = LogicalSides1D::new(false, false);
1532
1533 let mut positioning_context =
1534 PositioningContext::new_for_layout_box_base(&caption.context.base);
1535 let mut box_fragment = caption.context.layout_in_flow_block_level(
1536 layout_context,
1537 positioning_context
1538 .as_mut()
1539 .unwrap_or(parent_positioning_context),
1540 containing_block,
1541 None, ignore_block_margins_for_stretch,
1543 false, );
1545
1546 if let Some(mut positioning_context) = positioning_context.take() {
1547 positioning_context.layout_collected_children(layout_context, &mut box_fragment);
1548 parent_positioning_context.append(positioning_context);
1549 }
1550
1551 box_fragment
1552 }
1553
1554 #[servo_tracing::instrument(name = "Table::layout", skip_all)]
1557 fn layout(
1558 mut self,
1559 layout_context: &LayoutContext,
1560 positioning_context: &mut PositioningContext,
1561 containing_block_for_children: &ContainingBlock,
1562 containing_block_for_table: &ContainingBlock,
1563 ) -> IndependentFormattingContextLayoutResult {
1564 let table_writing_mode = containing_block_for_children.style.writing_mode;
1565 self.compute_border_collapse(table_writing_mode);
1566 let layout_style = self.table.layout_style(Some(&self));
1567
1568 self.pbm = layout_style
1569 .padding_border_margin_with_writing_mode_and_containing_block_inline_size(
1570 table_writing_mode,
1571 containing_block_for_table.size.inline,
1572 );
1573 self.compute_measures(layout_context, table_writing_mode);
1574 self.compute_table_width(containing_block_for_children);
1575
1576 let containing_block_for_logical_conversion = ContainingBlock {
1591 size: ContainingBlockSize {
1592 inline: self.table_width,
1593 block: containing_block_for_table.size.block,
1594 },
1595 style: containing_block_for_children.style,
1596 };
1597 let offset_from_wrapper = -self.pbm.padding - self.pbm.border;
1598 let mut current_block_offset = offset_from_wrapper.block_start;
1599
1600 let mut table_layout = IndependentFormattingContextLayoutResult {
1601 fragments: Vec::new(),
1602 content_block_size: Zero::zero(),
1603 content_inline_size_for_table: None,
1604 baselines: Baselines::default(),
1605 depends_on_block_constraints: true,
1606 specific_layout_info: Some(SpecificLayoutInfo::TableWrapper),
1607 collapsible_margins_in_children: CollapsedBlockMargins::zero(),
1608 };
1609
1610 #[derive(EnumIter, PartialEq)]
1611 enum TableWrapperSection {
1612 TopCaptions,
1613 Grid,
1614 BottomCaptions,
1615 }
1616 impl TableWrapperSection {
1617 fn accepts_caption(&self, caption: &TableCaption) -> bool {
1618 match caption.context.style().clone_caption_side() {
1619 CaptionSide::Top => *self == TableWrapperSection::TopCaptions,
1620 CaptionSide::Bottom => *self == TableWrapperSection::BottomCaptions,
1621 }
1622 }
1623 }
1624
1625 for section in TableWrapperSection::iter() {
1626 if section == TableWrapperSection::Grid {
1627 let original_positioning_context_length = positioning_context.len();
1628 let grid_fragment = self.layout_grid(
1629 layout_context,
1630 positioning_context,
1631 &containing_block_for_logical_conversion,
1632 containing_block_for_children,
1633 );
1634
1635 let logical_grid_content_rect = grid_fragment
1638 .content_rect()
1639 .to_logical(&containing_block_for_logical_conversion);
1640 let grid_pbm = grid_fragment
1641 .padding_border_margin()
1642 .to_logical(table_writing_mode);
1643 table_layout.baselines = grid_fragment.baselines(table_writing_mode).offset(
1644 current_block_offset +
1645 logical_grid_content_rect.start_corner.block +
1646 grid_pbm.block_start,
1647 );
1648
1649 grid_fragment.base.set_rect(
1650 LogicalRect {
1651 start_corner: LogicalVec2 {
1652 inline: offset_from_wrapper.inline_start + grid_pbm.inline_start,
1653 block: current_block_offset + grid_pbm.block_start,
1654 },
1655 size: grid_fragment
1656 .base
1657 .rect()
1658 .size
1659 .to_logical(table_writing_mode),
1660 }
1661 .as_physical(Some(&containing_block_for_logical_conversion)),
1662 );
1663
1664 current_block_offset += grid_fragment
1665 .border_rect()
1666 .size
1667 .to_logical(table_writing_mode)
1668 .block;
1669 if logical_grid_content_rect.size.inline < self.table_width {
1670 table_layout.content_inline_size_for_table =
1672 Some(logical_grid_content_rect.size.inline);
1673 }
1674
1675 let grid_fragment = Fragment::Box(grid_fragment.into());
1676 positioning_context.adjust_static_position_of_hoisted_fragments(
1677 &grid_fragment,
1678 original_positioning_context_length,
1679 );
1680 table_layout.fragments.push(grid_fragment);
1681 } else {
1682 let caption_fragments = self.table.captions.iter().filter_map(|caption| {
1683 let caption = caption.borrow();
1684 if !section.accepts_caption(&caption) {
1685 return None;
1686 }
1687
1688 let original_positioning_context_length = positioning_context.len();
1689 let caption_fragment =
1690 self.layout_caption(&caption, layout_context, positioning_context);
1691
1692 let caption_pbm = caption_fragment
1695 .padding_border_margin()
1696 .to_logical(table_writing_mode);
1697
1698 let caption_style = caption_fragment.style().clone();
1699 let caption_relative_offset = match caption_style.clone_position() {
1700 Position::Relative => {
1701 relative_adjustement(&caption_style, containing_block_for_children)
1702 },
1703 _ => LogicalVec2::zero(),
1704 };
1705
1706 caption_fragment.base.set_rect(
1707 LogicalRect {
1708 start_corner: LogicalVec2 {
1709 inline: offset_from_wrapper.inline_start + caption_pbm.inline_start,
1710 block: current_block_offset + caption_pbm.block_start,
1711 } + caption_relative_offset,
1712 size: caption_fragment
1713 .content_rect()
1714 .size
1715 .to_logical(table_writing_mode),
1716 }
1717 .as_physical(Some(&containing_block_for_logical_conversion)),
1718 );
1719
1720 current_block_offset += caption_fragment
1721 .margin_rect()
1722 .size
1723 .to_logical(table_writing_mode)
1724 .block;
1725
1726 let caption_fragment = Fragment::Box(caption_fragment.into());
1727 positioning_context.adjust_static_position_of_hoisted_fragments(
1728 &caption_fragment,
1729 original_positioning_context_length,
1730 );
1731
1732 caption.context.base.set_fragment(caption_fragment.clone());
1733 Some(caption_fragment)
1734 });
1735 table_layout.fragments.extend(caption_fragments);
1736 }
1737 }
1738
1739 table_layout.content_block_size = current_block_offset + offset_from_wrapper.block_end;
1740 table_layout
1741 }
1742
1743 fn layout_grid(
1746 &mut self,
1747 layout_context: &LayoutContext,
1748 positioning_context: &mut PositioningContext,
1749 containing_block_for_logical_conversion: &ContainingBlock,
1750 containing_block_for_children: &ContainingBlock,
1751 ) -> BoxFragment {
1752 self.distributed_column_widths =
1753 Self::distribute_width_to_columns(self.assignable_width, &self.columns);
1754 self.layout_cells_in_row(layout_context, containing_block_for_children);
1755 let table_writing_mode = containing_block_for_children.style.writing_mode;
1756 let first_layout_row_heights = self.do_first_row_layout(table_writing_mode);
1757 self.compute_table_height_and_final_row_heights(
1758 first_layout_row_heights,
1759 containing_block_for_children,
1760 );
1761
1762 assert_eq!(self.table.size.height, self.row_sizes.len());
1763 assert_eq!(self.table.size.width, self.distributed_column_widths.len());
1764
1765 if self.table.size.width == 0 && self.table.size.height == 0 {
1766 let content_rect = LogicalRect {
1767 start_corner: LogicalVec2::zero(),
1768 size: LogicalVec2 {
1769 inline: self.table_width,
1770 block: self.final_table_height,
1771 },
1772 }
1773 .as_physical(Some(containing_block_for_logical_conversion));
1774 return BoxFragment::new(
1775 self.table.grid_base_fragment_info,
1776 self.table.grid_style.clone(),
1777 Vec::new(),
1778 content_rect,
1779 self.pbm.padding.to_physical(table_writing_mode),
1780 self.pbm.border.to_physical(table_writing_mode),
1781 PhysicalSides::zero(),
1782 self.specific_layout_info_for_grid(),
1783 );
1784 }
1785
1786 let mut table_fragments = Vec::new();
1787 let table_and_track_dimensions = TableAndTrackDimensions::new(self);
1788 self.make_fragments_for_columns_and_column_groups(
1789 &table_and_track_dimensions,
1790 &mut table_fragments,
1791 );
1792
1793 let mut baselines = Baselines::default();
1794 let mut row_group_fragment_layout = None;
1795 for row_index in 0..self.table.size.height {
1796 if row_index == 0 {
1806 let row_end = table_and_track_dimensions
1807 .get_row_rect(0)
1808 .max_block_position();
1809 baselines.first = Some(row_end);
1810 baselines.last = Some(row_end);
1811 }
1812
1813 let row_is_collapsed = self.is_row_collapsed(row_index);
1814 let table_row = self.table.rows[row_index].borrow();
1815 let mut row_fragment_layout = RowFragmentLayout::new(
1816 &table_row,
1817 row_index,
1818 &table_and_track_dimensions,
1819 &self.table.style,
1820 );
1821
1822 let old_row_group_index = row_group_fragment_layout
1823 .as_ref()
1824 .map(|layout: &RowGroupFragmentLayout| layout.index);
1825 if table_row.group_index != old_row_group_index {
1826 if let Some(old_row_group_layout) = row_group_fragment_layout.take() {
1828 table_fragments.push(old_row_group_layout.finish(
1829 layout_context,
1830 positioning_context,
1831 containing_block_for_logical_conversion,
1832 containing_block_for_children,
1833 ));
1834 }
1835
1836 if let Some(new_group_index) = table_row.group_index {
1838 row_group_fragment_layout = Some(RowGroupFragmentLayout::new(
1839 self.table.row_groups[new_group_index].clone(),
1840 new_group_index,
1841 &table_and_track_dimensions,
1842 ));
1843 }
1844 }
1845
1846 let column_indices = 0..self.table.size.width;
1847 row_fragment_layout.fragments.reserve(self.table.size.width);
1848 for column_index in column_indices {
1849 self.do_final_cell_layout(
1850 row_index,
1851 column_index,
1852 &table_and_track_dimensions,
1853 &mut baselines,
1854 &mut row_fragment_layout,
1855 row_group_fragment_layout.as_mut(),
1856 positioning_context,
1857 self.is_column_collapsed(column_index) || row_is_collapsed,
1858 );
1859 }
1860
1861 let row_fragment = row_fragment_layout.finish(
1862 layout_context,
1863 positioning_context,
1864 containing_block_for_logical_conversion,
1865 containing_block_for_children,
1866 &mut row_group_fragment_layout,
1867 );
1868
1869 match row_group_fragment_layout.as_mut() {
1870 Some(layout) => layout.fragments.push(row_fragment),
1871 None => table_fragments.push(row_fragment),
1872 }
1873 }
1874
1875 if let Some(row_group_layout) = row_group_fragment_layout.take() {
1876 table_fragments.push(row_group_layout.finish(
1877 layout_context,
1878 positioning_context,
1879 containing_block_for_logical_conversion,
1880 containing_block_for_children,
1881 ));
1882 }
1883
1884 let content_rect = LogicalRect {
1885 start_corner: LogicalVec2::zero(),
1886 size: LogicalVec2 {
1887 inline: table_and_track_dimensions.table_rect.max_inline_position(),
1888 block: table_and_track_dimensions.table_rect.max_block_position(),
1889 },
1890 }
1891 .as_physical(Some(containing_block_for_logical_conversion));
1892 BoxFragment::new(
1893 self.table.grid_base_fragment_info,
1894 self.table.grid_style.clone(),
1895 table_fragments,
1896 content_rect,
1897 self.pbm.padding.to_physical(table_writing_mode),
1898 self.pbm.border.to_physical(table_writing_mode),
1899 PhysicalSides::zero(),
1900 self.specific_layout_info_for_grid(),
1901 )
1902 .with_baselines(baselines)
1903 }
1904
1905 fn specific_layout_info_for_grid(&mut self) -> Option<SpecificLayoutInfo> {
1906 mem::take(&mut self.collapsed_borders).map(|mut collapsed_borders| {
1907 let mut track_sizes = LogicalVec2 {
1910 inline: mem::take(&mut self.distributed_column_widths),
1911 block: mem::take(&mut self.row_sizes),
1912 };
1913 for (column_index, column_size) in track_sizes.inline.iter_mut().enumerate() {
1914 if self.is_column_collapsed(column_index) {
1915 mem::take(column_size);
1916 }
1917 }
1918 for (row_index, row_size) in track_sizes.block.iter_mut().enumerate() {
1919 if self.is_row_collapsed(row_index) {
1920 mem::take(row_size);
1921 }
1922 }
1923 let writing_mode = self.table.style.writing_mode;
1924 if !writing_mode.is_bidi_ltr() {
1925 track_sizes.inline.reverse();
1926 collapsed_borders.inline.reverse();
1927 for border_line in &mut collapsed_borders.block {
1928 border_line.reverse();
1929 }
1930 }
1931 SpecificLayoutInfo::TableGridWithCollapsedBorders(Box::new(SpecificTableGridInfo {
1932 collapsed_borders: if writing_mode.is_horizontal() {
1933 PhysicalVec::new(collapsed_borders.inline, collapsed_borders.block)
1934 } else {
1935 PhysicalVec::new(collapsed_borders.block, collapsed_borders.inline)
1936 },
1937 track_sizes: if writing_mode.is_horizontal() {
1938 PhysicalVec::new(track_sizes.inline, track_sizes.block)
1939 } else {
1940 PhysicalVec::new(track_sizes.block, track_sizes.inline)
1941 },
1942 }))
1943 })
1944 }
1945
1946 fn is_row_collapsed(&self, row_index: usize) -> bool {
1947 let Some(row) = &self.table.rows.get(row_index) else {
1948 return false;
1949 };
1950
1951 let row = row.borrow();
1952 if row.base.style.get_inherited_box().visibility == Visibility::Collapse {
1953 return true;
1954 }
1955 let row_group = match row.group_index {
1956 Some(group_index) => self.table.row_groups[group_index].borrow(),
1957 None => return false,
1958 };
1959 row_group.base.style.get_inherited_box().visibility == Visibility::Collapse
1960 }
1961
1962 fn is_column_collapsed(&self, column_index: usize) -> bool {
1963 let Some(column) = &self.table.columns.get(column_index) else {
1964 return false;
1965 };
1966 let column = column.borrow();
1967 if column.base.style.get_inherited_box().visibility == Visibility::Collapse {
1968 return true;
1969 }
1970 let col_group = match column.group_index {
1971 Some(group_index) => self.table.column_groups[group_index].borrow(),
1972 None => return false,
1973 };
1974 col_group.base.style.get_inherited_box().visibility == Visibility::Collapse
1975 }
1976
1977 #[allow(clippy::too_many_arguments)]
1978 fn do_final_cell_layout(
1979 &mut self,
1980 row_index: usize,
1981 column_index: usize,
1982 dimensions: &TableAndTrackDimensions,
1983 baselines: &mut Baselines,
1984 row_fragment_layout: &mut RowFragmentLayout,
1985 row_group_fragment_layout: Option<&mut RowGroupFragmentLayout>,
1986 positioning_context_for_table: &mut PositioningContext,
1987 is_collapsed: bool,
1988 ) {
1989 let row_group_positioning_context =
1992 row_group_fragment_layout.and_then(|layout| layout.positioning_context.as_mut());
1993 let positioning_context = row_fragment_layout
1994 .positioning_context
1995 .as_mut()
1996 .or(row_group_positioning_context)
1997 .unwrap_or(positioning_context_for_table);
1998
1999 let layout = match self.cells_laid_out[row_index][column_index].take() {
2000 Some(layout) => layout,
2001 None => {
2002 return;
2003 },
2004 };
2005 let cell = match self.table.slots[row_index][column_index] {
2006 TableSlot::Cell(ref cell) => cell,
2007 _ => {
2008 warn!("Did not find a non-spanned cell at index with layout.");
2009 return;
2010 },
2011 }
2012 .borrow();
2013
2014 let row_block_offset = row_fragment_layout.rect.start_corner.block;
2016 let row_baseline = self.row_baselines[row_index];
2017 if cell.content_alignment() == CellContentAlignment::Baseline && !layout.is_empty() {
2018 let baseline = row_block_offset + row_baseline;
2019 if row_index == 0 {
2020 baselines.first = Some(baseline);
2021 }
2022 baselines.last = Some(baseline);
2023 }
2024 let mut row_relative_cell_rect = dimensions.get_cell_rect(
2025 TableSlotCoordinates::new(column_index, row_index),
2026 cell.rowspan,
2027 cell.colspan,
2028 );
2029 row_relative_cell_rect.start_corner -= row_fragment_layout.rect.start_corner;
2030 let mut fragment = cell.create_fragment(
2031 layout,
2032 row_relative_cell_rect,
2033 row_baseline,
2034 positioning_context,
2035 &self.table.style,
2036 &row_fragment_layout.containing_block,
2037 is_collapsed,
2038 );
2039
2040 let make_relative_to_row_start = |mut rect: LogicalRect<Au>| {
2049 rect.start_corner -= row_fragment_layout.rect.start_corner;
2050 let writing_mode = self.table.style.writing_mode;
2051 PhysicalRect::new(
2052 if writing_mode.is_horizontal() {
2053 PhysicalPoint::new(rect.start_corner.inline, rect.start_corner.block)
2054 } else {
2055 PhysicalPoint::new(rect.start_corner.block, rect.start_corner.inline)
2056 },
2057 rect.size.to_physical_size(writing_mode),
2058 )
2059 };
2060
2061 let column = self.table.columns.get(column_index);
2062 let column_group = column
2063 .and_then(|column| column.borrow().group_index)
2064 .and_then(|index| self.table.column_groups.get(index));
2065 if let Some(column_group) = column_group {
2066 let column_group = column_group.borrow();
2067 let rect = make_relative_to_row_start(dimensions.get_column_group_rect(&column_group));
2068 fragment.add_extra_background(ExtraBackground {
2069 style: column_group.shared_background_style.clone(),
2070 rect,
2071 })
2072 }
2073 if let Some(column) = column {
2074 let column = column.borrow();
2075 if !column.is_anonymous {
2076 let rect = make_relative_to_row_start(dimensions.get_column_rect(column_index));
2077 fragment.add_extra_background(ExtraBackground {
2078 style: column.shared_background_style.clone(),
2079 rect,
2080 })
2081 }
2082 }
2083 let row = self.table.rows.get(row_index);
2084 let row_group = row
2085 .and_then(|row| row.borrow().group_index)
2086 .and_then(|index| self.table.row_groups.get(index));
2087 if let Some(row_group) = row_group {
2088 let rect =
2089 make_relative_to_row_start(dimensions.get_row_group_rect(&row_group.borrow()));
2090 fragment.add_extra_background(ExtraBackground {
2091 style: row_group.borrow().shared_background_style.clone(),
2092 rect,
2093 })
2094 }
2095 if let Some(row) = row {
2096 let row = row.borrow();
2097 let rect = make_relative_to_row_start(row_fragment_layout.rect);
2098 fragment.add_extra_background(ExtraBackground {
2099 style: row.shared_background_style.clone(),
2100 rect,
2101 })
2102 }
2103
2104 let fragment = Fragment::Box(fragment.into());
2105 cell.context.base.set_fragment(fragment.clone());
2106 row_fragment_layout.fragments.push(fragment);
2107 }
2108
2109 fn make_fragments_for_columns_and_column_groups(
2110 &self,
2111 dimensions: &TableAndTrackDimensions,
2112 fragments: &mut Vec<Fragment>,
2113 ) {
2114 for column_group in self.table.column_groups.iter() {
2115 let column_group = column_group.borrow();
2116 if !column_group.is_empty() {
2117 let fragment = Fragment::Positioning(PositioningFragment::new_empty(
2118 column_group.base.base_fragment_info,
2119 dimensions
2120 .get_column_group_rect(&column_group)
2121 .as_physical(None),
2122 column_group.base.style.clone(),
2123 ));
2124 column_group.base.set_fragment(fragment.clone());
2125 fragments.push(fragment);
2126 }
2127 }
2128
2129 for (column_index, column) in self.table.columns.iter().enumerate() {
2130 let column = column.borrow();
2131 let fragment = Fragment::Positioning(PositioningFragment::new_empty(
2132 column.base.base_fragment_info,
2133 dimensions.get_column_rect(column_index).as_physical(None),
2134 column.base.style.clone(),
2135 ));
2136 column.base.set_fragment(fragment.clone());
2137 fragments.push(fragment);
2138 }
2139 }
2140
2141 fn compute_border_collapse(&mut self, writing_mode: WritingMode) {
2142 if self.table.style.get_inherited_table().border_collapse != BorderCollapse::Collapse {
2143 self.collapsed_borders = None;
2144 return;
2145 }
2146
2147 let mut collapsed_borders = LogicalVec2 {
2148 block: vec![
2149 vec![Default::default(); self.table.size.width];
2150 self.table.size.height + 1
2151 ],
2152 inline: vec![
2153 vec![Default::default(); self.table.size.height];
2154 self.table.size.width + 1
2155 ],
2156 };
2157
2158 let apply_border = |collapsed_borders: &mut CollapsedBorders,
2159 layout_style: &LayoutStyle,
2160 block: &Range<usize>,
2161 inline: &Range<usize>| {
2162 let border = CollapsedBorder::from_layout_style(layout_style, writing_mode);
2163 border
2164 .block_start
2165 .max_assign_to_slice(&mut collapsed_borders.block[block.start][inline.clone()]);
2166 border
2167 .block_end
2168 .max_assign_to_slice(&mut collapsed_borders.block[block.end][inline.clone()]);
2169 border
2170 .inline_start
2171 .max_assign_to_slice(&mut collapsed_borders.inline[inline.start][block.clone()]);
2172 border
2173 .inline_end
2174 .max_assign_to_slice(&mut collapsed_borders.inline[inline.end][block.clone()]);
2175 };
2176 let hide_inner_borders = |collapsed_borders: &mut CollapsedBorders,
2177 block: &Range<usize>,
2178 inline: &Range<usize>| {
2179 for x in inline.clone() {
2180 for y in block.clone() {
2181 if x != inline.start {
2182 collapsed_borders.inline[x][y].hide();
2183 }
2184 if y != block.start {
2185 collapsed_borders.block[y][x].hide();
2186 }
2187 }
2188 }
2189 };
2190 let all_rows = 0..self.table.size.height;
2191 let all_columns = 0..self.table.size.width;
2192 for row_index in all_rows.clone() {
2193 for column_index in all_columns.clone() {
2194 let cell = match self.table.slots[row_index][column_index] {
2195 TableSlot::Cell(ref cell) => cell,
2196 _ => continue,
2197 }
2198 .borrow();
2199 let block_range = row_index..row_index + cell.rowspan;
2200 let inline_range = column_index..column_index + cell.colspan;
2201 hide_inner_borders(&mut collapsed_borders, &block_range, &inline_range);
2202 apply_border(
2203 &mut collapsed_borders,
2204 &cell.context.layout_style(),
2205 &block_range,
2206 &inline_range,
2207 );
2208 }
2209 }
2210 for (row_index, row) in self.table.rows.iter().enumerate() {
2211 let row = row.borrow();
2212 apply_border(
2213 &mut collapsed_borders,
2214 &row.layout_style(),
2215 &(row_index..row_index + 1),
2216 &all_columns,
2217 );
2218 }
2219 for row_group in &self.table.row_groups {
2220 let row_group = row_group.borrow();
2221 apply_border(
2222 &mut collapsed_borders,
2223 &row_group.layout_style(),
2224 &row_group.track_range,
2225 &all_columns,
2226 );
2227 }
2228 for (column_index, column) in self.table.columns.iter().enumerate() {
2229 let column = column.borrow();
2230 apply_border(
2231 &mut collapsed_borders,
2232 &column.layout_style(),
2233 &all_rows,
2234 &(column_index..column_index + 1),
2235 );
2236 }
2237 for column_group in &self.table.column_groups {
2238 let column_group = column_group.borrow();
2239 apply_border(
2240 &mut collapsed_borders,
2241 &column_group.layout_style(),
2242 &all_rows,
2243 &column_group.track_range,
2244 );
2245 }
2246 apply_border(
2247 &mut collapsed_borders,
2248 &self.table.layout_style_for_grid(),
2249 &all_rows,
2250 &all_columns,
2251 );
2252
2253 self.collapsed_borders = Some(collapsed_borders);
2254 }
2255
2256 fn get_collapsed_border_widths_for_area(
2257 &self,
2258 area: LogicalSides<usize>,
2259 ) -> Option<LogicalSides<Au>> {
2260 let collapsed_borders = self.collapsed_borders.as_ref()?;
2261 let columns = || area.inline_start..area.inline_end;
2262 let rows = || area.block_start..area.block_end;
2263 let max_width = |slice: &[CollapsedBorder]| {
2264 let slice_widths = slice.iter().map(|collapsed_border| collapsed_border.width);
2265 slice_widths.max().unwrap_or_default()
2266 };
2267 Some(area.map_inline_and_block_axes(
2268 |column| max_width(&collapsed_borders.inline[*column][rows()]) / 2,
2269 |row| max_width(&collapsed_borders.block[*row][columns()]) / 2,
2270 ))
2271 }
2272}
2273
2274struct RowFragmentLayout<'a> {
2275 row: &'a TableTrack,
2276 rect: LogicalRect<Au>,
2277 containing_block: ContainingBlock<'a>,
2278 positioning_context: Option<PositioningContext>,
2279 fragments: Vec<Fragment>,
2280}
2281
2282impl<'a> RowFragmentLayout<'a> {
2283 fn new(
2284 table_row: &'a TableTrack,
2285 index: usize,
2286 dimensions: &TableAndTrackDimensions,
2287 table_style: &'a ComputedValues,
2288 ) -> Self {
2289 let rect = dimensions.get_row_rect(index);
2290 let containing_block = ContainingBlock {
2291 size: ContainingBlockSize {
2292 inline: rect.size.inline,
2293 block: SizeConstraint::Definite(rect.size.block),
2294 },
2295 style: table_style,
2296 };
2297 Self {
2298 row: table_row,
2299 rect,
2300 positioning_context: PositioningContext::new_for_layout_box_base(&table_row.base),
2301 containing_block,
2302 fragments: Vec::new(),
2303 }
2304 }
2305 fn finish(
2306 mut self,
2307 layout_context: &LayoutContext,
2308 table_positioning_context: &mut PositioningContext,
2309 containing_block_for_logical_conversion: &ContainingBlock,
2310 containing_block_for_children: &ContainingBlock,
2311 row_group_fragment_layout: &mut Option<RowGroupFragmentLayout>,
2312 ) -> Fragment {
2313 if self.positioning_context.is_some() {
2314 self.rect.start_corner +=
2315 relative_adjustement(&self.row.base.style, containing_block_for_children);
2316 }
2317
2318 let (inline_size, block_size) = if let Some(row_group_layout) = row_group_fragment_layout {
2319 self.rect.start_corner -= row_group_layout.rect.start_corner;
2320 (
2321 row_group_layout.rect.size.inline,
2322 SizeConstraint::Definite(row_group_layout.rect.size.block),
2323 )
2324 } else {
2325 (
2326 containing_block_for_logical_conversion.size.inline,
2327 containing_block_for_logical_conversion.size.block,
2328 )
2329 };
2330
2331 let row_group_containing_block = ContainingBlock {
2332 size: ContainingBlockSize {
2333 inline: inline_size,
2334 block: block_size,
2335 },
2336 style: containing_block_for_logical_conversion.style,
2337 };
2338
2339 let mut row_fragment = BoxFragment::new(
2340 self.row.base.base_fragment_info,
2341 self.row.base.style.clone(),
2342 self.fragments,
2343 self.rect.as_physical(Some(&row_group_containing_block)),
2344 PhysicalSides::zero(), PhysicalSides::zero(), PhysicalSides::zero(), None, );
2349 row_fragment.set_does_not_paint_background();
2350
2351 if let Some(mut row_positioning_context) = self.positioning_context.take() {
2352 row_positioning_context.layout_collected_children(layout_context, &mut row_fragment);
2353
2354 let parent_positioning_context = row_group_fragment_layout
2355 .as_mut()
2356 .and_then(|layout| layout.positioning_context.as_mut())
2357 .unwrap_or(table_positioning_context);
2358 parent_positioning_context.append(row_positioning_context);
2359 }
2360
2361 let fragment = Fragment::Box(row_fragment.into());
2362 self.row.base.set_fragment(fragment.clone());
2363 fragment
2364 }
2365}
2366
2367struct RowGroupFragmentLayout {
2368 row_group: ArcRefCell<TableTrackGroup>,
2369 rect: LogicalRect<Au>,
2370 positioning_context: Option<PositioningContext>,
2371 index: usize,
2372 fragments: Vec<Fragment>,
2373}
2374
2375impl RowGroupFragmentLayout {
2376 fn new(
2377 row_group: ArcRefCell<TableTrackGroup>,
2378 index: usize,
2379 dimensions: &TableAndTrackDimensions,
2380 ) -> Self {
2381 let (rect, positioning_context) = {
2382 let row_group = row_group.borrow();
2383 (
2384 dimensions.get_row_group_rect(&row_group),
2385 PositioningContext::new_for_layout_box_base(&row_group.base),
2386 )
2387 };
2388 Self {
2389 row_group,
2390 rect,
2391 positioning_context,
2392 index,
2393 fragments: Vec::new(),
2394 }
2395 }
2396
2397 fn finish(
2398 mut self,
2399 layout_context: &LayoutContext,
2400 table_positioning_context: &mut PositioningContext,
2401 containing_block_for_logical_conversion: &ContainingBlock,
2402 containing_block_for_children: &ContainingBlock,
2403 ) -> Fragment {
2404 let row_group = self.row_group.borrow();
2405 if self.positioning_context.is_some() {
2406 self.rect.start_corner +=
2407 relative_adjustement(&row_group.base.style, containing_block_for_children);
2408 }
2409
2410 let mut row_group_fragment = BoxFragment::new(
2411 row_group.base.base_fragment_info,
2412 row_group.base.style.clone(),
2413 self.fragments,
2414 self.rect
2415 .as_physical(Some(containing_block_for_logical_conversion)),
2416 PhysicalSides::zero(), PhysicalSides::zero(), PhysicalSides::zero(), None, );
2421 row_group_fragment.set_does_not_paint_background();
2422
2423 if let Some(mut row_positioning_context) = self.positioning_context.take() {
2424 row_positioning_context
2425 .layout_collected_children(layout_context, &mut row_group_fragment);
2426 table_positioning_context.append(row_positioning_context);
2427 }
2428
2429 let fragment = Fragment::Box(row_group_fragment.into());
2430 row_group.base.set_fragment(fragment.clone());
2431 fragment
2432 }
2433}
2434
2435struct TableAndTrackDimensions {
2436 table_rect: LogicalRect<Au>,
2438 table_cells_rect: LogicalRect<Au>,
2441 row_dimensions: Vec<(Au, Au)>,
2443 column_dimensions: Vec<(Au, Au)>,
2445}
2446
2447impl TableAndTrackDimensions {
2448 fn new(table_layout: &TableLayout) -> Self {
2449 let border_spacing = table_layout.table.border_spacing();
2450
2451 let fallback_inline_size = table_layout.assignable_width;
2453 let fallback_block_size = table_layout.final_table_height;
2454
2455 let mut column_dimensions = Vec::new();
2456 let mut column_offset = Au::zero();
2457 for column_index in 0..table_layout.table.size.width {
2458 if table_layout.is_column_collapsed(column_index) {
2459 column_dimensions.push((column_offset, column_offset));
2460 continue;
2461 }
2462 let start_offset = column_offset + border_spacing.inline;
2463 let end_offset = start_offset + table_layout.distributed_column_widths[column_index];
2464 column_dimensions.push((start_offset, end_offset));
2465 column_offset = end_offset;
2466 }
2467 column_offset += if table_layout.table.size.width == 0 {
2468 fallback_inline_size
2469 } else {
2470 border_spacing.inline
2471 };
2472
2473 let mut row_dimensions = Vec::new();
2474 let mut row_offset = Au::zero();
2475 for row_index in 0..table_layout.table.size.height {
2476 if table_layout.is_row_collapsed(row_index) {
2477 row_dimensions.push((row_offset, row_offset));
2478 continue;
2479 }
2480 let start_offset = row_offset + border_spacing.block;
2481 let end_offset = start_offset + table_layout.row_sizes[row_index];
2482 row_dimensions.push((start_offset, end_offset));
2483 row_offset = end_offset;
2484 }
2485 row_offset += if table_layout.table.size.height == 0 {
2486 fallback_block_size
2487 } else {
2488 border_spacing.block
2489 };
2490
2491 let table_start_corner = LogicalVec2 {
2492 inline: column_dimensions.first().map_or_else(Au::zero, |v| v.0),
2493 block: row_dimensions.first().map_or_else(Au::zero, |v| v.0),
2494 };
2495 let table_size = LogicalVec2 {
2496 inline: column_dimensions
2497 .last()
2498 .map_or(fallback_inline_size, |v| v.1),
2499 block: row_dimensions.last().map_or(fallback_block_size, |v| v.1),
2500 } - table_start_corner;
2501 let table_cells_rect = LogicalRect {
2502 start_corner: table_start_corner,
2503 size: table_size,
2504 };
2505
2506 let table_rect = LogicalRect {
2507 start_corner: LogicalVec2::zero(),
2508 size: LogicalVec2 {
2509 inline: column_offset,
2510 block: row_offset,
2511 },
2512 };
2513
2514 Self {
2515 table_rect,
2516 table_cells_rect,
2517 row_dimensions,
2518 column_dimensions,
2519 }
2520 }
2521
2522 fn get_row_rect(&self, row_index: usize) -> LogicalRect<Au> {
2523 let mut row_rect = self.table_cells_rect;
2524 let row_dimensions = self.row_dimensions[row_index];
2525 row_rect.start_corner.block = row_dimensions.0;
2526 row_rect.size.block = row_dimensions.1 - row_dimensions.0;
2527 row_rect
2528 }
2529
2530 fn get_column_rect(&self, column_index: usize) -> LogicalRect<Au> {
2531 let mut row_rect = self.table_cells_rect;
2532 let column_dimensions = self.column_dimensions[column_index];
2533 row_rect.start_corner.inline = column_dimensions.0;
2534 row_rect.size.inline = column_dimensions.1 - column_dimensions.0;
2535 row_rect
2536 }
2537
2538 fn get_row_group_rect(&self, row_group: &TableTrackGroup) -> LogicalRect<Au> {
2539 if row_group.is_empty() {
2540 return LogicalRect::zero();
2541 }
2542
2543 let mut row_group_rect = self.table_cells_rect;
2544 let block_start = self.row_dimensions[row_group.track_range.start].0;
2545 let block_end = self.row_dimensions[row_group.track_range.end - 1].1;
2546 row_group_rect.start_corner.block = block_start;
2547 row_group_rect.size.block = block_end - block_start;
2548 row_group_rect
2549 }
2550
2551 fn get_column_group_rect(&self, column_group: &TableTrackGroup) -> LogicalRect<Au> {
2552 if column_group.is_empty() {
2553 return LogicalRect::zero();
2554 }
2555
2556 let mut column_group_rect = self.table_cells_rect;
2557 let inline_start = self.column_dimensions[column_group.track_range.start].0;
2558 let inline_end = self.column_dimensions[column_group.track_range.end - 1].1;
2559 column_group_rect.start_corner.inline = inline_start;
2560 column_group_rect.size.inline = inline_end - inline_start;
2561 column_group_rect
2562 }
2563
2564 fn get_cell_rect(
2565 &self,
2566 coordinates: TableSlotCoordinates,
2567 rowspan: usize,
2568 colspan: usize,
2569 ) -> LogicalRect<Au> {
2570 let start_corner = LogicalVec2 {
2571 inline: self.column_dimensions[coordinates.x].0,
2572 block: self.row_dimensions[coordinates.y].0,
2573 };
2574 let size = LogicalVec2 {
2575 inline: self.column_dimensions[coordinates.x + colspan - 1].1,
2576 block: self.row_dimensions[coordinates.y + rowspan - 1].1,
2577 } - start_corner;
2578 LogicalRect { start_corner, size }
2579 }
2580}
2581
2582impl Table {
2583 fn border_spacing(&self) -> LogicalVec2<Au> {
2584 if self.style.clone_border_collapse() == BorderCollapse::Collapse {
2585 LogicalVec2::zero()
2586 } else {
2587 let border_spacing = self.style.clone_border_spacing();
2588 LogicalVec2 {
2589 inline: border_spacing.horizontal(),
2590 block: border_spacing.vertical(),
2591 }
2592 }
2593 }
2594
2595 fn total_border_spacing(&self) -> LogicalVec2<Au> {
2596 let border_spacing = self.border_spacing();
2597 LogicalVec2 {
2598 inline: if self.size.width > 0 {
2599 border_spacing.inline * (self.size.width as i32 + 1)
2600 } else {
2601 Au::zero()
2602 },
2603 block: if self.size.height > 0 {
2604 border_spacing.block * (self.size.height as i32 + 1)
2605 } else {
2606 Au::zero()
2607 },
2608 }
2609 }
2610
2611 fn get_column_measure_for_column_at_index(
2612 &self,
2613 writing_mode: WritingMode,
2614 column_index: usize,
2615 is_in_fixed_mode: bool,
2616 ) -> CellOrTrackMeasure {
2617 let column = match self.columns.get(column_index) {
2618 Some(column) => column,
2619 None => return CellOrTrackMeasure::zero(),
2620 }
2621 .borrow();
2622
2623 let CellOrColumnOuterSizes {
2624 preferred: preferred_size,
2625 min: min_size,
2626 max: max_size,
2627 percentage: percentage_size,
2628 } = CellOrColumnOuterSizes::new(
2629 &column.base.style,
2630 writing_mode,
2631 &Default::default(),
2632 is_in_fixed_mode,
2633 );
2634
2635 CellOrTrackMeasure {
2636 content_sizes: ContentSizes {
2637 min_content: min_size.inline,
2642 max_content: preferred_size
2646 .inline
2647 .clamp_between_extremums(min_size.inline, max_size.inline),
2648 },
2649 percentage: percentage_size.inline,
2650 }
2651 }
2652
2653 fn get_row_measure_for_row_at_index(
2654 &self,
2655 writing_mode: WritingMode,
2656 row_index: usize,
2657 ) -> CellOrTrackMeasure {
2658 let row = match self.rows.get(row_index) {
2659 Some(row) => row,
2660 None => return CellOrTrackMeasure::zero(),
2661 };
2662
2663 let row = row.borrow();
2667 let size = row.base.style.box_size(writing_mode);
2668 let max_size = row.base.style.max_box_size(writing_mode);
2669 let percentage_contribution = get_size_percentage_contribution(&size, &max_size);
2670
2671 CellOrTrackMeasure {
2672 content_sizes: size
2673 .block
2674 .to_numeric()
2675 .and_then(|size| size.to_length())
2676 .map_or_else(Au::zero, Au::from)
2677 .into(),
2678 percentage: percentage_contribution.block,
2679 }
2680 }
2681
2682 pub(crate) fn layout(
2683 &self,
2684 layout_context: &LayoutContext,
2685 positioning_context: &mut PositioningContext,
2686 containing_block_for_children: &ContainingBlock,
2687 containing_block_for_table: &ContainingBlock,
2688 ) -> IndependentFormattingContextLayoutResult {
2689 TableLayout::new(self).layout(
2690 layout_context,
2691 positioning_context,
2692 containing_block_for_children,
2693 containing_block_for_table,
2694 )
2695 }
2696
2697 pub(crate) fn attached_to_tree(&self, layout_box: WeakLayoutBox) {
2698 for caption in &self.captions {
2699 caption
2700 .borrow_mut()
2701 .context
2702 .base
2703 .parent_box
2704 .replace(layout_box.clone());
2705 }
2706 for row_group in &self.row_groups {
2707 row_group
2708 .borrow_mut()
2709 .base
2710 .parent_box
2711 .replace(layout_box.clone());
2712 }
2713 for column_group in &self.column_groups {
2714 column_group
2715 .borrow_mut()
2716 .base
2717 .parent_box
2718 .replace(layout_box.clone());
2719 }
2720 for row in &self.rows {
2721 let row = &mut *row.borrow_mut();
2722 if let Some(group_index) = row.group_index {
2723 row.base.parent_box.replace(WeakLayoutBox::TableLevelBox(
2724 WeakTableLevelBox::TrackGroup(self.row_groups[group_index].downgrade()),
2725 ));
2726 } else {
2727 row.base.parent_box.replace(layout_box.clone());
2728 }
2729 }
2730 for column in &self.columns {
2731 let column = &mut *column.borrow_mut();
2732 if let Some(group_index) = column.group_index {
2733 column.base.parent_box.replace(WeakLayoutBox::TableLevelBox(
2734 WeakTableLevelBox::TrackGroup(self.column_groups[group_index].downgrade()),
2735 ));
2736 } else {
2737 column.base.parent_box.replace(layout_box.clone());
2738 }
2739 }
2740 for row_index in 0..self.size.height {
2741 let row = WeakLayoutBox::TableLevelBox(WeakTableLevelBox::Track(
2742 self.rows[row_index].downgrade(),
2743 ));
2744 for column_index in 0..self.size.width {
2745 if let TableSlot::Cell(ref cell) = self.slots[row_index][column_index] {
2746 cell.borrow_mut()
2747 .context
2748 .base
2749 .parent_box
2750 .replace(row.clone());
2751 }
2752 }
2753 }
2754 }
2755}
2756
2757impl ComputeInlineContentSizes for Table {
2758 #[servo_tracing::instrument(name = "Table::compute_inline_content_sizes", skip_all)]
2759 fn compute_inline_content_sizes(
2760 &self,
2761 layout_context: &LayoutContext,
2762 constraint_space: &ConstraintSpace,
2763 ) -> InlineContentSizesResult {
2764 let writing_mode = constraint_space.style.writing_mode;
2765 let mut layout = TableLayout::new(self);
2766 layout.compute_border_collapse(writing_mode);
2767 layout.pbm = self
2768 .layout_style(Some(&layout))
2769 .padding_border_margin_with_writing_mode_and_containing_block_inline_size(
2770 writing_mode,
2771 Au::zero(),
2772 );
2773 layout.compute_measures(layout_context, writing_mode);
2774
2775 let grid_content_sizes = layout.compute_grid_min_max();
2776
2777 let caption_content_sizes = ContentSizes::from(
2781 layout.compute_caption_minimum_inline_size(layout_context) -
2782 layout.pbm.padding_border_sums.inline,
2783 );
2784
2785 InlineContentSizesResult {
2786 sizes: grid_content_sizes.max(caption_content_sizes),
2787 depends_on_block_constraints: false,
2788 }
2789 }
2790}
2791
2792impl Table {
2793 #[inline]
2794 pub(crate) fn layout_style<'a>(
2795 &'a self,
2796 layout: Option<&'a TableLayout<'a>>,
2797 ) -> LayoutStyle<'a> {
2798 LayoutStyle::Table(TableLayoutStyle {
2799 table: self,
2800 layout,
2801 })
2802 }
2803
2804 #[inline]
2805 pub(crate) fn layout_style_for_grid(&self) -> LayoutStyle<'_> {
2806 LayoutStyle::Default(&self.grid_style)
2807 }
2808}
2809
2810impl TableTrack {
2811 #[inline]
2812 pub(crate) fn layout_style(&self) -> LayoutStyle<'_> {
2813 LayoutStyle::Default(&self.base.style)
2814 }
2815}
2816
2817impl TableTrackGroup {
2818 #[inline]
2819 pub(crate) fn layout_style(&self) -> LayoutStyle<'_> {
2820 LayoutStyle::Default(&self.base.style)
2821 }
2822}
2823
2824impl TableLayoutStyle<'_> {
2825 #[inline]
2826 pub(crate) fn style(&self) -> &ComputedValues {
2827 &self.table.style
2828 }
2829
2830 #[inline]
2831 pub(crate) fn collapses_borders(&self) -> bool {
2832 self.style().get_inherited_table().border_collapse == BorderCollapse::Collapse
2833 }
2834
2835 pub(crate) fn halved_collapsed_border_widths(&self) -> LogicalSides<Au> {
2836 debug_assert!(self.collapses_borders());
2837 let area = LogicalSides {
2838 inline_start: 0,
2839 inline_end: self.table.size.width,
2840 block_start: 0,
2841 block_end: self.table.size.height,
2842 };
2843 if let Some(layout) = self.layout {
2844 layout.get_collapsed_border_widths_for_area(area)
2845 } else {
2846 let mut layout = TableLayout::new(self.table);
2848 layout.compute_border_collapse(self.style().writing_mode);
2849 layout.get_collapsed_border_widths_for_area(area)
2850 }
2851 .expect("Collapsed borders should be computed")
2852 }
2853}
2854
2855impl TableSlotCell {
2856 fn content_alignment(&self) -> CellContentAlignment {
2857 let style_box = self.context.base.style.get_box();
2861 match style_box.baseline_shift {
2862 BaselineShift::Keyword(BaselineShiftKeyword::Top) => CellContentAlignment::Top,
2863 BaselineShift::Keyword(BaselineShiftKeyword::Bottom) => CellContentAlignment::Bottom,
2864 _ => match style_box.alignment_baseline {
2865 AlignmentBaseline::Middle => CellContentAlignment::Middle,
2866 _ => CellContentAlignment::Baseline,
2867 },
2868 }
2869 }
2870
2871 #[allow(clippy::too_many_arguments)]
2872 fn create_fragment(
2873 &self,
2874 mut layout: CellLayout,
2875 cell_rect: LogicalRect<Au>,
2876 cell_baseline: Au,
2877 positioning_context: &mut PositioningContext,
2878 table_style: &ComputedValues,
2879 containing_block: &ContainingBlock,
2880 is_collapsed: bool,
2881 ) -> BoxFragment {
2882 use style::Zero as StyleZero;
2884
2885 let cell_content_rect = cell_rect.deflate(&(layout.padding + layout.border));
2886 let content_block_size = layout.layout.content_block_size;
2887 let free_space = || Au::zero().max(cell_content_rect.size.block - content_block_size);
2888 let vertical_align_offset = match self.content_alignment() {
2889 CellContentAlignment::Top => Au::zero(),
2890 CellContentAlignment::Bottom => free_space(),
2891 CellContentAlignment::Middle => free_space().scale_by(0.5),
2892 CellContentAlignment::Baseline => {
2893 cell_baseline -
2894 (layout.padding.block_start + layout.border.block_start) -
2895 layout.ascent()
2896 },
2897 };
2898
2899 let mut base_fragment_info = self.context.base.base_fragment_info;
2900 if self.context.base.style.get_inherited_table().empty_cells == EmptyCells::Hide &&
2901 table_style.get_inherited_table().border_collapse != BorderCollapse::Collapse &&
2902 layout.is_empty_for_empty_cells()
2903 {
2904 base_fragment_info.flags.insert(FragmentFlags::DO_NOT_PAINT);
2905 }
2906
2907 if is_collapsed {
2908 base_fragment_info.flags.insert(FragmentFlags::IS_COLLAPSED);
2909 }
2910
2911 let mut vertical_align_fragment_rect = cell_content_rect;
2913 vertical_align_fragment_rect.start_corner = LogicalVec2 {
2914 inline: Au::zero(),
2915 block: vertical_align_offset,
2916 };
2917 let vertical_align_fragment = PositioningFragment::new_anonymous(
2918 self.context.base.style.clone(),
2919 vertical_align_fragment_rect.as_physical(None),
2920 layout.layout.fragments,
2921 );
2922
2923 let physical_cell_rect = cell_content_rect.as_physical(Some(containing_block));
2931 layout
2932 .positioning_context
2933 .adjust_static_position_of_hoisted_fragments_with_offset(
2934 &physical_cell_rect.origin.to_vector(),
2935 PositioningContextLength::zero(),
2936 );
2937 positioning_context.append(layout.positioning_context);
2938
2939 let specific_layout_info = (table_style.get_inherited_table().border_collapse ==
2940 BorderCollapse::Collapse)
2941 .then_some(SpecificLayoutInfo::TableCellWithCollapsedBorders);
2942
2943 BoxFragment::new(
2944 base_fragment_info,
2945 self.context.base.style.clone(),
2946 vec![Fragment::Positioning(vertical_align_fragment)],
2947 physical_cell_rect,
2948 layout.padding.to_physical(table_style.writing_mode),
2949 layout.border.to_physical(table_style.writing_mode),
2950 PhysicalSides::zero(), specific_layout_info,
2952 )
2953 .with_baselines(layout.layout.baselines)
2954 }
2955}
2956
2957fn get_size_percentage_contribution(
2958 size: &LogicalVec2<Size<ComputedLengthPercentage>>,
2959 max_size: &LogicalVec2<Size<ComputedLengthPercentage>>,
2960) -> LogicalVec2<Option<Percentage>> {
2961 LogicalVec2 {
2969 inline: max_two_optional_percentages(
2970 size.inline.to_percentage(),
2971 max_size.inline.to_percentage(),
2972 ),
2973 block: max_two_optional_percentages(
2974 size.block.to_percentage(),
2975 max_size.block.to_percentage(),
2976 ),
2977 }
2978}
2979
2980struct CellOrColumnOuterSizes {
2981 min: LogicalVec2<Au>,
2982 preferred: LogicalVec2<Au>,
2983 max: LogicalVec2<Option<Au>>,
2984 percentage: LogicalVec2<Option<Percentage>>,
2985}
2986
2987impl CellOrColumnOuterSizes {
2988 fn new(
2989 style: &Arc<ComputedValues>,
2990 writing_mode: WritingMode,
2991 padding_border_sums: &LogicalVec2<Au>,
2992 is_in_fixed_mode: bool,
2993 ) -> Self {
2994 let box_sizing = style.get_position().box_sizing;
2995 let outer_size = |size: LogicalVec2<Au>| match box_sizing {
2996 BoxSizing::ContentBox => size + *padding_border_sums,
2997 BoxSizing::BorderBox => LogicalVec2 {
2998 inline: size.inline.max(padding_border_sums.inline),
2999 block: size.block.max(padding_border_sums.block),
3000 },
3001 };
3002
3003 let outer_option_size = |size: LogicalVec2<Option<Au>>| match box_sizing {
3004 BoxSizing::ContentBox => size.map_inline_and_block_axes(
3005 |inline| inline.map(|inline| inline + padding_border_sums.inline),
3006 |block| block.map(|block| block + padding_border_sums.block),
3007 ),
3008 BoxSizing::BorderBox => size.map_inline_and_block_axes(
3009 |inline| inline.map(|inline| inline.max(padding_border_sums.inline)),
3010 |block| block.map(|block| block.max(padding_border_sums.block)),
3011 ),
3012 };
3013
3014 let get_size_for_axis = |size: &Size<ComputedLengthPercentage>| {
3015 size.to_numeric()
3018 .and_then(|length_percentage| length_percentage.to_length())
3019 .map(Au::from)
3020 };
3021
3022 let size = style.box_size(writing_mode);
3023 if is_in_fixed_mode {
3024 return Self {
3025 percentage: size.map(|v| v.to_percentage()),
3026 preferred: outer_option_size(size.map(get_size_for_axis))
3027 .map(|v| v.unwrap_or_default()),
3028 min: LogicalVec2::default(),
3029 max: LogicalVec2::default(),
3030 };
3031 }
3032
3033 let min_size = style.min_box_size(writing_mode);
3034 let max_size = style.max_box_size(writing_mode);
3035
3036 Self {
3037 min: outer_size(min_size.map(|v| get_size_for_axis(v).unwrap_or_default())),
3038 preferred: outer_size(size.map(|v| get_size_for_axis(v).unwrap_or_default())),
3039 max: outer_option_size(max_size.map(get_size_for_axis)),
3040 percentage: get_size_percentage_contribution(&size, &max_size),
3041 }
3042 }
3043}
3044
3045struct RowspanToDistribute<'a> {
3046 coordinates: TableSlotCoordinates,
3047 cell: AtomicRef<'a, TableSlotCell>,
3048 measure: &'a CellOrTrackMeasure,
3049}
3050
3051impl RowspanToDistribute<'_> {
3052 fn range(&self) -> Range<usize> {
3053 self.coordinates.y..self.coordinates.y + self.cell.rowspan
3054 }
3055
3056 fn fully_encloses(&self, other: &RowspanToDistribute) -> bool {
3057 other.coordinates.y > self.coordinates.y && other.range().end < self.range().end
3058 }
3059}
3060
3061#[derive(Debug)]
3064struct ColspanToDistribute {
3065 starting_column: usize,
3066 span: usize,
3067 content_sizes: ContentSizes,
3068 percentage: Option<Percentage>,
3069}
3070
3071impl ColspanToDistribute {
3072 fn comparison_for_sort(a: &Self, b: &Self) -> Ordering {
3076 a.span
3077 .cmp(&b.span)
3078 .then_with(|| a.starting_column.cmp(&b.starting_column))
3079 }
3080
3081 fn range(&self) -> Range<usize> {
3082 self.starting_column..self.starting_column + self.span
3083 }
3084}
3085
3086#[cfg(test)]
3087mod test {
3088 use app_units::MIN_AU;
3089
3090 use super::*;
3091 use crate::sizing::ContentSizes;
3092
3093 #[test]
3094 fn test_colspan_to_distribute_first_sort_by_span() {
3095 let a = ColspanToDistribute {
3096 starting_column: 0,
3097 span: 0,
3098 content_sizes: ContentSizes {
3099 min_content: MIN_AU,
3100 max_content: MIN_AU,
3101 },
3102 percentage: None,
3103 };
3104
3105 let b = ColspanToDistribute {
3106 starting_column: 0,
3107 span: 1,
3108 content_sizes: ContentSizes {
3109 min_content: MIN_AU,
3110 max_content: MIN_AU,
3111 },
3112 percentage: None,
3113 };
3114
3115 let ordering = ColspanToDistribute::comparison_for_sort(&a, &b);
3116 assert_eq!(ordering, Ordering::Less);
3117 }
3118
3119 #[test]
3120 fn test_colspan_to_distribute_if_spans_are_equal_sort_by_starting_column() {
3121 let a = ColspanToDistribute {
3122 starting_column: 0,
3123 span: 0,
3124 content_sizes: ContentSizes {
3125 min_content: MIN_AU,
3126 max_content: MIN_AU,
3127 },
3128 percentage: None,
3129 };
3130
3131 let b = ColspanToDistribute {
3132 starting_column: 1,
3133 span: 0,
3134 content_sizes: ContentSizes {
3135 min_content: MIN_AU,
3136 max_content: MIN_AU,
3137 },
3138 percentage: None,
3139 };
3140
3141 let ordering = ColspanToDistribute::comparison_for_sort(&a, &b);
3142 assert_eq!(ordering, Ordering::Less);
3143 }
3144}