/* * Copyright (c) 2022, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include namespace Web::Layout { enum class SizeConstraint { None, MinContent, MaxContent, }; class AvailableSize; class AvailableSpace; struct LayoutState { LayoutState() : m_root(*this) { } explicit LayoutState(LayoutState const* parent) : m_parent(parent) , m_root(find_root()) { used_values_per_layout_node.resize(m_root.used_values_per_layout_node.size()); } LayoutState const& find_root() const { LayoutState const* root = this; for (auto* state = m_parent; state; state = state->m_parent) root = state; return *root; } struct UsedValues { NodeWithStyleAndBoxModelMetrics const& node() const { return *m_node; } void set_node(NodeWithStyleAndBoxModelMetrics&, UsedValues const* containing_block_used_values); CSSPixels content_width() const { return m_content_width; } CSSPixels content_height() const { return m_content_height; } void set_content_width(CSSPixels); void set_content_height(CSSPixels); // NOTE: These are used by FlexFormattingContext to assign a temporary main size to items // early on, so that descendants have something to resolve percentages against. void set_temporary_content_width(CSSPixels); void set_temporary_content_height(CSSPixels); bool has_definite_width() const { return m_has_definite_width && width_constraint == SizeConstraint::None; } bool has_definite_height() const { return m_has_definite_height && height_constraint == SizeConstraint::None; } // Returns the available space for content inside this layout box. // If the space in an axis is indefinite, and the outer space is an intrinsic sizing constraint, // the constraint is used in that axis instead. AvailableSpace available_inner_space_or_constraints_from(AvailableSpace const& outer_space) const; void set_content_offset(CSSPixelPoint); void set_content_x(CSSPixels); void set_content_y(CSSPixels); CSSPixelPoint offset; SizeConstraint width_constraint { SizeConstraint::None }; SizeConstraint height_constraint { SizeConstraint::None }; CSSPixels margin_left { 0 }; CSSPixels margin_right { 0 }; CSSPixels margin_top { 0 }; CSSPixels margin_bottom { 0 }; CSSPixels border_left { 0 }; CSSPixels border_right { 0 }; CSSPixels border_top { 0 }; CSSPixels border_bottom { 0 }; CSSPixels padding_left { 0 }; CSSPixels padding_right { 0 }; CSSPixels padding_top { 0 }; CSSPixels padding_bottom { 0 }; CSSPixels inset_left { 0 }; CSSPixels inset_right { 0 }; CSSPixels inset_top { 0 }; CSSPixels inset_bottom { 0 }; Vector line_boxes; CSSPixels margin_box_left() const { return margin_left + border_left + padding_left; } CSSPixels margin_box_right() const { return margin_right + border_right + padding_right; } CSSPixels margin_box_top() const { return margin_top + border_top + padding_top; } CSSPixels margin_box_bottom() const { return margin_bottom + border_bottom + padding_bottom; } CSSPixels margin_box_width() const { return margin_box_left() + content_width() + margin_box_right(); } CSSPixels margin_box_height() const { return margin_box_top() + content_height() + margin_box_bottom(); } CSSPixels border_box_left() const { return border_left + padding_left; } CSSPixels border_box_right() const { return border_right + padding_right; } CSSPixels border_box_top() const { return border_top + padding_top; } CSSPixels border_box_bottom() const { return border_bottom + padding_bottom; } CSSPixels border_box_width() const { return border_box_left() + content_width() + border_box_right(); } CSSPixels border_box_height() const { return border_box_top() + content_height() + border_box_bottom(); } Optional overflow_data; Painting::PaintableBox::OverflowData& ensure_overflow_data() { if (!overflow_data.has_value()) overflow_data = Painting::PaintableBox::OverflowData {}; return *overflow_data; } Optional containing_line_box_fragment; void add_floating_descendant(Box const& box) { m_floating_descendants.set(&box); } auto const& floating_descendants() const { return m_floating_descendants; } private: AvailableSize available_width_inside() const; AvailableSize available_height_inside() const; Layout::NodeWithStyleAndBoxModelMetrics* m_node { nullptr }; CSSPixels m_content_width { 0 }; CSSPixels m_content_height { 0 }; bool m_has_definite_width { false }; bool m_has_definite_height { false }; HashTable m_floating_descendants; }; CSSPixels resolved_definite_width(Box const&) const; CSSPixels resolved_definite_height(Box const&) const; void commit(); // NOTE: get_mutable() will CoW the UsedValues if it's inherited from an ancestor state; UsedValues& get_mutable(NodeWithStyleAndBoxModelMetrics const&); // NOTE: get() will not CoW the UsedValues. UsedValues const& get(NodeWithStyleAndBoxModelMetrics const&) const; Vector> used_values_per_layout_node; // We cache intrinsic sizes once determined, as they will not change over the course of a full layout. // This avoids computing them several times while performing flex layout. struct IntrinsicSizes { Optional min_content_width; Optional max_content_width; // NOTE: Since intrinsic heights depend on the amount of available width, we have to cache // three separate kinds of results, depending on the available width at the time of calculation. HashMap> min_content_height_with_definite_available_width; HashMap> max_content_height_with_definite_available_width; Optional min_content_height_with_min_content_available_width; Optional max_content_height_with_min_content_available_width; Optional min_content_height_with_max_content_available_width; Optional max_content_height_with_max_content_available_width; }; HashMap> mutable intrinsic_sizes; LayoutState const* m_parent { nullptr }; LayoutState const& m_root; }; CSSPixelRect absolute_content_rect(Box const&, LayoutState const&); CSSPixelRect margin_box_rect(Box const&, LayoutState const&); CSSPixelRect margin_box_rect_in_ancestor_coordinate_space(Box const& box, Box const& ancestor_box, LayoutState const&); CSSPixelRect border_box_rect(Box const&, LayoutState const&); CSSPixelRect border_box_rect_in_ancestor_coordinate_space(Box const& box, Box const& ancestor_box, LayoutState const&); CSSPixelRect content_box_rect(Box const&, LayoutState const&); CSSPixelRect content_box_rect_in_ancestor_coordinate_space(Box const& box, Box const& ancestor_box, LayoutState const&); CSSPixels box_baseline(LayoutState const& state, Box const& box); }