2020-01-18 09:38:21 +01:00
/*
2023-02-20 18:56:08 +01:00
* Copyright ( c ) 2018 - 2023 , Andreas Kling < kling @ serenityos . org >
2023-06-08 15:25:16 +01:00
* Copyright ( c ) 2021 - 2023 , Sam Atkins < atkinssj @ serenityos . org >
2020-01-18 09:38:21 +01:00
*
2021-04-22 01:24:48 -07:00
* SPDX - License - Identifier : BSD - 2 - Clause
2020-01-18 09:38:21 +01:00
*/
2021-01-01 16:42:44 +01:00
# include <AK/Demangle.h>
2023-03-24 16:42:50 +00:00
# include <LibWeb/CSS/StyleValues/AbstractImageStyleValue.h>
2023-03-23 17:44:13 +00:00
# include <LibWeb/CSS/StyleValues/BackgroundRepeatStyleValue.h>
2023-03-23 17:54:05 +00:00
# include <LibWeb/CSS/StyleValues/BackgroundSizeStyleValue.h>
2023-03-25 00:33:20 +00:00
# include <LibWeb/CSS/StyleValues/BorderRadiusStyleValue.h>
2023-04-03 00:04:00 +01:00
# include <LibWeb/CSS/StyleValues/EdgeStyleValue.h>
2023-06-08 15:25:16 +01:00
# include <LibWeb/CSS/StyleValues/IdentifierStyleValue.h>
2023-09-06 18:30:57 +02:00
# include <LibWeb/CSS/StyleValues/IntegerStyleValue.h>
2023-05-27 12:28:25 +01:00
# include <LibWeb/CSS/StyleValues/LengthStyleValue.h>
2023-09-07 15:29:54 +01:00
# include <LibWeb/CSS/StyleValues/MathDepthStyleValue.h>
2023-06-01 16:16:15 +01:00
# include <LibWeb/CSS/StyleValues/NumberStyleValue.h>
2023-05-21 09:50:56 +02:00
# include <LibWeb/CSS/StyleValues/PercentageStyleValue.h>
2023-06-08 15:25:16 +01:00
# include <LibWeb/CSS/StyleValues/RatioStyleValue.h>
2023-03-25 00:12:21 +00:00
# include <LibWeb/CSS/StyleValues/StyleValueList.h>
2023-05-27 22:05:48 +02:00
# include <LibWeb/CSS/StyleValues/TimeStyleValue.h>
2023-04-23 01:31:17 +01:00
# include <LibWeb/CSS/StyleValues/URLStyleValue.h>
2020-03-07 10:32:51 +01:00
# include <LibWeb/DOM/Document.h>
2020-11-26 21:18:34 +01:00
# include <LibWeb/Dump.h>
2021-11-18 15:01:28 +01:00
# include <LibWeb/HTML/BrowsingContext.h>
2020-12-05 20:10:02 +01:00
# include <LibWeb/HTML/HTMLHtmlElement.h>
2021-10-06 20:02:41 +02:00
# include <LibWeb/Layout/BlockContainer.h>
2020-12-05 20:10:39 +01:00
# include <LibWeb/Layout/FormattingContext.h>
2020-11-22 15:53:01 +01:00
# include <LibWeb/Layout/Node.h>
2023-06-22 12:22:29 +00:00
# include <LibWeb/Layout/TableWrapper.h>
2021-01-01 18:55:47 +01:00
# include <LibWeb/Layout/TextNode.h>
2023-02-25 11:04:29 +01:00
# include <LibWeb/Layout/Viewport.h>
2022-09-17 21:25:50 +02:00
# include <LibWeb/Platform/FontPlugin.h>
2019-06-15 22:49:44 +02:00
2020-11-22 15:53:01 +01:00
namespace Web : : Layout {
2020-03-07 10:27:02 +01:00
2020-11-22 15:53:01 +01:00
Node : : Node ( DOM : : Document & document , DOM : : Node * node )
2022-10-17 14:41:50 +02:00
: m_dom_node ( node ? * node : document )
2022-10-20 18:20:34 +02:00
, m_browsing_context ( * document . browsing_context ( ) )
2022-10-16 16:42:39 +02:00
, m_anonymous ( node = = nullptr )
2019-06-15 22:49:44 +02:00
{
2022-10-16 16:42:39 +02:00
if ( node )
2022-10-17 14:41:50 +02:00
node - > set_layout_node ( { } , * this ) ;
2019-06-15 22:49:44 +02:00
}
2022-10-17 14:41:50 +02:00
Node : : ~ Node ( ) = default ;
void Node : : visit_edges ( Cell : : Visitor & visitor )
2019-06-15 22:49:44 +02:00
{
2022-10-17 14:41:50 +02:00
Base : : visit_edges ( visitor ) ;
visitor . visit ( m_dom_node ) ;
2023-01-11 12:51:49 +01:00
visitor . visit ( m_paintable ) ;
2023-09-16 23:30:15 +02:00
visitor . visit ( m_pseudo_element_generator ) ;
2022-10-20 18:20:34 +02:00
visitor . visit ( m_browsing_context ) ;
2022-10-17 14:41:50 +02:00
TreeNode : : visit_edges ( visitor ) ;
2019-06-15 22:49:44 +02:00
}
2022-03-22 18:34:02 +01:00
// https://www.w3.org/TR/css-display-3/#out-of-flow
bool Node : : is_out_of_flow ( FormattingContext const & formatting_context ) const
{
// A layout node is out of flow if either:
// 1. It is floated (which requires that floating is not inhibited).
if ( ! formatting_context . inhibits_floating ( ) & & computed_values ( ) . float_ ( ) ! = CSS : : Float : : None )
return true ;
// 2. It is "absolutely positioned".
if ( is_absolutely_positioned ( ) )
return true ;
return false ;
}
2020-11-22 15:53:01 +01:00
bool Node : : can_contain_boxes_with_position_absolute ( ) const
2020-06-05 16:54:28 +02:00
{
2023-02-24 02:51:08 +03:00
if ( computed_values ( ) . position ( ) ! = CSS : : Position : : Static )
return true ;
2023-02-25 11:04:29 +01:00
if ( is < Viewport > ( * this ) )
2023-02-24 02:51:08 +03:00
return true ;
// https://w3c.github.io/csswg-drafts/css-transforms-1/#propdef-transform
// Any computed value other than none for the transform affects containing block and stacking context
if ( ! computed_values ( ) . transformations ( ) . is_empty ( ) )
return true ;
return false ;
2020-06-05 16:54:28 +02:00
}
2023-01-23 14:59:51 +01:00
static Box const * nearest_ancestor_capable_of_forming_a_containing_block ( Node const & node )
{
for ( auto const * ancestor = node . parent ( ) ; ancestor ; ancestor = ancestor - > parent ( ) ) {
2023-01-23 15:03:45 +01:00
if ( ancestor - > is_block_container ( )
2023-01-23 15:19:32 +01:00
| | ancestor - > display ( ) . is_flex_inside ( )
2023-07-28 14:23:12 +02:00
| | ancestor - > display ( ) . is_grid_inside ( )
| | ancestor - > is_svg_svg_box ( ) ) {
2023-01-23 14:59:51 +01:00
return verify_cast < Box > ( ancestor ) ;
2023-01-23 15:03:45 +01:00
}
2023-01-23 14:59:51 +01:00
}
return nullptr ;
}
Box const * Node : : containing_block ( ) const
2019-07-01 07:28:37 +02:00
{
2021-01-01 18:55:47 +01:00
if ( is < TextNode > ( * this ) )
2023-01-23 14:59:51 +01:00
return nearest_ancestor_capable_of_forming_a_containing_block ( * this ) ;
2020-06-05 16:54:28 +02:00
2021-01-06 11:07:02 +01:00
auto position = computed_values ( ) . position ( ) ;
2020-06-12 14:19:03 +02:00
2023-01-23 14:59:51 +01:00
// https://drafts.csswg.org/css-position-3/#absolute-cb
2020-06-12 14:19:03 +02:00
if ( position = = CSS : : Position : : Absolute ) {
2023-01-23 14:59:51 +01:00
auto const * ancestor = parent ( ) ;
2020-06-05 16:54:28 +02:00
while ( ancestor & & ! ancestor - > can_contain_boxes_with_position_absolute ( ) )
ancestor = ancestor - > parent ( ) ;
2023-01-23 14:59:51 +01:00
while ( ancestor & & ancestor - > is_anonymous ( ) )
ancestor = nearest_ancestor_capable_of_forming_a_containing_block ( * ancestor ) ;
2023-02-19 12:28:50 +01:00
return static_cast < Box const * > ( ancestor ) ;
2020-06-05 16:54:28 +02:00
}
2020-06-12 14:19:03 +02:00
if ( position = = CSS : : Position : : Fixed )
2020-06-05 16:54:28 +02:00
return & root ( ) ;
2023-01-23 14:59:51 +01:00
return nearest_ancestor_capable_of_forming_a_containing_block ( * this ) ;
2019-07-01 07:28:37 +02:00
}
2019-09-25 12:40:37 +03:00
2023-06-14 15:31:40 +02:00
Box const * Node : : non_anonymous_containing_block ( ) const
2023-05-24 06:57:38 +00:00
{
auto nearest_ancestor_box = containing_block ( ) ;
VERIFY ( nearest_ancestor_box ) ;
while ( nearest_ancestor_box - > is_anonymous ( ) ) {
nearest_ancestor_box = nearest_ancestor_box - > containing_block ( ) ;
VERIFY ( nearest_ancestor_box ) ;
}
return nearest_ancestor_box ;
}
2022-10-23 17:57:31 +02:00
// https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Positioning/Understanding_z_index/The_stacking_context
2021-05-07 19:03:25 +03:00
bool Node : : establishes_stacking_context ( ) const
{
2022-10-23 17:57:31 +02:00
// NOTE: While MDN is not authoritative, there isn't a single convenient location
// in the CSS specifications where the rules for stacking contexts is described.
// That's why the "spec link" here points to MDN.
2021-05-07 19:03:25 +03:00
if ( ! has_style ( ) )
return false ;
2023-03-18 12:09:22 +01:00
// We make a stacking context for the viewport. Painting and hit testing starts from here.
if ( is_viewport ( ) )
return true ;
// Root element of the document (<html>).
if ( is_root_element ( ) )
2021-05-07 19:03:25 +03:00
return true ;
2023-03-18 12:09:22 +01:00
2021-05-07 19:03:25 +03:00
auto position = computed_values ( ) . position ( ) ;
2022-10-23 17:57:31 +02:00
// Element with a position value absolute or relative and z-index value other than auto.
if ( position = = CSS : : Position : : Absolute | | position = = CSS : : Position : : Relative ) {
if ( computed_values ( ) . z_index ( ) . has_value ( ) ) {
return true ;
}
}
// Element with a position value fixed or sticky.
if ( position = = CSS : : Position : : Fixed | | position = = CSS : : Position : : Sticky )
2021-05-07 19:03:25 +03:00
return true ;
2022-10-23 17:57:31 +02:00
2022-03-18 01:19:03 +01:00
if ( ! computed_values ( ) . transformations ( ) . is_empty ( ) )
return true ;
2022-09-13 19:51:47 +02:00
// Element that is a child of a flex container, with z-index value other than auto.
2022-10-06 16:02:53 +02:00
if ( parent ( ) & & parent ( ) - > display ( ) . is_flex_inside ( ) & & computed_values ( ) . z_index ( ) . has_value ( ) )
2022-09-13 19:51:47 +02:00
return true ;
// Element that is a child of a grid container, with z-index value other than auto.
2022-10-06 16:02:53 +02:00
if ( parent ( ) & & parent ( ) - > display ( ) . is_grid_inside ( ) & & computed_values ( ) . z_index ( ) . has_value ( ) )
2022-09-13 19:51:47 +02:00
return true ;
2022-10-24 16:05:10 +01:00
// https://drafts.fxtf.org/filter-effects-2/#backdrop-filter-operation
// A computed value of other than none results in the creation of both a stacking context [CSS21] and a Containing Block for absolute and fixed position descendants,
// unless the element it applies to is a document root element in the current browsing context.
// Spec Note: This rule works in the same way as for the filter property.
if ( ! computed_values ( ) . backdrop_filter ( ) . is_none ( ) )
return true ;
2021-10-19 15:27:40 +02:00
return computed_values ( ) . opacity ( ) < 1.0f ;
2021-05-07 19:03:25 +03:00
}
2021-11-18 15:01:28 +01:00
HTML : : BrowsingContext const & Node : : browsing_context ( ) const
2020-06-14 16:45:45 +02:00
{
2022-10-20 18:20:34 +02:00
return * m_browsing_context ;
2020-06-14 16:45:45 +02:00
}
2021-11-18 15:01:28 +01:00
HTML : : BrowsingContext & Node : : browsing_context ( )
2020-06-14 16:45:45 +02:00
{
2022-10-20 18:20:34 +02:00
return * m_browsing_context ;
2020-06-14 16:45:45 +02:00
}
2023-08-22 16:00:42 +02:00
JS : : GCPtr < HTML : : Navigable > Node : : navigable ( ) const
{
return document ( ) . navigable ( ) ;
}
2023-02-25 11:04:29 +01:00
Viewport const & Node : : root ( ) const
2019-11-04 19:37:52 +01:00
{
2021-02-23 20:42:32 +01:00
VERIFY ( document ( ) . layout_node ( ) ) ;
2019-11-04 19:37:52 +01:00
return * document ( ) . layout_node ( ) ;
}
2023-02-25 11:04:29 +01:00
Viewport & Node : : root ( )
2019-11-04 19:37:52 +01:00
{
2021-02-23 20:42:32 +01:00
VERIFY ( document ( ) . layout_node ( ) ) ;
2019-11-04 19:37:52 +01:00
return * document ( ) . layout_node ( ) ;
}
2020-11-22 15:53:01 +01:00
void Node : : set_needs_display ( )
2019-10-09 21:25:29 +02:00
{
2022-09-27 15:16:53 +02:00
auto * containing_block = this - > containing_block ( ) ;
if ( ! containing_block )
return ;
2023-04-20 16:00:42 +01:00
if ( ! containing_block - > paintable_box ( ) )
2022-09-27 15:16:53 +02:00
return ;
2023-04-20 16:00:42 +01:00
if ( ! is < Painting : : PaintableWithLines > ( * containing_block - > paintable_box ( ) ) )
2023-01-23 14:59:51 +01:00
return ;
2023-04-20 16:00:42 +01:00
static_cast < Painting : : PaintableWithLines const & > ( * containing_block - > paintable_box ( ) ) . for_each_fragment ( [ & ] ( auto & fragment ) {
2022-09-27 15:16:53 +02:00
if ( & fragment . layout_node ( ) = = this | | is_ancestor_of ( fragment . layout_node ( ) ) ) {
2023-08-22 16:00:42 +02:00
if ( navigable ( ) )
navigable ( ) - > set_needs_display ( fragment . absolute_rect ( ) ) ;
2022-09-27 15:16:53 +02:00
}
return IterationDecision : : Continue ;
} ) ;
2019-10-20 09:14:12 +02:00
}
2022-11-08 11:31:24 +00:00
CSSPixelPoint Node : : box_type_agnostic_position ( ) const
2019-10-20 09:14:12 +02:00
{
2021-01-01 18:55:47 +01:00
if ( is < Box > ( * this ) )
2023-04-20 16:00:42 +01:00
return verify_cast < Box > ( * this ) . paintable_box ( ) - > absolute_position ( ) ;
2021-02-23 20:42:32 +01:00
VERIFY ( is_inline ( ) ) ;
2022-11-08 11:31:24 +00:00
CSSPixelPoint position ;
2023-08-15 15:51:54 +02:00
if ( auto * block = containing_block ( ) ; block & & block - > paintable ( ) & & is < Painting : : PaintableWithLines > ( * block - > paintable ( ) ) ) {
static_cast < Painting : : PaintableWithLines const & > ( * block - > paintable_box ( ) ) . for_each_fragment ( [ & ] ( auto & fragment ) {
if ( & fragment . layout_node ( ) = = this | | is_ancestor_of ( fragment . layout_node ( ) ) ) {
position = fragment . absolute_rect ( ) . location ( ) ;
return IterationDecision : : Break ;
}
return IterationDecision : : Continue ;
} ) ;
2019-10-20 09:14:12 +02:00
}
return position ;
2019-10-09 21:25:29 +02:00
}
2020-03-07 10:27:02 +01:00
2020-11-22 15:53:01 +01:00
bool Node : : is_floating ( ) const
2020-06-26 15:08:42 +02:00
{
if ( ! has_style ( ) )
return false ;
2021-05-29 23:03:05 +02:00
// flex-items don't float.
if ( is_flex_item ( ) )
return false ;
2021-01-06 11:07:02 +01:00
return computed_values ( ) . float_ ( ) ! = CSS : : Float : : None ;
2020-06-26 15:08:42 +02:00
}
2020-12-06 19:54:23 +01:00
bool Node : : is_positioned ( ) const
{
2021-01-06 11:07:02 +01:00
return has_style ( ) & & computed_values ( ) . position ( ) ! = CSS : : Position : : Static ;
2020-12-06 19:54:23 +01:00
}
2020-11-22 15:53:01 +01:00
bool Node : : is_absolutely_positioned ( ) const
2020-06-05 16:54:28 +02:00
{
2020-06-15 12:57:43 +02:00
if ( ! has_style ( ) )
return false ;
2021-01-06 11:07:02 +01:00
auto position = computed_values ( ) . position ( ) ;
2020-06-12 14:19:03 +02:00
return position = = CSS : : Position : : Absolute | | position = = CSS : : Position : : Fixed ;
}
2020-11-22 15:53:01 +01:00
bool Node : : is_fixed_position ( ) const
2020-06-12 14:19:03 +02:00
{
2020-06-15 12:57:43 +02:00
if ( ! has_style ( ) )
return false ;
2021-01-06 11:07:02 +01:00
auto position = computed_values ( ) . position ( ) ;
2020-06-23 23:15:23 +02:00
return position = = CSS : : Position : : Fixed ;
}
2022-03-24 17:38:05 +01:00
NodeWithStyle : : NodeWithStyle ( DOM : : Document & document , DOM : : Node * node , NonnullRefPtr < CSS : : StyleProperties > computed_style )
2020-11-22 15:53:01 +01:00
: Node ( document , node )
2020-06-23 23:15:23 +02:00
{
m_has_style = true ;
2022-03-24 17:38:05 +01:00
apply_style ( * computed_style ) ;
2021-01-06 14:10:53 +01:00
}
NodeWithStyle : : NodeWithStyle ( DOM : : Document & document , DOM : : Node * node , CSS : : ComputedValues computed_values )
: Node ( document , node )
, m_computed_values ( move ( computed_values ) )
{
m_has_style = true ;
2022-09-17 21:25:50 +02:00
m_font = Platform : : FontPlugin : : the ( ) . default_font ( ) ;
2020-06-24 14:17:05 +02:00
}
2023-07-30 16:15:38 +01:00
// https://www.w3.org/TR/css-values-4/#snap-a-length-as-a-border-width
static CSSPixels snap_a_length_as_a_border_width ( double device_pixels_per_css_pixel , CSSPixels length )
{
// 1. Assert: len is non-negative.
VERIFY ( length > = 0 ) ;
// 2. If len is an integer number of device pixels, do nothing.
auto device_pixels = length . to_double ( ) * device_pixels_per_css_pixel ;
if ( device_pixels = = trunc ( device_pixels ) )
return length ;
// 3. If len is greater than zero, but less than 1 device pixel, round len up to 1 device pixel.
if ( device_pixels > 0 & & device_pixels < 1 )
2023-08-26 15:57:31 +01:00
return CSSPixels : : nearest_value_for ( 1 / device_pixels_per_css_pixel ) ;
2023-07-30 16:15:38 +01:00
// 4. If len is greater than 1 device pixel, round it down to the nearest integer number of device pixels.
if ( device_pixels > 1 )
2023-08-26 15:57:31 +01:00
return CSSPixels : : nearest_value_for ( floor ( device_pixels ) / device_pixels_per_css_pixel ) ;
2023-07-30 16:15:38 +01:00
return length ;
}
2022-03-24 17:38:05 +01:00
void NodeWithStyle : : apply_style ( const CSS : : StyleProperties & computed_style )
2020-06-24 14:17:05 +02:00
{
2021-01-06 10:34:31 +01:00
auto & computed_values = static_cast < CSS : : MutableComputedValues & > ( m_computed_values ) ;
2020-06-24 14:17:05 +02:00
2023-03-05 22:31:44 +00:00
// NOTE: color must be set first to ensure currentColor can be resolved in other properties (e.g. background-color).
computed_values . set_color ( computed_style . color_or_fallback ( CSS : : PropertyID : : Color , * this , CSS : : InitialValues : : color ( ) ) ) ;
2022-03-23 15:57:05 +01:00
// NOTE: We have to be careful that font-related properties get set in the right order.
// m_font is used by Length::to_px() when resolving sizes against this layout node.
// That's why it has to be set before everything else.
2022-03-24 17:38:05 +01:00
m_font = computed_style . computed_font ( ) ;
2023-08-31 19:53:41 -05:00
computed_values . set_font_size ( computed_style . property ( CSS : : PropertyID : : FontSize ) - > as_length ( ) . length ( ) . to_px ( * this ) ) ;
2023-06-01 17:01:09 +01:00
computed_values . set_font_weight ( round_to < int > ( computed_style . property ( CSS : : PropertyID : : FontWeight ) - > as_number ( ) . number ( ) ) ) ;
2022-03-24 17:38:05 +01:00
m_line_height = computed_style . line_height ( * this ) ;
2021-01-06 11:31:19 +01:00
2022-03-24 17:38:05 +01:00
computed_values . set_vertical_align ( computed_style . vertical_align ( ) ) ;
2022-02-26 01:34:07 +01:00
2021-11-12 12:11:01 +00:00
{
2022-03-24 17:38:05 +01:00
auto attachments = computed_style . property ( CSS : : PropertyID : : BackgroundAttachment ) ;
auto clips = computed_style . property ( CSS : : PropertyID : : BackgroundClip ) ;
auto images = computed_style . property ( CSS : : PropertyID : : BackgroundImage ) ;
auto origins = computed_style . property ( CSS : : PropertyID : : BackgroundOrigin ) ;
2023-04-03 00:04:00 +01:00
auto x_positions = computed_style . property ( CSS : : PropertyID : : BackgroundPositionX ) ;
auto y_positions = computed_style . property ( CSS : : PropertyID : : BackgroundPositionY ) ;
2022-03-24 17:38:05 +01:00
auto repeats = computed_style . property ( CSS : : PropertyID : : BackgroundRepeat ) ;
auto sizes = computed_style . property ( CSS : : PropertyID : : BackgroundSize ) ;
2021-11-12 12:11:01 +00:00
auto count_layers = [ ] ( auto maybe_style_value ) - > size_t {
2022-04-14 11:52:35 +01:00
if ( maybe_style_value - > is_value_list ( ) )
return maybe_style_value - > as_value_list ( ) . size ( ) ;
2021-11-12 12:11:01 +00:00
else
return 1 ;
} ;
2023-02-20 18:56:08 +01:00
auto value_for_layer = [ ] ( auto & style_value , size_t layer_index ) - > RefPtr < CSS : : StyleValue const > {
2021-11-12 12:11:01 +00:00
if ( style_value - > is_value_list ( ) )
return style_value - > as_value_list ( ) . value_at ( layer_index , true ) ;
return style_value ;
} ;
size_t layer_count = 1 ;
layer_count = max ( layer_count , count_layers ( attachments ) ) ;
layer_count = max ( layer_count , count_layers ( clips ) ) ;
layer_count = max ( layer_count , count_layers ( images ) ) ;
layer_count = max ( layer_count , count_layers ( origins ) ) ;
2023-04-03 00:04:00 +01:00
layer_count = max ( layer_count , count_layers ( x_positions ) ) ;
layer_count = max ( layer_count , count_layers ( y_positions ) ) ;
2021-11-12 12:11:01 +00:00
layer_count = max ( layer_count , count_layers ( repeats ) ) ;
layer_count = max ( layer_count , count_layers ( sizes ) ) ;
Vector < CSS : : BackgroundLayerData > layers ;
layers . ensure_capacity ( layer_count ) ;
for ( size_t layer_index = 0 ; layer_index < layer_count ; layer_index + + ) {
CSS : : BackgroundLayerData layer ;
2022-07-12 00:28:19 +01:00
if ( auto image_value = value_for_layer ( images , layer_index ) ; image_value ) {
2022-07-31 01:11:59 +01:00
if ( image_value - > is_abstract_image ( ) ) {
layer . background_image = image_value - > as_abstract_image ( ) ;
2023-02-20 18:56:08 +01:00
const_cast < CSS : : AbstractImageStyleValue & > ( * layer . background_image ) . load_any_resources ( document ( ) ) ;
2022-07-12 00:28:19 +01:00
}
2021-11-12 12:11:01 +00:00
}
2023-04-19 12:00:38 +01:00
if ( auto attachment_value = value_for_layer ( attachments , layer_index ) ; attachment_value & & attachment_value - > is_identifier ( ) ) {
2021-11-12 16:30:21 +00:00
switch ( attachment_value - > to_identifier ( ) ) {
case CSS : : ValueID : : Fixed :
layer . attachment = CSS : : BackgroundAttachment : : Fixed ;
break ;
case CSS : : ValueID : : Local :
layer . attachment = CSS : : BackgroundAttachment : : Local ;
break ;
case CSS : : ValueID : : Scroll :
layer . attachment = CSS : : BackgroundAttachment : : Scroll ;
break ;
default :
break ;
}
}
auto as_box = [ ] ( auto value_id ) {
switch ( value_id ) {
case CSS : : ValueID : : BorderBox :
return CSS : : BackgroundBox : : BorderBox ;
case CSS : : ValueID : : ContentBox :
return CSS : : BackgroundBox : : ContentBox ;
case CSS : : ValueID : : PaddingBox :
return CSS : : BackgroundBox : : PaddingBox ;
default :
VERIFY_NOT_REACHED ( ) ;
}
} ;
2023-04-19 12:00:38 +01:00
if ( auto origin_value = value_for_layer ( origins , layer_index ) ; origin_value & & origin_value - > is_identifier ( ) ) {
2021-11-12 16:30:21 +00:00
layer . origin = as_box ( origin_value - > to_identifier ( ) ) ;
}
2023-04-19 12:00:38 +01:00
if ( auto clip_value = value_for_layer ( clips , layer_index ) ; clip_value & & clip_value - > is_identifier ( ) ) {
2021-11-12 16:30:21 +00:00
layer . clip = as_box ( clip_value - > to_identifier ( ) ) ;
}
2023-04-03 00:04:00 +01:00
if ( auto position_value = value_for_layer ( x_positions , layer_index ) ; position_value & & position_value - > is_edge ( ) ) {
auto & position = position_value - > as_edge ( ) ;
layer . position_edge_x = position . edge ( ) ;
layer . position_offset_x = position . offset ( ) ;
2021-11-12 16:30:21 +00:00
}
2023-04-03 00:04:00 +01:00
if ( auto position_value = value_for_layer ( y_positions , layer_index ) ; position_value & & position_value - > is_edge ( ) ) {
auto & position = position_value - > as_edge ( ) ;
layer . position_edge_y = position . edge ( ) ;
layer . position_offset_y = position . offset ( ) ;
} ;
2021-11-12 16:30:21 +00:00
if ( auto size_value = value_for_layer ( sizes , layer_index ) ; size_value ) {
if ( size_value - > is_background_size ( ) ) {
auto & size = size_value - > as_background_size ( ) ;
layer . size_type = CSS : : BackgroundSize : : LengthPercentage ;
layer . size_x = size . size_x ( ) ;
layer . size_y = size . size_y ( ) ;
2023-04-19 12:00:38 +01:00
} else if ( size_value - > is_identifier ( ) ) {
2021-11-12 16:30:21 +00:00
switch ( size_value - > to_identifier ( ) ) {
case CSS : : ValueID : : Contain :
layer . size_type = CSS : : BackgroundSize : : Contain ;
break ;
case CSS : : ValueID : : Cover :
layer . size_type = CSS : : BackgroundSize : : Cover ;
break ;
default :
break ;
}
}
}
2021-11-12 12:11:01 +00:00
if ( auto repeat_value = value_for_layer ( repeats , layer_index ) ; repeat_value & & repeat_value - > is_background_repeat ( ) ) {
layer . repeat_x = repeat_value - > as_background_repeat ( ) . repeat_x ( ) ;
layer . repeat_y = repeat_value - > as_background_repeat ( ) . repeat_y ( ) ;
}
layers . append ( move ( layer ) ) ;
}
computed_values . set_background_layers ( move ( layers ) ) ;
}
2022-03-24 17:38:05 +01:00
computed_values . set_background_color ( computed_style . color_or_fallback ( CSS : : PropertyID : : BackgroundColor , * this , CSS : : InitialValues : : background_color ( ) ) ) ;
2021-11-12 12:11:01 +00:00
2022-04-13 19:33:56 +01:00
if ( auto box_sizing = computed_style . box_sizing ( ) ; box_sizing . has_value ( ) )
computed_values . set_box_sizing ( box_sizing . release_value ( ) ) ;
2021-10-05 16:55:02 +01:00
2022-03-24 17:38:05 +01:00
if ( auto maybe_font_variant = computed_style . font_variant ( ) ; maybe_font_variant . has_value ( ) )
2022-03-23 14:54:21 +01:00
computed_values . set_font_variant ( maybe_font_variant . release_value ( ) ) ;
2021-08-06 16:55:08 +01:00
// FIXME: BorderXRadius properties are now BorderRadiusStyleValues, so make use of that.
2022-03-24 17:38:05 +01:00
auto border_bottom_left_radius = computed_style . property ( CSS : : PropertyID : : BorderBottomLeftRadius ) ;
2022-06-12 15:05:47 +01:00
if ( border_bottom_left_radius - > is_border_radius ( ) ) {
computed_values . set_border_bottom_left_radius (
CSS : : BorderRadiusData {
border_bottom_left_radius - > as_border_radius ( ) . horizontal_radius ( ) ,
border_bottom_left_radius - > as_border_radius ( ) . vertical_radius ( ) } ) ;
}
2022-03-24 17:38:05 +01:00
auto border_bottom_right_radius = computed_style . property ( CSS : : PropertyID : : BorderBottomRightRadius ) ;
2022-06-12 15:05:47 +01:00
if ( border_bottom_right_radius - > is_border_radius ( ) ) {
computed_values . set_border_bottom_right_radius (
CSS : : BorderRadiusData {
border_bottom_right_radius - > as_border_radius ( ) . horizontal_radius ( ) ,
border_bottom_right_radius - > as_border_radius ( ) . vertical_radius ( ) } ) ;
}
2022-03-24 17:38:05 +01:00
auto border_top_left_radius = computed_style . property ( CSS : : PropertyID : : BorderTopLeftRadius ) ;
2022-06-12 15:05:47 +01:00
if ( border_top_left_radius - > is_border_radius ( ) ) {
computed_values . set_border_top_left_radius (
CSS : : BorderRadiusData {
border_top_left_radius - > as_border_radius ( ) . horizontal_radius ( ) ,
border_top_left_radius - > as_border_radius ( ) . vertical_radius ( ) } ) ;
}
2022-03-24 17:38:05 +01:00
auto border_top_right_radius = computed_style . property ( CSS : : PropertyID : : BorderTopRightRadius ) ;
2022-06-12 15:05:47 +01:00
if ( border_top_right_radius - > is_border_radius ( ) ) {
computed_values . set_border_top_right_radius (
CSS : : BorderRadiusData {
border_top_right_radius - > as_border_radius ( ) . horizontal_radius ( ) ,
border_top_right_radius - > as_border_radius ( ) . vertical_radius ( ) } ) ;
}
2022-03-24 17:38:05 +01:00
computed_values . set_display ( computed_style . display ( ) ) ;
2021-01-07 14:41:50 +01:00
2022-03-24 17:38:05 +01:00
auto flex_direction = computed_style . flex_direction ( ) ;
2021-01-18 17:41:57 +01:00
if ( flex_direction . has_value ( ) )
computed_values . set_flex_direction ( flex_direction . value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto flex_wrap = computed_style . flex_wrap ( ) ;
2021-05-30 12:11:32 +02:00
if ( flex_wrap . has_value ( ) )
computed_values . set_flex_wrap ( flex_wrap . value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto flex_basis = computed_style . flex_basis ( ) ;
2021-05-30 14:23:43 +02:00
if ( flex_basis . has_value ( ) )
computed_values . set_flex_basis ( flex_basis . value ( ) ) ;
2022-03-24 17:38:05 +01:00
computed_values . set_flex_grow ( computed_style . flex_grow ( ) ) ;
computed_values . set_flex_shrink ( computed_style . flex_shrink ( ) ) ;
2022-03-31 22:11:38 +02:00
computed_values . set_order ( computed_style . order ( ) ) ;
2022-07-31 18:47:09 +02:00
computed_values . set_clip ( computed_style . clip ( ) ) ;
2022-09-15 08:31:19 +01:00
computed_values . set_backdrop_filter ( computed_style . backdrop_filter ( ) ) ;
2021-05-30 20:22:25 +02:00
2022-03-24 17:38:05 +01:00
auto justify_content = computed_style . justify_content ( ) ;
2021-07-16 18:38:26 +02:00
if ( justify_content . has_value ( ) )
computed_values . set_justify_content ( justify_content . value ( ) ) ;
2023-07-14 20:49:22 +02:00
auto justify_items = computed_style . justify_items ( ) ;
if ( justify_items . has_value ( ) )
computed_values . set_justify_items ( justify_items . value ( ) ) ;
2023-07-14 14:41:22 +02:00
auto justify_self = computed_style . justify_self ( ) ;
if ( justify_self . has_value ( ) )
computed_values . set_justify_self ( justify_self . value ( ) ) ;
2023-03-18 20:49:00 +01:00
auto accent_color = computed_style . accent_color ( * this ) ;
if ( accent_color . has_value ( ) )
computed_values . set_accent_color ( accent_color . value ( ) ) ;
2022-10-14 13:50:06 +02:00
auto align_content = computed_style . align_content ( ) ;
if ( align_content . has_value ( ) )
computed_values . set_align_content ( align_content . value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto align_items = computed_style . align_items ( ) ;
2021-09-15 18:27:20 +02:00
if ( align_items . has_value ( ) )
computed_values . set_align_items ( align_items . value ( ) ) ;
2022-07-11 23:52:36 +02:00
auto align_self = computed_style . align_self ( ) ;
if ( align_self . has_value ( ) )
computed_values . set_align_self ( align_self . value ( ) ) ;
2022-07-22 16:05:11 +01:00
auto appearance = computed_style . appearance ( ) ;
if ( appearance . has_value ( ) )
computed_values . set_appearance ( appearance . value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto position = computed_style . position ( ) ;
2021-10-13 18:54:00 +02:00
if ( position . has_value ( ) )
2021-01-06 10:34:31 +01:00
computed_values . set_position ( position . value ( ) ) ;
2020-12-14 18:38:02 +01:00
2022-03-24 17:38:05 +01:00
auto text_align = computed_style . text_align ( ) ;
2020-12-14 18:38:02 +01:00
if ( text_align . has_value ( ) )
2021-01-06 10:34:31 +01:00
computed_values . set_text_align ( text_align . value ( ) ) ;
2020-06-24 16:37:44 +02:00
2022-03-24 17:38:05 +01:00
auto text_justify = computed_style . text_justify ( ) ;
2022-03-12 19:31:32 +00:00
if ( text_align . has_value ( ) )
computed_values . set_text_justify ( text_justify . value ( ) ) ;
2023-05-15 16:42:28 +02:00
if ( auto text_indent = computed_style . length_percentage ( CSS : : PropertyID : : TextIndent ) ; text_indent . has_value ( ) )
computed_values . set_text_indent ( text_indent . release_value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto white_space = computed_style . white_space ( ) ;
2020-06-24 16:37:44 +02:00
if ( white_space . has_value ( ) )
2021-01-06 10:34:31 +01:00
computed_values . set_white_space ( white_space . value ( ) ) ;
2020-06-24 16:37:44 +02:00
2022-03-24 17:38:05 +01:00
auto float_ = computed_style . float_ ( ) ;
2020-06-26 15:08:42 +02:00
if ( float_ . has_value ( ) )
2021-01-06 10:34:31 +01:00
computed_values . set_float ( float_ . value ( ) ) ;
2020-06-26 15:08:42 +02:00
2023-06-16 02:35:03 +00:00
computed_values . set_border_spacing_horizontal ( computed_style . border_spacing_horizontal ( ) ) ;
computed_values . set_border_spacing_vertical ( computed_style . border_spacing_vertical ( ) ) ;
2023-06-07 02:10:55 +00:00
auto caption_side = computed_style . caption_side ( ) ;
if ( caption_side . has_value ( ) )
computed_values . set_caption_side ( caption_side . value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto clear = computed_style . clear ( ) ;
2020-12-06 01:45:51 +01:00
if ( clear . has_value ( ) )
2021-01-06 10:34:31 +01:00
computed_values . set_clear ( clear . value ( ) ) ;
2020-12-06 01:45:51 +01:00
2022-03-24 17:38:05 +01:00
auto overflow_x = computed_style . overflow_x ( ) ;
2021-02-22 15:20:31 +01:00
if ( overflow_x . has_value ( ) )
computed_values . set_overflow_x ( overflow_x . value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto overflow_y = computed_style . overflow_y ( ) ;
2021-02-22 15:20:31 +01:00
if ( overflow_y . has_value ( ) )
computed_values . set_overflow_y ( overflow_y . value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto cursor = computed_style . cursor ( ) ;
2021-02-21 17:41:00 +00:00
if ( cursor . has_value ( ) )
computed_values . set_cursor ( cursor . value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto image_rendering = computed_style . image_rendering ( ) ;
2022-02-18 12:21:27 +01:00
if ( image_rendering . has_value ( ) )
computed_values . set_image_rendering ( image_rendering . value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto pointer_events = computed_style . pointer_events ( ) ;
2021-10-05 19:47:13 +01:00
if ( pointer_events . has_value ( ) )
computed_values . set_pointer_events ( pointer_events . value ( ) ) ;
2022-04-14 16:22:35 +01:00
computed_values . set_text_decoration_line ( computed_style . text_decoration_line ( ) ) ;
2020-12-15 13:36:27 +01:00
2022-03-24 17:38:05 +01:00
auto text_decoration_style = computed_style . text_decoration_style ( ) ;
2022-01-20 20:27:55 +01:00
if ( text_decoration_style . has_value ( ) )
computed_values . set_text_decoration_style ( text_decoration_style . value ( ) ) ;
2022-03-24 17:38:05 +01:00
auto text_transform = computed_style . text_transform ( ) ;
2020-12-15 14:15:49 +01:00
if ( text_transform . has_value ( ) )
2021-01-06 10:34:31 +01:00
computed_values . set_text_transform ( text_transform . value ( ) ) ;
2020-12-15 14:15:49 +01:00
2022-03-24 17:38:05 +01:00
if ( auto list_style_type = computed_style . list_style_type ( ) ; list_style_type . has_value ( ) )
2021-01-06 10:34:31 +01:00
computed_values . set_list_style_type ( list_style_type . value ( ) ) ;
2020-12-15 16:50:39 +01:00
2022-03-24 17:38:05 +01:00
auto list_style_image = computed_style . property ( CSS : : PropertyID : : ListStyleImage ) ;
2022-07-31 01:11:59 +01:00
if ( list_style_image - > is_abstract_image ( ) ) {
m_list_style_image = list_style_image - > as_abstract_image ( ) ;
2023-02-20 18:56:08 +01:00
const_cast < CSS : : AbstractImageStyleValue & > ( * m_list_style_image ) . load_any_resources ( document ( ) ) ;
2021-10-29 09:00:30 -04:00
}
2023-06-02 23:05:15 +02:00
if ( auto list_style_position = computed_style . list_style_position ( ) ; list_style_position . has_value ( ) )
computed_values . set_list_style_position ( list_style_position . value ( ) ) ;
2022-03-06 00:25:42 +01:00
// FIXME: The default text decoration color value is `currentcolor`, but since we can't resolve that easily,
// we just manually grab the value from `color`. This makes it dependent on `color` being
// specified first, so it's far from ideal.
2022-03-24 17:38:05 +01:00
computed_values . set_text_decoration_color ( computed_style . color_or_fallback ( CSS : : PropertyID : : TextDecorationColor , * this , computed_values . color ( ) ) ) ;
if ( auto maybe_text_decoration_thickness = computed_style . length_percentage ( CSS : : PropertyID : : TextDecorationThickness ) ; maybe_text_decoration_thickness . has_value ( ) )
2022-03-06 19:48:09 +01:00
computed_values . set_text_decoration_thickness ( maybe_text_decoration_thickness . release_value ( ) ) ;
2022-03-06 00:25:42 +01:00
2023-05-31 16:07:06 -04:00
computed_values . set_text_shadow ( computed_style . text_shadow ( * this ) ) ;
2022-03-23 21:16:36 +00:00
2022-03-24 17:38:05 +01:00
computed_values . set_z_index ( computed_style . z_index ( ) ) ;
computed_values . set_opacity ( computed_style . opacity ( ) ) ;
2022-03-21 15:42:57 +01:00
2022-03-24 17:38:05 +01:00
if ( auto maybe_visibility = computed_style . visibility ( ) ; maybe_visibility . has_value ( ) )
2022-03-21 15:42:57 +01:00
computed_values . set_visibility ( maybe_visibility . release_value ( ) ) ;
2022-09-15 21:01:43 +02:00
m_visible = computed_values . opacity ( ) ! = 0 & & computed_values . visibility ( ) = = CSS : : Visibility : : Visible ;
2021-05-30 13:43:47 +02:00
2022-09-25 15:48:23 +02:00
computed_values . set_width ( computed_style . size_value ( CSS : : PropertyID : : Width ) ) ;
computed_values . set_min_width ( computed_style . size_value ( CSS : : PropertyID : : MinWidth ) ) ;
computed_values . set_max_width ( computed_style . size_value ( CSS : : PropertyID : : MaxWidth ) ) ;
computed_values . set_height ( computed_style . size_value ( CSS : : PropertyID : : Height ) ) ;
computed_values . set_min_height ( computed_style . size_value ( CSS : : PropertyID : : MinHeight ) ) ;
computed_values . set_max_height ( computed_style . size_value ( CSS : : PropertyID : : MaxHeight ) ) ;
2020-06-24 17:45:42 +02:00
2022-03-27 15:42:23 +02:00
computed_values . set_inset ( computed_style . length_box ( CSS : : PropertyID : : Left , CSS : : PropertyID : : Top , CSS : : PropertyID : : Right , CSS : : PropertyID : : Bottom , CSS : : Length : : make_auto ( ) ) ) ;
2022-03-24 17:38:05 +01:00
computed_values . set_margin ( computed_style . length_box ( CSS : : PropertyID : : MarginLeft , CSS : : PropertyID : : MarginTop , CSS : : PropertyID : : MarginRight , CSS : : PropertyID : : MarginBottom , CSS : : Length : : make_px ( 0 ) ) ) ;
computed_values . set_padding ( computed_style . length_box ( CSS : : PropertyID : : PaddingLeft , CSS : : PropertyID : : PaddingTop , CSS : : PropertyID : : PaddingRight , CSS : : PropertyID : : PaddingBottom , CSS : : Length : : make_px ( 0 ) ) ) ;
2020-06-24 19:41:12 +02:00
2023-05-31 16:07:06 -04:00
computed_values . set_box_shadow ( computed_style . box_shadow ( * this ) ) ;
2021-07-23 21:22:31 +02:00
2022-03-24 17:38:05 +01:00
computed_values . set_transformations ( computed_style . transformations ( ) ) ;
computed_values . set_transform_origin ( computed_style . transform_origin ( ) ) ;
2021-09-18 17:20:00 +02:00
2023-05-27 22:05:48 +02:00
auto transition_delay_property = computed_style . property ( CSS : : PropertyID : : TransitionDelay ) ;
if ( transition_delay_property - > is_time ( ) ) {
auto & transition_delay = transition_delay_property - > as_time ( ) ;
computed_values . set_transition_delay ( transition_delay . time ( ) ) ;
} else if ( transition_delay_property - > is_calculated ( ) ) {
auto & transition_delay = transition_delay_property - > as_calculated ( ) ;
computed_values . set_transition_delay ( transition_delay . resolve_time ( ) . value ( ) ) ;
}
2021-01-06 10:34:31 +01:00
auto do_border_style = [ & ] ( CSS : : BorderData & border , CSS : : PropertyID width_property , CSS : : PropertyID color_property , CSS : : PropertyID style_property ) {
2021-09-17 20:04:35 +01:00
// FIXME: The default border color value is `currentcolor`, but since we can't resolve that easily,
// we just manually grab the value from `color`. This makes it dependent on `color` being
// specified first, so it's far from ideal.
2022-03-24 17:38:05 +01:00
border . color = computed_style . color_or_fallback ( color_property , * this , computed_values . color ( ) ) ;
border . line_style = computed_style . line_style ( style_property ) . value_or ( CSS : : LineStyle : : None ) ;
2022-12-31 16:53:33 +00:00
// https://w3c.github.io/csswg-drafts/css-backgrounds/#border-style
// none
// No border. Color and width are ignored (i.e., the border has width 0). Note this means that the initial value of border-image-width will also resolve to zero.
// hidden
// Same as none, but has different behavior in the border conflict resolution rules for border-collapsed tables [CSS2].
if ( border . line_style = = CSS : : LineStyle : : None | | border . line_style = = CSS : : LineStyle : : Hidden ) {
2021-04-12 17:23:17 +03:00
border . width = 0 ;
2022-11-24 21:34:28 +01:00
} else {
2023-07-30 15:54:57 +01:00
auto resolve_border_width = [ & ] ( ) - > CSSPixels {
2022-11-24 21:34:28 +01:00
auto value = computed_style . property ( width_property ) ;
if ( value - > is_calculated ( ) )
2023-07-30 15:54:57 +01:00
return value - > as_calculated ( ) . resolve_length ( * this ) - > to_px ( * this ) ;
2023-05-27 12:28:25 +01:00
if ( value - > is_length ( ) )
2023-07-30 15:54:57 +01:00
return value - > as_length ( ) . length ( ) . to_px ( * this ) ;
2022-11-24 21:34:28 +01:00
if ( value - > is_identifier ( ) ) {
2023-02-23 16:18:55 +00:00
// https://www.w3.org/TR/css-backgrounds-3/#valdef-line-width-thin
2022-11-24 21:34:28 +01:00
switch ( value - > to_identifier ( ) ) {
case CSS : : ValueID : : Thin :
2023-08-26 15:03:04 +01:00
return 1 ;
2022-11-24 21:34:28 +01:00
case CSS : : ValueID : : Medium :
2023-08-26 15:03:04 +01:00
return 3 ;
2022-11-24 21:34:28 +01:00
case CSS : : ValueID : : Thick :
2023-08-26 15:03:04 +01:00
return 5 ;
2022-11-24 21:34:28 +01:00
default :
VERIFY_NOT_REACHED ( ) ;
}
}
VERIFY_NOT_REACHED ( ) ;
} ;
2023-07-30 16:15:38 +01:00
border . width = snap_a_length_as_a_border_width ( document ( ) . page ( ) - > client ( ) . device_pixels_per_css_pixel ( ) , resolve_border_width ( ) ) ;
2022-11-24 21:34:28 +01:00
}
2020-12-04 16:11:55 +01:00
} ;
2021-01-06 10:34:31 +01:00
do_border_style ( computed_values . border_left ( ) , CSS : : PropertyID : : BorderLeftWidth , CSS : : PropertyID : : BorderLeftColor , CSS : : PropertyID : : BorderLeftStyle ) ;
do_border_style ( computed_values . border_top ( ) , CSS : : PropertyID : : BorderTopWidth , CSS : : PropertyID : : BorderTopColor , CSS : : PropertyID : : BorderTopStyle ) ;
do_border_style ( computed_values . border_right ( ) , CSS : : PropertyID : : BorderRightWidth , CSS : : PropertyID : : BorderRightColor , CSS : : PropertyID : : BorderRightStyle ) ;
do_border_style ( computed_values . border_bottom ( ) , CSS : : PropertyID : : BorderBottomWidth , CSS : : PropertyID : : BorderBottomColor , CSS : : PropertyID : : BorderBottomStyle ) ;
2021-09-16 12:28:14 +01:00
2023-08-02 17:24:14 +01:00
if ( auto outline_color = computed_style . property ( CSS : : PropertyID : : OutlineColor ) ; outline_color - > has_color ( ) )
computed_values . set_outline_color ( outline_color - > to_color ( * this ) ) ;
2023-08-02 20:09:10 +01:00
if ( auto outline_offset = computed_style . property ( CSS : : PropertyID : : OutlineOffset ) ; outline_offset - > is_length ( ) )
computed_values . set_outline_offset ( outline_offset - > as_length ( ) . length ( ) ) ;
2023-08-02 17:24:14 +01:00
if ( auto outline_style = computed_style . outline_style ( ) ; outline_style . has_value ( ) )
computed_values . set_outline_style ( outline_style . value ( ) ) ;
if ( auto outline_width = computed_style . property ( CSS : : PropertyID : : OutlineWidth ) ; outline_width - > is_length ( ) )
computed_values . set_outline_width ( outline_width - > as_length ( ) . length ( ) ) ;
2022-03-24 17:38:05 +01:00
computed_values . set_content ( computed_style . content ( ) ) ;
2023-05-22 10:25:07 +03:00
computed_values . set_grid_auto_columns ( computed_style . grid_auto_columns ( ) ) ;
computed_values . set_grid_auto_rows ( computed_style . grid_auto_rows ( ) ) ;
2022-08-23 19:49:07 +02:00
computed_values . set_grid_template_columns ( computed_style . grid_template_columns ( ) ) ;
computed_values . set_grid_template_rows ( computed_style . grid_template_rows ( ) ) ;
2022-08-23 19:58:00 +02:00
computed_values . set_grid_column_end ( computed_style . grid_column_end ( ) ) ;
computed_values . set_grid_column_start ( computed_style . grid_column_start ( ) ) ;
computed_values . set_grid_row_end ( computed_style . grid_row_end ( ) ) ;
computed_values . set_grid_row_start ( computed_style . grid_row_start ( ) ) ;
2023-01-16 18:17:05 +01:00
computed_values . set_grid_template_areas ( computed_style . grid_template_areas ( ) ) ;
2023-08-17 20:25:18 +02:00
computed_values . set_grid_auto_flow ( computed_style . grid_auto_flow ( ) ) ;
2022-02-24 16:52:58 +00:00
2023-04-23 01:31:17 +01:00
auto fill = computed_style . property ( CSS : : PropertyID : : Fill ) ;
if ( fill - > has_color ( ) )
2022-04-14 11:52:35 +01:00
computed_values . set_fill ( fill - > to_color ( * this ) ) ;
2023-04-23 01:31:17 +01:00
else if ( fill - > is_url ( ) )
computed_values . set_fill ( fill - > as_url ( ) . url ( ) ) ;
2023-06-06 20:40:10 +01:00
auto stroke = computed_style . property ( CSS : : PropertyID : : Stroke ) ;
if ( stroke - > has_color ( ) )
2022-04-14 11:52:35 +01:00
computed_values . set_stroke ( stroke - > to_color ( * this ) ) ;
2023-06-06 20:40:10 +01:00
else if ( stroke - > is_url ( ) )
computed_values . set_stroke ( stroke - > as_url ( ) . url ( ) ) ;
2023-04-19 18:51:00 +01:00
if ( auto stop_color = computed_style . property ( CSS : : PropertyID : : StopColor ) ; stop_color - > has_color ( ) )
computed_values . set_stop_color ( stop_color - > to_color ( * this ) ) ;
2022-04-14 11:52:35 +01:00
auto stroke_width = computed_style . property ( CSS : : PropertyID : : StrokeWidth ) ;
// FIXME: Converting to pixels isn't really correct - values should be in "user units"
// https://svgwg.org/svg2-draft/coords.html#TermUserUnits
2023-06-01 16:16:15 +01:00
if ( stroke_width - > is_number ( ) )
2023-08-26 15:57:31 +01:00
computed_values . set_stroke_width ( CSS : : Length : : make_px ( CSSPixels : : nearest_value_for ( stroke_width - > as_number ( ) . number ( ) ) ) ) ;
2023-05-21 09:50:56 +02:00
else if ( stroke_width - > is_length ( ) )
2023-05-27 12:28:25 +01:00
computed_values . set_stroke_width ( stroke_width - > as_length ( ) . length ( ) ) ;
2023-05-21 09:50:56 +02:00
else if ( stroke_width - > is_percentage ( ) )
computed_values . set_stroke_width ( CSS : : LengthPercentage { stroke_width - > as_percentage ( ) . percentage ( ) } ) ;
2022-11-06 12:42:22 +01:00
2023-06-11 16:43:46 +01:00
if ( auto fill_rule = computed_style . fill_rule ( ) ; fill_rule . has_value ( ) )
computed_values . set_fill_rule ( * fill_rule ) ;
2023-05-19 20:35:39 +01:00
computed_values . set_fill_opacity ( computed_style . fill_opacity ( ) ) ;
computed_values . set_stroke_opacity ( computed_style . stroke_opacity ( ) ) ;
computed_values . set_stop_opacity ( computed_style . stop_opacity ( ) ) ;
2023-07-19 19:12:00 +01:00
if ( auto text_anchor = computed_style . text_anchor ( ) ; text_anchor . has_value ( ) )
computed_values . set_text_anchor ( * text_anchor ) ;
2023-09-06 18:30:57 +02:00
if ( auto column_count = computed_style . property ( CSS : : PropertyID : : ColumnCount ) ; column_count - > is_integer ( ) )
computed_values . set_column_count ( CSS : : ColumnCount : : make_integer ( column_count - > as_integer ( ) . integer ( ) ) ) ;
2022-11-06 12:42:22 +01:00
computed_values . set_column_gap ( computed_style . size_value ( CSS : : PropertyID : : ColumnGap ) ) ;
computed_values . set_row_gap ( computed_style . size_value ( CSS : : PropertyID : : RowGap ) ) ;
2023-01-02 23:01:29 +01:00
if ( auto border_collapse = computed_style . border_collapse ( ) ; border_collapse . has_value ( ) )
computed_values . set_border_collapse ( border_collapse . value ( ) ) ;
2023-06-08 15:25:16 +01:00
2023-08-07 01:32:52 +00:00
if ( auto table_layout = computed_style . table_layout ( ) ; table_layout . has_value ( ) )
computed_values . set_table_layout ( table_layout . value ( ) ) ;
2023-06-08 15:25:16 +01:00
auto aspect_ratio = computed_style . property ( CSS : : PropertyID : : AspectRatio ) ;
if ( aspect_ratio - > is_value_list ( ) ) {
auto & values_list = aspect_ratio - > as_value_list ( ) . values ( ) ;
if ( values_list . size ( ) = = 2
& & values_list [ 0 ] - > is_identifier ( ) & & values_list [ 0 ] - > as_identifier ( ) . id ( ) = = CSS : : ValueID : : Auto
& & values_list [ 1 ] - > is_ratio ( ) ) {
computed_values . set_aspect_ratio ( { true , values_list [ 1 ] - > as_ratio ( ) . ratio ( ) } ) ;
}
} else if ( aspect_ratio - > is_identifier ( ) & & aspect_ratio - > as_identifier ( ) . id ( ) = = CSS : : ValueID : : Auto ) {
computed_values . set_aspect_ratio ( { true , { } } ) ;
} else if ( aspect_ratio - > is_ratio ( ) ) {
computed_values . set_aspect_ratio ( { false , aspect_ratio - > as_ratio ( ) . ratio ( ) } ) ;
}
2023-06-22 12:22:29 +00:00
if ( display ( ) . is_table_inside ( ) & & is < TableWrapper > ( parent ( ) ) ) {
auto & wrapper_computed_values = static_cast < TableWrapper * > ( parent ( ) ) - > m_computed_values ;
transfer_table_box_computed_values_to_wrapper_computed_values ( wrapper_computed_values ) ;
}
2023-07-03 12:49:13 +02:00
2023-09-05 20:23:15 +01:00
auto math_shift_value = computed_style . property ( CSS : : PropertyID : : MathShift ) ;
if ( auto math_shift = value_id_to_math_shift ( math_shift_value - > to_identifier ( ) ) ; math_shift . has_value ( ) )
computed_values . set_math_shift ( math_shift . value ( ) ) ;
auto math_style_value = computed_style . property ( CSS : : PropertyID : : MathStyle ) ;
if ( auto math_style = value_id_to_math_style ( math_style_value - > to_identifier ( ) ) ; math_style . has_value ( ) )
computed_values . set_math_style ( math_style . value ( ) ) ;
2023-09-07 15:29:54 +01:00
computed_values . set_math_depth ( computed_style . math_depth ( ) ) ;
2023-07-03 12:49:13 +02:00
// Update any anonymous children that inherit from this node.
// FIXME: This is pretty hackish. It would be nicer if they shared the inherited style
// data structure somehow, so this wasn't necessary.
for_each_child ( [ & ] ( auto & child ) {
if ( child . is_anonymous ( ) ) {
auto & child_computed_values = static_cast < CSS : : MutableComputedValues & > ( static_cast < CSS : : ComputedValues & > ( const_cast < CSS : : ImmutableComputedValues & > ( child . computed_values ( ) ) ) ) ;
child_computed_values . inherit_from ( computed_values ) ;
}
} ) ;
2020-06-05 16:54:28 +02:00
}
2020-12-05 20:10:02 +01:00
bool Node : : is_root_element ( ) const
{
if ( is_anonymous ( ) )
return false ;
return is < HTML : : HTMLHtmlElement > ( * dom_node ( ) ) ;
}
2022-12-04 18:02:33 +00:00
DeprecatedString Node : : debug_description ( ) const
2022-02-19 16:39:32 +01:00
{
StringBuilder builder ;
2022-10-17 14:41:50 +02:00
builder . append ( class_name ( ) ) ;
2022-02-19 16:39:32 +01:00
if ( dom_node ( ) ) {
builder . appendff ( " <{}> " , dom_node ( ) - > node_name ( ) ) ;
if ( dom_node ( ) - > is_element ( ) ) {
auto & element = static_cast < DOM : : Element const & > ( * dom_node ( ) ) ;
if ( auto id = element . get_attribute ( HTML : : AttributeNames : : id ) ; ! id . is_null ( ) )
builder . appendff ( " #{} " , id ) ;
for ( auto const & class_name : element . class_names ( ) )
builder . appendff ( " .{} " , class_name ) ;
}
} else {
2022-07-11 17:32:29 +00:00
builder . append ( " (anonymous) " sv ) ;
2022-02-19 16:39:32 +01:00
}
2022-12-06 01:12:49 +00:00
return builder . to_deprecated_string ( ) ;
2022-02-19 16:39:32 +01:00
}
2022-10-06 15:33:09 +02:00
CSS : : Display Node : : display ( ) const
{
if ( ! has_style ( ) ) {
// NOTE: No style means this is dumb text content.
2023-09-04 17:39:15 +01:00
return CSS : : Display ( CSS : : DisplayOutside : : Inline , CSS : : DisplayInside : : Flow ) ;
2022-10-06 15:33:09 +02:00
}
return computed_values ( ) . display ( ) ;
}
2022-10-06 14:39:11 +02:00
bool Node : : is_inline ( ) const
{
2022-10-06 16:02:53 +02:00
return display ( ) . is_inline_outside ( ) ;
2022-10-06 14:39:11 +02:00
}
2021-01-01 18:55:47 +01:00
bool Node : : is_inline_block ( ) const
{
2022-10-06 16:02:53 +02:00
auto display = this - > display ( ) ;
return display . is_inline_outside ( ) & & display . is_flow_root_inside ( ) ;
2021-01-01 18:55:47 +01:00
}
2021-01-06 14:10:53 +01:00
2023-01-16 13:51:49 +01:00
bool Node : : is_inline_table ( ) const
{
auto display = this - > display ( ) ;
return display . is_inline_outside ( ) & & display . is_table_inside ( ) ;
}
2022-10-17 14:41:50 +02:00
JS : : NonnullGCPtr < NodeWithStyle > NodeWithStyle : : create_anonymous_wrapper ( ) const
2021-01-06 14:10:53 +01:00
{
2022-10-17 14:41:50 +02:00
auto wrapper = heap ( ) . allocate_without_realm < BlockContainer > ( const_cast < DOM : : Document & > ( document ( ) ) , nullptr , m_computed_values . clone_inherited_values ( ) ) ;
2023-09-04 17:39:15 +01:00
static_cast < CSS : : MutableComputedValues & > ( wrapper - > m_computed_values ) . set_display ( CSS : : Display ( CSS : : DisplayOutside : : Block , CSS : : DisplayInside : : Flow ) ) ;
2021-01-06 14:10:53 +01:00
wrapper - > m_font = m_font ;
wrapper - > m_line_height = m_line_height ;
2022-10-17 14:41:50 +02:00
return * wrapper ;
2021-01-06 14:10:53 +01:00
}
2023-01-09 08:27:39 +03:00
void NodeWithStyle : : reset_table_box_computed_values_used_by_wrapper_to_init_values ( )
{
2023-05-29 14:11:19 +03:00
VERIFY ( this - > display ( ) . is_table_inside ( ) ) ;
2023-01-09 08:27:39 +03:00
CSS : : MutableComputedValues & mutable_computed_values = static_cast < CSS : : MutableComputedValues & > ( m_computed_values ) ;
2023-06-22 18:11:28 +00:00
mutable_computed_values . set_position ( CSS : : InitialValues : : position ( ) ) ;
mutable_computed_values . set_float ( CSS : : InitialValues : : float_ ( ) ) ;
mutable_computed_values . set_clear ( CSS : : InitialValues : : clear ( ) ) ;
mutable_computed_values . set_inset ( CSS : : InitialValues : : inset ( ) ) ;
mutable_computed_values . set_margin ( CSS : : InitialValues : : margin ( ) ) ;
2023-01-09 08:27:39 +03:00
}
2023-06-22 12:22:29 +00:00
void NodeWithStyle : : transfer_table_box_computed_values_to_wrapper_computed_values ( CSS : : ComputedValues & wrapper_computed_values )
{
// The computed values of properties 'position', 'float', 'margin-*', 'top', 'right', 'bottom', and 'left' on the table element are used on the table wrapper box and not the table box;
// all other values of non-inheritable properties are used on the table box and not the table wrapper box.
// (Where the table element's values are not used on the table and table wrapper boxes, the initial values are used instead.)
auto & mutable_wrapper_computed_values = static_cast < CSS : : MutableComputedValues & > ( wrapper_computed_values ) ;
if ( display ( ) . is_inline_outside ( ) )
mutable_wrapper_computed_values . set_display ( CSS : : Display : : from_short ( CSS : : Display : : Short : : InlineBlock ) ) ;
else
mutable_wrapper_computed_values . set_display ( CSS : : Display : : from_short ( CSS : : Display : : Short : : FlowRoot ) ) ;
mutable_wrapper_computed_values . set_position ( computed_values ( ) . position ( ) ) ;
mutable_wrapper_computed_values . set_inset ( computed_values ( ) . inset ( ) ) ;
mutable_wrapper_computed_values . set_float ( computed_values ( ) . float_ ( ) ) ;
mutable_wrapper_computed_values . set_clear ( computed_values ( ) . clear ( ) ) ;
mutable_wrapper_computed_values . set_margin ( computed_values ( ) . margin ( ) ) ;
reset_table_box_computed_values_used_by_wrapper_to_init_values ( ) ;
}
2023-01-11 12:51:49 +01:00
void Node : : set_paintable ( JS : : GCPtr < Painting : : Paintable > paintable )
2022-03-10 15:50:57 +01:00
{
m_paintable = move ( paintable ) ;
}
2023-01-11 12:51:49 +01:00
JS : : GCPtr < Painting : : Paintable > Node : : create_paintable ( ) const
2022-03-10 15:50:57 +01:00
{
return nullptr ;
}
2022-08-28 13:42:07 +02:00
bool Node : : is_anonymous ( ) const
{
2022-10-16 16:42:39 +02:00
return m_anonymous ;
2022-08-28 13:42:07 +02:00
}
DOM : : Node const * Node : : dom_node ( ) const
{
2022-10-16 16:42:39 +02:00
if ( m_anonymous )
return nullptr ;
2022-08-28 13:42:07 +02:00
return m_dom_node . ptr ( ) ;
}
DOM : : Node * Node : : dom_node ( )
{
2022-10-16 16:42:39 +02:00
if ( m_anonymous )
return nullptr ;
2022-08-28 13:42:07 +02:00
return m_dom_node . ptr ( ) ;
}
2023-08-06 01:29:55 +02:00
DOM : : Element const * Node : : pseudo_element_generator ( ) const
{
VERIFY ( m_generated_for ! = GeneratedFor : : NotGenerated ) ;
return m_pseudo_element_generator . ptr ( ) ;
}
DOM : : Element * Node : : pseudo_element_generator ( )
{
VERIFY ( m_generated_for ! = GeneratedFor : : NotGenerated ) ;
return m_pseudo_element_generator . ptr ( ) ;
}
2022-08-28 13:42:07 +02:00
DOM : : Document & Node : : document ( )
{
2022-10-16 16:42:39 +02:00
return m_dom_node - > document ( ) ;
2022-08-28 13:42:07 +02:00
}
DOM : : Document const & Node : : document ( ) const
{
2022-10-16 16:42:39 +02:00
return m_dom_node - > document ( ) ;
2022-08-28 13:42:07 +02:00
}
2020-03-07 10:27:02 +01:00
}