2020-11-22 13:38:18 +01:00
/*
* Copyright ( c ) 2020 , Andreas Kling < kling @ serenityos . org >
*
2021-04-22 01:24:48 -07:00
* SPDX - License - Identifier : BSD - 2 - Clause
2020-11-22 13:38:18 +01:00
*/
# include <LibWeb/DOM/Node.h>
2021-11-18 15:01:28 +01:00
# include <LibWeb/HTML/BrowsingContext.h>
2023-05-29 17:15:00 +03:00
# include <LibWeb/HTML/HTMLTableCellElement.h>
2023-06-07 02:10:55 +00:00
# include <LibWeb/Layout/BlockFormattingContext.h>
2020-11-22 15:53:01 +01:00
# include <LibWeb/Layout/Box.h>
2020-11-22 13:38:18 +01:00
# include <LibWeb/Layout/InlineFormattingContext.h>
# include <LibWeb/Layout/TableFormattingContext.h>
2022-12-04 22:39:38 +03:00
struct GridPosition {
size_t x ;
size_t y ;
} ;
inline bool operator = = ( GridPosition const & a , GridPosition const & b )
{
return a . x = = b . x & & a . y = = b . y ;
}
namespace AK {
template < >
struct Traits < GridPosition > : public GenericTraits < GridPosition > {
static unsigned hash ( GridPosition const & key )
{
return pair_int_hash ( key . x , key . y ) ;
}
} ;
}
2020-11-22 13:38:18 +01:00
namespace Web : : Layout {
2023-05-29 14:11:19 +03:00
TableFormattingContext : : TableFormattingContext ( LayoutState & state , Box const & root , FormattingContext * parent )
2022-12-04 21:57:21 +03:00
: FormattingContext ( Type : : Table , state , root , parent )
2020-11-22 13:38:18 +01:00
{
}
2022-03-14 13:21:51 -06:00
TableFormattingContext : : ~ TableFormattingContext ( ) = default ;
2020-11-22 13:38:18 +01:00
2023-05-29 15:54:22 +03:00
static inline bool is_table_row_group ( Box const & box )
{
auto const & display = box . display ( ) ;
return display . is_table_row_group ( ) | | display . is_table_header_group ( ) | | display . is_table_footer_group ( ) ;
}
2023-05-29 16:07:28 +03:00
static inline bool is_table_row ( Box const & box )
{
return box . display ( ) . is_table_row ( ) ;
}
2023-05-29 15:54:22 +03:00
template < typename Matcher , typename Callback >
static void for_each_child_box_matching ( Box const & parent , Matcher matcher , Callback callback )
{
parent . for_each_child_of_type < Box > ( [ & ] ( Box const & child_box ) {
if ( matcher ( child_box ) )
callback ( child_box ) ;
} ) ;
}
2023-06-07 02:10:55 +00:00
CSSPixels TableFormattingContext : : run_caption_layout ( LayoutMode layout_mode , CSS : : CaptionSide phase )
{
CSSPixels caption_height = 0 ;
for ( auto * child = table_box ( ) . first_child ( ) ; child ; child = child - > next_sibling ( ) ) {
if ( ! child - > display ( ) . is_table_caption ( ) | | child - > computed_values ( ) . caption_side ( ) ! = phase ) {
continue ;
}
// The caption boxes are principal block-level boxes that retain their own content, padding, margin, and border areas,
// and are rendered as normal block boxes inside the table wrapper box, as described in https://www.w3.org/TR/CSS22/tables.html#model
auto caption_context = make < BlockFormattingContext > ( m_state , * verify_cast < BlockContainer > ( child ) , this ) ;
caption_context - > run ( table_box ( ) , layout_mode , * m_available_space ) ;
VERIFY ( child - > is_box ( ) ) ;
auto const & child_box = static_cast < Box const & > ( * child ) ;
// FIXME: Since caption only has inline children, BlockFormattingContext doesn't resolve the vertical metrics.
// We need to do it manually here.
caption_context - > resolve_vertical_box_model_metrics ( child_box ) ;
auto const & caption_state = m_state . get ( child_box ) ;
if ( phase = = CSS : : CaptionSide : : Top ) {
m_state . get_mutable ( table_box ( ) ) . set_content_y ( caption_state . margin_box_height ( ) ) ;
} else {
m_state . get_mutable ( child_box ) . set_content_y (
m_state . get ( table_box ( ) ) . margin_box_height ( ) + caption_state . margin_box_top ( ) ) ;
}
caption_height + = caption_state . margin_box_height ( ) ;
}
return caption_height ;
}
2022-12-04 22:39:38 +03:00
void TableFormattingContext : : calculate_row_column_grid ( Box const & box )
2020-11-22 13:38:18 +01:00
{
2022-12-04 22:39:38 +03:00
// Implements https://html.spec.whatwg.org/multipage/tables.html#forming-a-table
HashMap < GridPosition , bool > grid ;
2022-02-20 15:51:24 +01:00
2022-12-04 22:39:38 +03:00
size_t x_width = 0 , y_height = 0 ;
size_t x_current = 0 , y_current = 0 ;
2020-11-22 13:38:18 +01:00
2022-12-04 22:39:38 +03:00
auto process_row = [ & ] ( auto & row ) {
if ( y_height = = y_current )
y_height + + ;
2020-11-29 22:37:15 +01:00
2022-12-04 22:39:38 +03:00
x_current = 0 ;
while ( x_current < x_width & & grid . contains ( GridPosition { x_current , y_current } ) )
x_current + + ;
2020-11-22 13:38:18 +01:00
2022-12-04 22:39:38 +03:00
for ( auto * child = row . first_child ( ) ; child ; child = child - > next_sibling ( ) ) {
2023-05-29 17:15:00 +03:00
if ( child - > display ( ) . is_table_cell ( ) ) {
2023-05-29 16:07:28 +03:00
Box const * box = static_cast < Box const * > ( child ) ;
2022-12-04 22:39:38 +03:00
if ( x_current = = x_width )
x_width + + ;
2020-11-22 13:38:18 +01:00
2023-05-29 17:15:00 +03:00
size_t colspan = 1 , rowspan = 1 ;
if ( box - > dom_node ( ) & & is < HTML : : HTMLTableCellElement > ( * box - > dom_node ( ) ) ) {
auto const & node = static_cast < HTML : : HTMLTableCellElement const & > ( * box - > dom_node ( ) ) ;
colspan = node . col_span ( ) ;
rowspan = node . row_span ( ) ;
}
2022-03-29 00:10:59 +02:00
2022-12-04 22:39:38 +03:00
if ( x_width < x_current + colspan )
x_width = x_current + colspan ;
if ( y_height < y_current + rowspan )
y_height = y_current + rowspan ;
2022-03-29 00:10:59 +02:00
2022-12-04 22:39:38 +03:00
for ( size_t y = y_current ; y < y_current + rowspan ; y + + )
for ( size_t x = x_current ; x < x_current + colspan ; x + + )
grid . set ( GridPosition { x , y } , true ) ;
m_cells . append ( Cell { * box , x_current , y_current , colspan , rowspan } ) ;
2022-03-29 00:10:59 +02:00
2022-12-04 22:39:38 +03:00
x_current + = colspan ;
2022-03-29 00:10:59 +02:00
}
}
2022-03-28 14:35:10 +02:00
2022-12-04 22:39:38 +03:00
m_rows . append ( Row { row } ) ;
y_current + + ;
} ;
2020-11-22 13:38:18 +01:00
2023-05-29 15:54:22 +03:00
for_each_child_box_matching ( box , is_table_row_group , [ & ] ( auto & row_group_box ) {
2023-05-29 16:07:28 +03:00
for_each_child_box_matching ( row_group_box , is_table_row , [ & ] ( auto & row_box ) {
process_row ( row_box ) ;
2022-12-04 22:39:38 +03:00
return IterationDecision : : Continue ;
2020-11-22 13:38:18 +01:00
} ) ;
} ) ;
2023-05-29 16:07:28 +03:00
for_each_child_box_matching ( box , is_table_row , [ & ] ( auto & row_box ) {
process_row ( row_box ) ;
2022-12-09 12:44:38 +03:00
return IterationDecision : : Continue ;
} ) ;
2022-12-04 22:39:38 +03:00
m_columns . resize ( x_width ) ;
}
2021-10-28 14:41:01 +02:00
2022-12-04 22:39:38 +03:00
void TableFormattingContext : : compute_table_measures ( )
{
2023-01-07 01:19:52 +03:00
size_t max_cell_column_span = 1 ;
2022-12-04 22:39:38 +03:00
for ( auto & cell : m_cells ) {
2023-01-24 17:13:59 +03:00
auto width_of_containing_block = m_state . get ( * table_wrapper ( ) . containing_block ( ) ) . content_width ( ) ;
2023-02-26 16:09:02 -07:00
auto & computed_values = cell . box - > computed_values ( ) ;
2023-05-06 16:34:55 +02:00
CSSPixels padding_left = computed_values . padding ( ) . left ( ) . to_px ( cell . box , width_of_containing_block ) ;
CSSPixels padding_right = computed_values . padding ( ) . right ( ) . to_px ( cell . box , width_of_containing_block ) ;
2023-01-02 23:06:55 +01:00
auto is_left_most_cell = cell . column_index = = 0 ;
auto is_right_most_cell = cell . column_index = = m_columns . size ( ) - 1 ;
2023-02-26 16:09:02 -07:00
auto should_hide_borders = cell . box - > computed_values ( ) . border_collapse ( ) = = CSS : : BorderCollapse : : Collapse ;
2022-12-15 12:57:36 +00:00
CSSPixels border_left = should_hide_borders & & ! is_left_most_cell ? 0 : computed_values . border_left ( ) . width ;
CSSPixels border_right = should_hide_borders & & ! is_right_most_cell ? 0 : computed_values . border_right ( ) . width ;
2023-01-02 23:06:55 +01:00
2023-05-06 16:34:55 +02:00
CSSPixels width = computed_values . width ( ) . to_px ( cell . box , width_of_containing_block ) ;
2022-12-09 13:44:41 +03:00
auto cell_intrinsic_offsets = padding_left + padding_right + border_left + border_right ;
auto min_content_width = calculate_min_content_width ( cell . box ) ;
auto max_content_width = calculate_max_content_width ( cell . box ) ;
2022-11-08 17:29:52 +00:00
CSSPixels min_width = min_content_width ;
2022-12-09 13:44:41 +03:00
if ( ! computed_values . min_width ( ) . is_auto ( ) )
2023-05-06 16:34:55 +02:00
min_width = max ( min_width , computed_values . min_width ( ) . to_px ( cell . box , width_of_containing_block ) ) ;
2022-12-09 13:44:41 +03:00
2023-01-24 17:09:27 +03:00
CSSPixels max_width = computed_values . width ( ) . is_auto ( ) ? max_content_width . value ( ) : width ;
2022-12-09 13:44:41 +03:00
if ( ! computed_values . max_width ( ) . is_none ( ) )
2023-05-06 16:34:55 +02:00
max_width = min ( max_width , computed_values . max_width ( ) . to_px ( cell . box , width_of_containing_block ) ) ;
2022-12-09 13:44:41 +03:00
2023-01-05 18:56:34 +03:00
auto computed_width = computed_values . width ( ) ;
if ( computed_width . is_percentage ( ) ) {
m_columns [ cell . column_index ] . type = ColumnType : : Percent ;
m_columns [ cell . column_index ] . percentage_width = max ( m_columns [ cell . column_index ] . percentage_width , computed_width . percentage ( ) . value ( ) ) ;
} else if ( computed_width . is_length ( ) ) {
m_columns [ cell . column_index ] . type = ColumnType : : Pixel ;
}
2023-01-07 01:19:52 +03:00
cell . min_width = min_width + cell_intrinsic_offsets ;
2023-01-24 17:09:27 +03:00
cell . max_width = max ( max ( width , min_width ) , max_width ) + cell_intrinsic_offsets ;
2023-01-07 01:19:52 +03:00
max_cell_column_span = max ( max_cell_column_span , cell . column_span ) ;
}
for ( auto & cell : m_cells ) {
if ( cell . column_span = = 1 ) {
m_columns [ cell . column_index ] . min_width = max ( m_columns [ cell . column_index ] . min_width , cell . min_width ) ;
m_columns [ cell . column_index ] . max_width = max ( m_columns [ cell . column_index ] . max_width , cell . max_width ) ;
}
}
for ( size_t current_column_span = 2 ; current_column_span < max_cell_column_span ; current_column_span + + ) {
// https://www.w3.org/TR/css-tables-3/#min-content-width-of-a-column-based-on-cells-of-span-up-to-n-n--1
Vector < Vector < CSSPixels > > cell_min_contributions_by_column_index ;
cell_min_contributions_by_column_index . resize ( m_columns . size ( ) ) ;
// https://www.w3.org/TR/css-tables-3/#max-content-width-of-a-column-based-on-cells-of-span-up-to-n-n--1
Vector < Vector < CSSPixels > > cell_max_contributions_by_column_index ;
cell_max_contributions_by_column_index . resize ( m_columns . size ( ) ) ;
for ( auto & cell : m_cells ) {
if ( cell . column_span = = current_column_span ) {
// Define the baseline max-content width as the sum of the max-content widths based on cells of span up to N-1 of all columns that the cell spans.
CSSPixels baseline_max_content_width = 0 ;
auto cell_start_column_index = cell . column_index ;
auto cell_end_column_index = cell . column_index + cell . column_span ;
for ( auto column_index = cell_start_column_index ; column_index < cell_end_column_index ; column_index + + ) {
baseline_max_content_width + = m_columns [ column_index ] . max_width ;
}
// The contribution of the cell is the sum of:
// the min-content width of the column based on cells of span up to N-1
auto cell_min_contribution = m_columns [ cell . column_index ] . min_width ;
// and the product of:
// - the ratio of the max-content width based on cells of span up to N-1 of the column to the baseline max-content width
// - the outer min-content width of the cell minus the baseline max-content width and baseline border spacing, or 0 if this is negative
cell_min_contribution + = ( m_columns [ cell . column_index ] . max_width / baseline_max_content_width ) * max ( CSSPixels ( 0 ) , cell . min_width - baseline_max_content_width ) ;
// The contribution of the cell is the sum of:
// the max-content width of the column based on cells of span up to N-1
auto cell_max_contribution = m_columns [ cell . column_index ] . max_width ;
// and the product of:
// - the ratio of the max-content width based on cells of span up to N-1 of the column to the baseline max-content width
// - the outer max-content width of the cell minus the baseline max-content width and the baseline border spacing, or 0 if this is negative
cell_max_contribution + = ( m_columns [ cell . column_index ] . max_width / baseline_max_content_width ) * max ( CSSPixels ( 0 ) , cell . max_width - baseline_max_content_width ) ;
cell_min_contributions_by_column_index [ cell . column_index ] . append ( cell_min_contribution ) ;
cell_max_contributions_by_column_index [ cell . column_index ] . append ( cell_max_contribution ) ;
}
}
for ( size_t column_index = 0 ; column_index < m_columns . size ( ) ; column_index + + ) {
// min-content width of a column based on cells of span up to N (N > 1) is
// the largest of the min-content width of the column based on cells of span up to N-1 and the contributions of the cells in the column whose colSpan is N
for ( auto min_contribution : cell_min_contributions_by_column_index [ column_index ] )
m_columns [ column_index ] . min_width = max ( m_columns [ column_index ] . min_width , min_contribution ) ;
// max-content width of a column based on cells of span up to N (N > 1) is
// the largest of the max-content width based on cells of span up to N-1 and the contributions of the cells in the column whose colSpan is N
for ( auto max_contribution : cell_max_contributions_by_column_index [ column_index ] )
m_columns [ column_index ] . max_width = max ( m_columns [ column_index ] . max_width , max_contribution ) ;
}
2022-12-04 22:39:38 +03:00
}
2020-11-22 13:38:18 +01:00
}
2023-06-07 02:10:55 +00:00
CSSPixels TableFormattingContext : : compute_capmin ( )
{
// The caption width minimum (CAPMIN) is the largest of the table captions min-content contribution:
// https://drafts.csswg.org/css-tables-3/#computing-the-table-width
CSSPixels capmin = 0 ;
for ( auto * child = table_box ( ) . first_child ( ) ; child ; child = child - > next_sibling ( ) ) {
if ( ! child - > display ( ) . is_table_caption ( ) ) {
continue ;
}
VERIFY ( child - > is_box ( ) ) ;
capmin = max ( calculate_min_content_width ( static_cast < Box const & > ( * child ) ) , capmin ) ;
}
return capmin ;
}
2023-01-05 18:56:34 +03:00
void TableFormattingContext : : compute_table_width ( )
2020-11-22 13:38:18 +01:00
{
2023-04-29 16:27:21 +03:00
// https://drafts.csswg.org/css-tables-3/#computing-the-table-width
2023-01-08 01:54:25 +03:00
auto & table_box_state = m_state . get_mutable ( table_box ( ) ) ;
2022-07-09 15:17:47 +02:00
2023-01-08 01:54:25 +03:00
auto & computed_values = table_box ( ) . computed_values ( ) ;
2022-03-29 00:10:59 +02:00
2023-05-03 14:40:39 +03:00
CSSPixels width_of_table_containing_block = m_state . get ( * table_box ( ) . containing_block ( ) ) . content_width ( ) ;
2023-01-14 15:05:08 +01:00
// Percentages on 'width' and 'height' on the table are relative to the table wrapper box's containing block,
// not the table wrapper box itself.
2023-05-03 14:40:39 +03:00
CSSPixels width_of_table_wrapper_containing_block = m_state . get ( * table_wrapper ( ) . containing_block ( ) ) . content_width ( ) ;
2022-12-04 22:39:38 +03:00
// The row/column-grid width minimum (GRIDMIN) width is the sum of the min-content width
// of all the columns plus cell spacing or borders.
2022-12-15 12:57:36 +00:00
CSSPixels grid_min = 0.0f ;
2022-12-04 22:39:38 +03:00
for ( auto & column : m_columns ) {
grid_min + = column . min_width ;
}
// The row/column-grid width maximum (GRIDMAX) width is the sum of the max-content width
// of all the columns plus cell spacing or borders.
2022-12-15 12:57:36 +00:00
CSSPixels grid_max = 0.0f ;
2022-12-04 22:39:38 +03:00
for ( auto & column : m_columns ) {
grid_max + = column . max_width ;
}
// The used min-width of a table is the greater of the resolved min-width, CAPMIN, and GRIDMIN.
2023-06-07 02:10:55 +00:00
auto used_min_width = max ( grid_min , compute_capmin ( ) ) ;
2022-12-04 22:39:38 +03:00
if ( ! computed_values . min_width ( ) . is_auto ( ) ) {
2023-05-06 16:34:55 +02:00
used_min_width = max ( used_min_width , computed_values . min_width ( ) . to_px ( table_box ( ) , width_of_table_wrapper_containing_block ) ) ;
2022-12-04 22:39:38 +03:00
}
2022-12-15 12:57:36 +00:00
CSSPixels used_width ;
2022-12-04 22:39:38 +03:00
if ( computed_values . width ( ) . is_auto ( ) ) {
// If the table-root has 'width: auto', the used width is the greater of
// min(GRIDMAX, the table’ s containing block width), the used min-width of the table.
used_width = max ( min ( grid_max , width_of_table_containing_block ) , used_min_width ) ;
} else {
// If the table-root’ s width property has a computed value (resolving to
// resolved-table-width) other than auto, the used width is the greater
// of resolved-table-width, and the used min-width of the table.
2023-05-06 16:34:55 +02:00
CSSPixels resolved_table_width = computed_values . width ( ) . to_px ( table_box ( ) , width_of_table_wrapper_containing_block ) ;
2022-12-04 22:39:38 +03:00
used_width = max ( resolved_table_width , used_min_width ) ;
2023-04-05 18:21:40 +02:00
if ( ! computed_values . max_width ( ) . is_none ( ) )
2023-05-06 16:34:55 +02:00
used_width = min ( used_width , computed_values . max_width ( ) . to_px ( table_box ( ) , width_of_table_wrapper_containing_block ) ) ;
2022-12-04 22:39:38 +03:00
}
2023-04-29 16:27:21 +03:00
// The assignable table width is the used width of the table minus the total horizontal border spacing (if any).
// This is the width that we will be able to allocate to the columns.
table_box_state . set_content_width ( used_width - table_box_state . border_left - table_box_state . border_right ) ;
2020-11-22 13:38:18 +01:00
}
2023-01-05 18:56:34 +03:00
void TableFormattingContext : : distribute_width_to_columns ( )
2020-11-22 13:38:18 +01:00
{
2023-01-05 18:56:34 +03:00
// Implements https://www.w3.org/TR/css-tables-3/#width-distribution-algorithm
2023-01-08 01:54:25 +03:00
CSSPixels available_width = m_state . get ( table_box ( ) ) . content_width ( ) ;
2023-01-05 18:56:34 +03:00
auto columns_total_used_width = [ & ] ( ) {
CSSPixels total_used_width = 0 ;
for ( auto & column : m_columns ) {
total_used_width + = column . used_width ;
}
return total_used_width ;
} ;
auto column_preferred_width = [ & ] ( Column & column ) {
switch ( column . type ) {
case ColumnType : : Percent : {
return max ( column . min_width , column . percentage_width / 100 * available_width ) ;
}
case ColumnType : : Pixel :
case ColumnType : : Auto : {
return column . max_width ;
}
default : {
VERIFY_NOT_REACHED ( ) ;
}
}
} ;
auto expand_columns_to_fill_available_width = [ & ] ( ColumnType column_type ) {
CSSPixels remaining_available_width = available_width ;
CSSPixels total_preferred_width_increment = 0 ;
for ( auto & column : m_columns ) {
remaining_available_width - = column . used_width ;
if ( column . type = = column_type ) {
total_preferred_width_increment + = column_preferred_width ( column ) - column . min_width ;
}
}
2023-01-24 17:14:47 +03:00
if ( total_preferred_width_increment = = 0 )
return ;
2023-01-05 18:56:34 +03:00
for ( auto & column : m_columns ) {
if ( column . type = = column_type ) {
CSSPixels preferred_width_increment = column_preferred_width ( column ) - column . min_width ;
column . used_width + = preferred_width_increment * remaining_available_width / total_preferred_width_increment ;
}
}
} ;
2020-11-22 13:38:18 +01:00
2023-01-05 18:56:34 +03:00
auto shrink_columns_to_fit_available_width = [ & ] ( ColumnType column_type ) {
for ( auto & column : m_columns ) {
if ( column . type = = column_type )
column . used_width = column . min_width ;
}
expand_columns_to_fill_available_width ( column_type ) ;
} ;
for ( auto & column : m_columns ) {
column . used_width = column . min_width ;
}
for ( auto & column : m_columns ) {
if ( column . type = = ColumnType : : Percent ) {
column . used_width = max ( column . min_width , column . percentage_width / 100 * available_width ) ;
}
}
if ( columns_total_used_width ( ) > available_width ) {
shrink_columns_to_fit_available_width ( ColumnType : : Percent ) ;
2023-01-02 22:53:19 +01:00
return ;
2023-01-05 18:56:34 +03:00
}
2023-01-02 22:53:19 +01:00
2023-01-05 18:56:34 +03:00
for ( auto & column : m_columns ) {
if ( column . type = = ColumnType : : Pixel ) {
column . used_width = column . max_width ;
}
}
if ( columns_total_used_width ( ) > available_width ) {
shrink_columns_to_fit_available_width ( ColumnType : : Pixel ) ;
return ;
}
if ( columns_total_used_width ( ) < available_width ) {
expand_columns_to_fill_available_width ( ColumnType : : Auto ) ;
}
if ( columns_total_used_width ( ) < available_width ) {
expand_columns_to_fill_available_width ( ColumnType : : Pixel ) ;
}
if ( columns_total_used_width ( ) < available_width ) {
expand_columns_to_fill_available_width ( ColumnType : : Percent ) ;
}
2023-01-24 17:20:24 +03:00
if ( columns_total_used_width ( ) < available_width ) {
// NOTE: if all columns got their max width and there is still width to distribute left
// it should be assigned to columns proportionally to their max width
CSSPixels grid_max = 0.0f ;
for ( auto & column : m_columns ) {
grid_max + = column . max_width ;
}
auto width_to_distribute = available_width - columns_total_used_width ( ) ;
2023-04-28 19:26:04 +03:00
if ( grid_max = = 0 ) {
// If total max width of columns is zero then divide distributable width equally among them
auto column_width = width_to_distribute / m_columns . size ( ) ;
for ( auto & column : m_columns ) {
column . used_width = column_width ;
}
} else {
// Distribute width to columns proportionally to their max width
for ( auto & column : m_columns ) {
column . used_width + = width_to_distribute * column . max_width / grid_max ;
}
2023-01-24 17:20:24 +03:00
}
}
2022-12-04 22:39:38 +03:00
}
2022-12-04 22:47:45 +03:00
void TableFormattingContext : : determine_intrisic_size_of_table_container ( AvailableSpace const & available_space )
{
2023-01-08 01:54:25 +03:00
auto & table_box_state = m_state . get_mutable ( table_box ( ) ) ;
2022-12-04 22:47:45 +03:00
if ( available_space . width . is_min_content ( ) ) {
// The min-content width of a table is the width required to fit all of its columns min-content widths and its undistributable spaces.
2022-12-15 12:57:36 +00:00
CSSPixels grid_min = 0.0f ;
2022-12-04 22:47:45 +03:00
for ( auto & column : m_columns )
grid_min + = column . min_width ;
2022-11-04 20:32:50 +00:00
table_box_state . set_content_width ( grid_min ) ;
2022-12-04 22:47:45 +03:00
}
if ( available_space . width . is_max_content ( ) ) {
// The max-content width of a table is the width required to fit all of its columns max-content widths and its undistributable spaces.
2022-12-15 12:57:36 +00:00
CSSPixels grid_max = 0.0f ;
2022-12-04 22:47:45 +03:00
for ( auto & column : m_columns )
grid_max + = column . max_width ;
2022-11-04 20:32:50 +00:00
table_box_state . set_content_width ( grid_max ) ;
2022-12-04 22:47:45 +03:00
}
}
2023-04-29 00:43:05 +03:00
void TableFormattingContext : : compute_table_height ( LayoutMode layout_mode )
2022-12-04 22:39:38 +03:00
{
2023-04-29 00:43:05 +03:00
// First pass of row height calculation:
for ( auto & row : m_rows ) {
auto row_computed_height = row . box - > computed_values ( ) . height ( ) ;
if ( row_computed_height . is_length ( ) ) {
auto height_of_containing_block = m_state . get ( * row . box - > containing_block ( ) ) . content_height ( ) ;
2023-05-06 16:34:55 +02:00
auto row_used_height = row_computed_height . to_px ( row . box , height_of_containing_block ) ;
2023-04-29 00:43:05 +03:00
row . base_height = max ( row . base_height , row_used_height ) ;
}
}
// First pass of cells layout:
2022-12-04 22:39:38 +03:00
for ( auto & cell : m_cells ) {
2023-04-28 02:43:48 +03:00
auto & row = m_rows [ cell . row_index ] ;
2022-12-04 22:39:38 +03:00
auto & cell_state = m_state . get_mutable ( cell . box ) ;
2022-03-21 20:39:19 +01:00
2022-12-15 12:57:36 +00:00
CSSPixels span_width = 0 ;
2022-12-04 22:39:38 +03:00
for ( size_t i = 0 ; i < cell . column_span ; + + i )
span_width + = m_columns [ cell . column_index + i ] . used_width ;
2022-03-28 14:48:08 +02:00
2023-02-26 16:09:02 -07:00
auto width_of_containing_block = m_state . get ( * cell . box - > containing_block ( ) ) . content_width ( ) ;
2022-12-04 22:39:38 +03:00
auto width_of_containing_block_as_length = CSS : : Length : : make_px ( width_of_containing_block ) ;
2023-04-28 02:43:48 +03:00
auto height_of_containing_block = m_state . get ( * cell . box - > containing_block ( ) ) . content_height ( ) ;
auto height_of_containing_block_as_length = CSS : : Length : : make_px ( height_of_containing_block ) ;
2020-11-22 13:38:18 +01:00
2023-05-06 16:34:55 +02:00
cell_state . padding_top = cell . box - > computed_values ( ) . padding ( ) . top ( ) . to_px ( cell . box , width_of_containing_block ) ;
cell_state . padding_bottom = cell . box - > computed_values ( ) . padding ( ) . bottom ( ) . to_px ( cell . box , width_of_containing_block ) ;
cell_state . padding_left = cell . box - > computed_values ( ) . padding ( ) . left ( ) . to_px ( cell . box , width_of_containing_block ) ;
cell_state . padding_right = cell . box - > computed_values ( ) . padding ( ) . right ( ) . to_px ( cell . box , width_of_containing_block ) ;
2023-01-02 23:06:55 +01:00
auto is_top_most_cell = cell . row_index = = 0 ;
auto is_left_most_cell = cell . column_index = = 0 ;
auto is_right_most_cell = cell . column_index = = m_columns . size ( ) - 1 ;
auto is_bottom_most_cell = cell . row_index = = m_rows . size ( ) - 1 ;
2023-02-26 16:09:02 -07:00
auto should_hide_borders = cell . box - > computed_values ( ) . border_collapse ( ) = = CSS : : BorderCollapse : : Collapse ;
2023-01-02 23:06:55 +01:00
2023-02-26 16:09:02 -07:00
cell_state . border_top = ( should_hide_borders & & is_top_most_cell ) ? 0 : cell . box - > computed_values ( ) . border_top ( ) . width ;
cell_state . border_bottom = ( should_hide_borders & & is_bottom_most_cell ) ? 0 : cell . box - > computed_values ( ) . border_bottom ( ) . width ;
cell_state . border_left = ( should_hide_borders & & is_left_most_cell ) ? 0 : cell . box - > computed_values ( ) . border_left ( ) . width ;
cell_state . border_right = ( should_hide_borders & & is_right_most_cell ) ? 0 : cell . box - > computed_values ( ) . border_right ( ) . width ;
2022-12-04 22:39:38 +03:00
2023-04-28 02:43:48 +03:00
auto cell_computed_height = cell . box - > computed_values ( ) . height ( ) ;
if ( cell_computed_height . is_length ( ) ) {
2023-05-06 16:34:55 +02:00
auto cell_used_height = cell_computed_height . to_px ( cell . box , height_of_containing_block ) ;
2023-04-28 02:43:48 +03:00
cell_state . set_content_height ( cell_used_height - cell_state . border_box_top ( ) - cell_state . border_box_bottom ( ) ) ;
2023-04-29 00:43:05 +03:00
row . base_height = max ( row . base_height , cell_used_height ) ;
2023-04-28 02:43:48 +03:00
}
2022-11-04 20:32:50 +00:00
cell_state . set_content_width ( ( span_width - cell_state . border_box_left ( ) - cell_state . border_box_right ( ) ) ) ;
2023-01-14 15:11:58 +01:00
if ( auto independent_formatting_context = layout_inside ( cell . box , layout_mode , cell_state . available_inner_space_or_constraints_from ( * m_available_space ) ) ) {
cell_state . set_content_height ( independent_formatting_context - > automatic_content_height ( ) ) ;
independent_formatting_context - > parent_context_did_dimension_child_root_box ( ) ;
}
2022-12-04 22:39:38 +03:00
2023-05-31 10:18:34 +02:00
cell . baseline = box_baseline ( cell . box ) ;
2022-12-04 20:02:28 +03:00
2023-05-18 03:55:39 +00:00
// Only cells spanning the current row exclusively are part of computing minimum height of a row,
// as described in https://www.w3.org/TR/css-tables-3/#computing-the-table-height
if ( cell . row_span = = 1 ) {
row . base_height = max ( row . base_height , cell_state . border_box_height ( ) ) ;
}
2022-12-04 20:02:28 +03:00
row . baseline = max ( row . baseline , cell . baseline ) ;
2022-12-04 22:39:38 +03:00
}
2023-04-28 00:35:47 +03:00
2023-04-29 00:43:05 +03:00
CSSPixels sum_rows_height = 0 ;
for ( auto & row : m_rows ) {
sum_rows_height + = row . base_height ;
}
// The height of a table is the sum of the row heights plus any cell spacing or borders.
m_table_height = sum_rows_height ;
if ( ! table_box ( ) . computed_values ( ) . height ( ) . is_auto ( ) ) {
// If the table has a height property with a value other than auto, it is treated as a minimum height for the
// table grid, and will eventually be distributed to the height of the rows if their collective minimum height
// ends up smaller than this number.
CSSPixels height_of_table_containing_block = m_state . get ( * table_wrapper ( ) . containing_block ( ) ) . content_height ( ) ;
2023-05-06 16:34:55 +02:00
auto specified_table_height = table_box ( ) . computed_values ( ) . height ( ) . to_px ( table_box ( ) , height_of_table_containing_block ) ;
2023-04-29 00:43:05 +03:00
if ( m_table_height < specified_table_height ) {
m_table_height = specified_table_height ;
}
}
for ( auto & row : m_rows ) {
// Reference size is the largest of
// - its initial base height and
// - its new base height (the one evaluated during the second layout pass, where percentages used in
// rowgroups/rows/cells' specified heights were resolved according to the table height, instead of
// being ignored as 0px).
// Assign reference size to base size. Later reference size might change to largee value during
// second pass of rows layout.
row . reference_height = row . base_height ;
}
// Second pass of rows height calculation:
// At this point percentage row height can be resolved because final table height is calculated.
2023-04-28 00:35:47 +03:00
for ( auto & row : m_rows ) {
auto row_computed_height = row . box - > computed_values ( ) . height ( ) ;
2023-04-29 00:43:05 +03:00
if ( row_computed_height . is_percentage ( ) ) {
2023-05-06 16:34:55 +02:00
auto row_used_height = row_computed_height . to_px ( row . box , m_table_height ) ;
2023-04-29 00:43:05 +03:00
row . reference_height = max ( row . reference_height , row_used_height ) ;
} else {
continue ;
}
}
// Second pass cells layout:
// At this point percantage cell height can be resolved because final table heigh is calculated.
for ( auto & cell : m_cells ) {
auto & row = m_rows [ cell . row_index ] ;
auto & cell_state = m_state . get_mutable ( cell . box ) ;
CSSPixels span_width = 0 ;
for ( size_t i = 0 ; i < cell . column_span ; + + i )
span_width + = m_columns [ cell . column_index + i ] . used_width ;
auto cell_computed_height = cell . box - > computed_values ( ) . height ( ) ;
if ( cell_computed_height . is_percentage ( ) ) {
2023-05-06 16:34:55 +02:00
auto cell_used_height = cell_computed_height . to_px ( cell . box , m_table_height ) ;
2023-04-29 00:43:05 +03:00
cell_state . set_content_height ( cell_used_height - cell_state . border_box_top ( ) - cell_state . border_box_bottom ( ) ) ;
row . reference_height = max ( row . reference_height , cell_used_height ) ;
} else {
continue ;
}
cell_state . set_content_width ( ( span_width - cell_state . border_box_left ( ) - cell_state . border_box_right ( ) ) ) ;
if ( auto independent_formatting_context = layout_inside ( cell . box , layout_mode , cell_state . available_inner_space_or_constraints_from ( * m_available_space ) ) ) {
independent_formatting_context - > parent_context_did_dimension_child_root_box ( ) ;
}
2023-05-31 10:18:34 +02:00
cell . baseline = box_baseline ( cell . box ) ;
2023-04-29 00:43:05 +03:00
row . reference_height = max ( row . reference_height , cell_state . border_box_height ( ) ) ;
row . baseline = max ( row . baseline , cell . baseline ) ;
}
}
void TableFormattingContext : : distribute_height_to_rows ( )
{
CSSPixels sum_reference_height = 0 ;
for ( auto & row : m_rows ) {
sum_reference_height + = row . reference_height ;
}
if ( sum_reference_height = = 0 )
return ;
Vector < Row & > rows_with_auto_height ;
for ( auto & row : m_rows ) {
if ( row . box - > computed_values ( ) . height ( ) . is_auto ( ) ) {
rows_with_auto_height . append ( row ) ;
}
}
if ( m_table_height < = sum_reference_height ) {
// If the table height is equal or smaller than sum of reference sizes, the final height assigned to each row
// will be the weighted mean of the base and the reference size that yields the correct total height.
for ( auto & row : m_rows ) {
auto weight = row . reference_height / sum_reference_height ;
auto final_height = m_table_height * weight ;
row . final_height = final_height ;
}
} else if ( rows_with_auto_height . size ( ) > 0 ) {
// Else, if the table owns any “auto-height” row (a row whose size is only determined by its content size and
// none of the specified heights), each non-auto-height row receives its reference height and auto-height rows
// receive their reference size plus some increment which is equal to the height missing to amount to the
// specified table height divided by the amount of such rows.
for ( auto & row : m_rows ) {
row . final_height = row . reference_height ;
}
auto auto_height_rows_increment = ( m_table_height - sum_reference_height ) / rows_with_auto_height . size ( ) ;
for ( auto & row : rows_with_auto_height ) {
row . final_height + = auto_height_rows_increment ;
}
} else {
// Else, all rows receive their reference size plus some increment which is equal to the height missing to
// amount to the specified table height divided by the amount of rows.
auto increment = ( m_table_height - sum_reference_height ) / m_rows . size ( ) ;
for ( auto & row : m_rows ) {
row . final_height = row . reference_height + increment ;
2023-04-28 00:35:47 +03:00
}
}
2023-01-08 00:31:00 +03:00
}
2023-01-20 00:59:43 +01:00
void TableFormattingContext : : position_row_boxes ( CSSPixels & total_content_height )
2023-01-08 00:31:00 +03:00
{
2023-01-19 21:23:46 +01:00
auto const & table_state = m_state . get ( table_box ( ) ) ;
2023-06-07 02:10:55 +00:00
CSSPixels row_top_offset = table_state . offset . y ( ) + table_state . padding_top ;
2023-01-19 21:23:46 +01:00
CSSPixels row_left_offset = table_state . border_left + table_state . padding_left ;
2022-12-04 22:39:38 +03:00
for ( size_t y = 0 ; y < m_rows . size ( ) ; y + + ) {
auto & row = m_rows [ y ] ;
auto & row_state = m_state . get_mutable ( row . box ) ;
2022-12-15 12:57:36 +00:00
CSSPixels row_width = 0.0f ;
2022-12-04 22:39:38 +03:00
for ( auto & column : m_columns ) {
row_width + = column . used_width ;
}
2023-04-29 00:43:05 +03:00
row_state . set_content_height ( row . final_height ) ;
2022-11-04 20:32:50 +00:00
row_state . set_content_width ( row_width ) ;
2023-01-19 21:23:46 +01:00
row_state . set_content_x ( row_left_offset ) ;
2022-11-04 20:32:50 +00:00
row_state . set_content_y ( row_top_offset ) ;
2022-12-31 14:32:15 +01:00
row_top_offset + = row_state . content_height ( ) ;
2022-12-04 22:39:38 +03:00
}
2023-01-19 21:23:46 +01:00
CSSPixels row_group_top_offset = table_state . border_top + table_state . padding_top ;
CSSPixels row_group_left_offset = table_state . border_left + table_state . padding_left ;
2023-05-29 15:54:22 +03:00
for_each_child_box_matching ( table_box ( ) , is_table_row_group , [ & ] ( auto & row_group_box ) {
2022-12-15 12:57:36 +00:00
CSSPixels row_group_height = 0.0f ;
CSSPixels row_group_width = 0.0f ;
2022-12-04 22:39:38 +03:00
auto & row_group_box_state = m_state . get_mutable ( row_group_box ) ;
2023-01-19 21:23:46 +01:00
row_group_box_state . set_content_x ( row_group_left_offset ) ;
2022-11-04 20:32:50 +00:00
row_group_box_state . set_content_y ( row_group_top_offset ) ;
2022-12-04 22:39:38 +03:00
2023-05-29 16:07:28 +03:00
for_each_child_box_matching ( row_group_box , is_table_row , [ & ] ( auto & row ) {
2023-01-09 07:17:28 +03:00
auto const & row_state = m_state . get ( row ) ;
2022-12-04 22:39:38 +03:00
row_group_height + = row_state . border_box_height ( ) ;
row_group_width = max ( row_group_width , row_state . border_box_width ( ) ) ;
} ) ;
2020-11-22 13:38:18 +01:00
2022-11-04 20:32:50 +00:00
row_group_box_state . set_content_height ( row_group_height ) ;
row_group_box_state . set_content_width ( row_group_width ) ;
2023-01-09 07:17:28 +03:00
row_group_top_offset + = row_group_height ;
2022-03-28 14:32:21 +02:00
} ) ;
2023-01-20 00:59:43 +01:00
2023-06-07 02:10:55 +00:00
total_content_height = max ( row_top_offset , row_group_top_offset ) - table_state . offset . y ( ) - table_state . padding_top ;
2023-01-08 00:48:17 +03:00
}
2023-01-08 00:57:57 +03:00
void TableFormattingContext : : position_cell_boxes ( )
2023-01-08 00:48:17 +03:00
{
CSSPixels left_column_offset = 0 ;
for ( auto & column : m_columns ) {
column . left_offset = left_column_offset ;
left_column_offset + = column . used_width ;
}
2022-12-04 22:39:38 +03:00
for ( auto & cell : m_cells ) {
auto & cell_state = m_state . get_mutable ( cell . box ) ;
auto & row_state = m_state . get ( m_rows [ cell . row_index ] . box ) ;
2022-12-15 12:57:36 +00:00
CSSPixels const cell_border_box_height = cell_state . content_height ( ) + cell_state . border_box_top ( ) + cell_state . border_box_bottom ( ) ;
2023-05-18 03:55:39 +00:00
CSSPixels const row_content_height = compute_row_content_height ( cell ) ;
2023-02-26 16:09:02 -07:00
auto const & vertical_align = cell . box - > computed_values ( ) . vertical_align ( ) ;
2022-12-04 20:02:28 +03:00
if ( vertical_align . has < CSS : : VerticalAlign > ( ) ) {
switch ( vertical_align . get < CSS : : VerticalAlign > ( ) ) {
case CSS : : VerticalAlign : : Middle : {
2022-11-04 20:32:50 +00:00
cell_state . padding_top + = ( row_content_height - cell_border_box_height ) / 2 ;
cell_state . padding_bottom + = ( row_content_height - cell_border_box_height ) / 2 ;
2022-12-04 20:02:28 +03:00
break ;
}
// FIXME: implement actual 'top' and 'bottom' support instead of fall back to 'baseline'
case CSS : : VerticalAlign : : Top :
case CSS : : VerticalAlign : : Bottom :
case CSS : : VerticalAlign : : Baseline : {
2022-11-04 20:32:50 +00:00
cell_state . padding_top + = m_rows [ cell . row_index ] . baseline - cell . baseline ;
cell_state . padding_bottom + = row_content_height - cell_border_box_height ;
2022-12-04 20:02:28 +03:00
break ;
}
default :
VERIFY_NOT_REACHED ( ) ;
}
}
2022-11-04 20:32:50 +00:00
cell_state . offset = row_state . offset . translated ( cell_state . border_box_left ( ) + m_columns [ cell . column_index ] . left_offset , cell_state . border_box_top ( ) ) ;
2020-11-22 13:38:18 +01:00
}
2023-01-08 00:57:57 +03:00
}
2023-05-18 03:55:39 +00:00
CSSPixels TableFormattingContext : : compute_row_content_height ( Cell const & cell ) const
{
auto & row_state = m_state . get ( m_rows [ cell . row_index ] . box ) ;
if ( cell . row_span = = 1 ) {
return row_state . content_height ( ) ;
}
// The height of a cell is the sum of all spanned rows, as described in
// https://www.w3.org/TR/css-tables-3/#bounding-box-assignment
// When the row span is greater than 1, the borders of inner rows within the span have to be
// included in the content height of the spanning cell. First top and final bottom borders are
// excluded to be consistent with the handling of row span 1 case above, which uses the content
// height (no top and bottom borders) of the row.
CSSPixels span_height = 0 ;
for ( size_t i = 0 ; i < cell . row_span ; + + i ) {
auto const & row_state = m_state . get ( m_rows [ cell . row_index + i ] . box ) ;
if ( i = = 0 ) {
span_height + = row_state . content_height ( ) + row_state . border_box_bottom ( ) ;
} else if ( i = = cell . row_span - 1 ) {
span_height + = row_state . border_box_top ( ) + row_state . content_height ( ) ;
} else {
span_height + = row_state . border_box_height ( ) ;
}
}
return span_height ;
}
2023-01-14 15:11:58 +01:00
void TableFormattingContext : : run ( Box const & box , LayoutMode layout_mode , AvailableSpace const & available_space )
2023-01-08 00:57:57 +03:00
{
m_available_space = available_space ;
2023-06-07 02:10:55 +00:00
auto total_captions_height = run_caption_layout ( layout_mode , CSS : : CaptionSide : : Top ) ;
2023-01-08 00:57:57 +03:00
CSSPixels total_content_height = 0 ;
// Determine the number of rows/columns the table requires.
calculate_row_column_grid ( box ) ;
// Compute the minimum width of each column.
compute_table_measures ( ) ;
2023-03-27 19:06:21 +03:00
if ( available_space . width . is_intrinsic_sizing_constraint ( ) & & ! available_space . height . is_intrinsic_sizing_constraint ( ) ) {
2023-01-08 00:57:57 +03:00
determine_intrisic_size_of_table_container ( available_space ) ;
return ;
}
// Compute the width of the table.
compute_table_width ( ) ;
// Distribute the width of the table among columns.
distribute_width_to_columns ( ) ;
2023-04-29 00:43:05 +03:00
compute_table_height ( layout_mode ) ;
distribute_height_to_rows ( ) ;
2023-01-20 00:59:43 +01:00
position_row_boxes ( total_content_height ) ;
2023-01-08 00:57:57 +03:00
position_cell_boxes ( ) ;
2022-12-04 22:39:38 +03:00
2023-01-08 01:54:25 +03:00
m_state . get_mutable ( table_box ( ) ) . set_content_height ( total_content_height ) ;
2022-12-04 22:39:38 +03:00
2023-06-07 02:10:55 +00:00
total_captions_height + = run_caption_layout ( layout_mode , CSS : : CaptionSide : : Bottom ) ;
// Table captions are positioned between the table margins and its borders (outside the grid box borders) as described in
// https://www.w3.org/TR/css-tables-3/#bounding-box-assignment
// A visual representation of this model can be found at https://www.w3.org/TR/css-tables-3/images/table_container.png
m_state . get_mutable ( table_box ( ) ) . margin_bottom + = total_captions_height ;
2022-12-04 22:39:38 +03:00
// FIXME: This is a total hack, we should respect the 'height' property.
m_automatic_content_height = total_content_height ;
2020-11-22 13:38:18 +01:00
}
2023-03-19 09:57:31 +01:00
CSSPixels TableFormattingContext : : automatic_content_width ( ) const
{
return greatest_child_width ( context_box ( ) ) ;
}
2022-11-23 17:46:10 +00:00
CSSPixels TableFormattingContext : : automatic_content_height ( ) const
2022-09-24 13:39:43 +02:00
{
return m_automatic_content_height ;
}
2020-11-22 13:38:18 +01:00
}