mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-10-21 07:29:53 +00:00
Some checks are pending
CI / macOS, arm64, Sanitizer, Clang (push) Waiting to run
CI / Linux, x86_64, Fuzzers, Clang (push) Waiting to run
CI / Linux, x86_64, Sanitizer, GNU (push) Waiting to run
Package the js repl as a binary artifact / macOS, arm64 (push) Waiting to run
Lint Code / lint (push) Waiting to run
Label PRs with merge conflicts / auto-labeler (push) Waiting to run
CI / Linux, x86_64, Sanitizer, Clang (push) Waiting to run
Package the js repl as a binary artifact / Linux, arm64 (push) Waiting to run
Package the js repl as a binary artifact / Linux, x86_64 (push) Waiting to run
Run test262 and test-wasm / run_and_update_results (push) Waiting to run
Push notes / build (push) Waiting to run
All fragments inside an atomic inline box should stay within that box, otherwise we'll screw up the paint order and paint them behind things that they're supposed to be on top of. This fixes an issue with inline-block content not appearing on sites like Google Docs and Reddit, among others.
363 lines
13 KiB
C++
363 lines
13 KiB
C++
/*
|
|
* Copyright (c) 2018-2025, Andreas Kling <andreas@ladybird.org>
|
|
* Copyright (c) 2025, Jelle Raaijmakers <jelle@ladybird.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/NonnullRefPtr.h>
|
|
#include <AK/Vector.h>
|
|
#include <LibJS/Heap/Cell.h>
|
|
#include <LibWeb/CSS/StyleValues/ImageStyleValue.h>
|
|
#include <LibWeb/DOM/Document.h>
|
|
#include <LibWeb/Export.h>
|
|
#include <LibWeb/Forward.h>
|
|
#include <LibWeb/Painting/DisplayListRecordingContext.h>
|
|
#include <LibWeb/Painting/Paintable.h>
|
|
#include <LibWeb/TreeNode.h>
|
|
|
|
namespace Web::Layout {
|
|
|
|
enum class LayoutMode {
|
|
// Normal layout. No min-content or max-content constraints applied.
|
|
Normal,
|
|
|
|
// Intrinsic size determination.
|
|
// Boxes honor min-content and max-content constraints (set via LayoutState::UsedValues::{width,height}_constraint)
|
|
// by considering their containing block to be 0-sized or infinitely large in the relevant axis.
|
|
// https://drafts.csswg.org/css-sizing-3/#intrinsic-sizing
|
|
IntrinsicSizing,
|
|
};
|
|
|
|
class WEB_API Node
|
|
: public JS::Cell
|
|
, public TreeNode<Node> {
|
|
GC_CELL(Node, JS::Cell);
|
|
|
|
public:
|
|
virtual ~Node();
|
|
|
|
bool is_anonymous() const;
|
|
DOM::Node const* dom_node() const;
|
|
DOM::Node* dom_node();
|
|
|
|
DOM::Element const* pseudo_element_generator() const;
|
|
DOM::Element* pseudo_element_generator();
|
|
|
|
bool needs_layout_update() const { return m_needs_layout_update; }
|
|
void set_needs_layout_update(DOM::SetNeedsLayoutReason);
|
|
void reset_needs_layout_update() { m_needs_layout_update = false; }
|
|
|
|
bool is_generated() const { return m_generated_for.has_value(); }
|
|
Optional<CSS::PseudoElement> generated_for_pseudo_element() const { return m_generated_for; }
|
|
bool is_generated_for_before_pseudo_element() const { return m_generated_for == CSS::PseudoElement::Before; }
|
|
bool is_generated_for_after_pseudo_element() const { return m_generated_for == CSS::PseudoElement::After; }
|
|
bool is_generated_for_backdrop_pseudo_element() const { return m_generated_for == CSS::PseudoElement::Backdrop; }
|
|
void set_generated_for(CSS::PseudoElement type, DOM::Element& element)
|
|
{
|
|
m_generated_for = type;
|
|
m_pseudo_element_generator = &element;
|
|
}
|
|
|
|
using PaintableList = IntrusiveList<&Painting::Paintable::m_list_node>;
|
|
|
|
Painting::Paintable* first_paintable() { return m_paintable.first(); }
|
|
Painting::Paintable const* first_paintable() const { return m_paintable.first(); }
|
|
PaintableList& paintables() { return m_paintable; }
|
|
PaintableList const& paintables() const { return m_paintable; }
|
|
void add_paintable(GC::Ptr<Painting::Paintable>);
|
|
void clear_paintables();
|
|
|
|
virtual GC::Ptr<Painting::Paintable> create_paintable() const;
|
|
|
|
DOM::Document& document();
|
|
DOM::Document const& document() const;
|
|
|
|
GC::Ptr<HTML::Navigable> navigable() const;
|
|
|
|
Viewport const& root() const;
|
|
Viewport& root();
|
|
|
|
bool is_root_element() const;
|
|
|
|
String debug_description() const;
|
|
|
|
bool has_style() const { return m_has_style; }
|
|
bool has_style_or_parent_with_style() const;
|
|
|
|
virtual bool can_have_children() const { return true; }
|
|
|
|
CSS::Display display() const;
|
|
|
|
bool is_inline() const;
|
|
bool is_inline_block() const;
|
|
bool is_inline_table() const;
|
|
|
|
bool is_atomic_inline() const;
|
|
|
|
bool is_out_of_flow(FormattingContext const&) const;
|
|
|
|
// These are used to optimize hot is<T> variants for some classes where dynamic_cast is too slow.
|
|
virtual bool is_box() const { return false; }
|
|
virtual bool is_block_container() const { return false; }
|
|
virtual bool is_break_node() const { return false; }
|
|
virtual bool is_text_node() const { return false; }
|
|
virtual bool is_viewport() const { return false; }
|
|
virtual bool is_svg_box() const { return false; }
|
|
virtual bool is_svg_geometry_box() const { return false; }
|
|
virtual bool is_svg_mask_box() const { return false; }
|
|
virtual bool is_svg_svg_box() const { return false; }
|
|
virtual bool is_svg_graphics_box() const { return false; }
|
|
virtual bool is_svg_foreign_object_box() const { return false; }
|
|
virtual bool is_label() const { return false; }
|
|
virtual bool is_replaced_box() const { return false; }
|
|
virtual bool is_list_item_box() const { return false; }
|
|
virtual bool is_list_item_marker_box() const { return false; }
|
|
virtual bool is_fieldset_box() const { return false; }
|
|
virtual bool is_legend_box() const { return false; }
|
|
virtual bool is_table_wrapper() const { return false; }
|
|
virtual bool is_node_with_style() const { return false; }
|
|
virtual bool is_node_with_style_and_box_model_metrics() const { return false; }
|
|
|
|
template<typename T>
|
|
bool fast_is() const = delete;
|
|
|
|
bool is_floating() const;
|
|
bool is_positioned() const;
|
|
bool is_absolutely_positioned() const;
|
|
bool is_fixed_position() const;
|
|
bool is_sticky_position() const;
|
|
|
|
bool is_flex_item() const { return m_is_flex_item; }
|
|
void set_flex_item(bool b) { m_is_flex_item = b; }
|
|
|
|
bool is_grid_item() const { return m_is_grid_item; }
|
|
void set_grid_item(bool b) { m_is_grid_item = b; }
|
|
|
|
[[nodiscard]] GC::Ptr<Box const> containing_block() const { return m_containing_block; }
|
|
[[nodiscard]] GC::Ptr<Box> containing_block() { return m_containing_block; }
|
|
|
|
void recompute_containing_block(Badge<DOM::Document>);
|
|
|
|
[[nodiscard]] Box const* static_position_containing_block() const;
|
|
[[nodiscard]] Box* static_position_containing_block() { return const_cast<Box*>(const_cast<Node const*>(this)->static_position_containing_block()); }
|
|
|
|
// Closest non-anonymous ancestor box, to be used when resolving percentage values.
|
|
// Anonymous block boxes are ignored when resolving percentage values that would refer to it:
|
|
// the closest non-anonymous ancestor box is used instead.
|
|
// https://www.w3.org/TR/CSS22/visuren.html#anonymous-block-level
|
|
Box const* non_anonymous_containing_block() const;
|
|
|
|
bool establishes_stacking_context() const;
|
|
|
|
bool can_contain_boxes_with_position_absolute() const;
|
|
|
|
Gfx::Font const& first_available_font() const;
|
|
Gfx::Font const& font(DisplayListRecordingContext&) const;
|
|
Gfx::Font const& font(float scale_factor) const;
|
|
|
|
CSS::ImmutableComputedValues const& computed_values() const;
|
|
|
|
NodeWithStyle* parent();
|
|
NodeWithStyle const* parent() const;
|
|
|
|
void inserted_into(Node&) { }
|
|
void removed_from(Node&) { }
|
|
void children_changed() { }
|
|
|
|
bool children_are_inline() const { return m_children_are_inline; }
|
|
void set_children_are_inline(bool value) { m_children_are_inline = value; }
|
|
|
|
u32 initial_quote_nesting_level() const { return m_initial_quote_nesting_level; }
|
|
void set_initial_quote_nesting_level(u32 value) { m_initial_quote_nesting_level = value; }
|
|
|
|
// An element is called out of flow if it is floated, absolutely positioned, or is the root element.
|
|
// https://www.w3.org/TR/CSS22/visuren.html#positioning-scheme
|
|
bool is_out_of_flow() const { return is_floating() || is_absolutely_positioned(); }
|
|
|
|
// An element is called in-flow if it is not out-of-flow.
|
|
// https://www.w3.org/TR/CSS22/visuren.html#positioning-scheme
|
|
bool is_in_flow() const { return !is_out_of_flow(); }
|
|
|
|
[[nodiscard]] bool has_css_transform() const
|
|
{
|
|
auto const& computed_values = this->computed_values();
|
|
return !computed_values.transformations().is_empty()
|
|
|| computed_values.rotate().has_value()
|
|
|| computed_values.translate().has_value()
|
|
|| computed_values.scale().has_value();
|
|
}
|
|
|
|
// https://drafts.csswg.org/css-ui/#propdef-user-select
|
|
CSS::UserSelect user_select_used_value() const;
|
|
|
|
// https://drafts.csswg.org/css-contain-2/#containment-types
|
|
bool has_size_containment() const;
|
|
bool has_inline_size_containment() const;
|
|
bool has_layout_containment() const;
|
|
bool has_style_containment() const;
|
|
bool has_paint_containment() const;
|
|
|
|
[[nodiscard]] bool has_been_wrapped_in_table_wrapper() const { return m_has_been_wrapped_in_table_wrapper; }
|
|
void set_has_been_wrapped_in_table_wrapper(bool value) { m_has_been_wrapped_in_table_wrapper = value; }
|
|
|
|
protected:
|
|
Node(DOM::Document&, DOM::Node*);
|
|
|
|
virtual void visit_edges(Cell::Visitor&) override;
|
|
|
|
private:
|
|
friend class NodeWithStyle;
|
|
|
|
GC::Ref<DOM::Node> m_dom_node;
|
|
PaintableList m_paintable;
|
|
|
|
GC::Ptr<Box> m_containing_block;
|
|
|
|
GC::Ptr<DOM::Element> m_pseudo_element_generator;
|
|
|
|
bool m_anonymous { false };
|
|
bool m_has_style { false };
|
|
bool m_children_are_inline { false };
|
|
|
|
bool m_is_flex_item { false };
|
|
bool m_is_grid_item { false };
|
|
|
|
bool m_has_been_wrapped_in_table_wrapper { false };
|
|
|
|
bool m_needs_layout_update { false };
|
|
|
|
Optional<CSS::PseudoElement> m_generated_for {};
|
|
|
|
u32 m_initial_quote_nesting_level { 0 };
|
|
};
|
|
|
|
class WEB_API NodeWithStyle : public Node {
|
|
GC_CELL(NodeWithStyle, Node);
|
|
|
|
public:
|
|
virtual ~NodeWithStyle() override = default;
|
|
|
|
CSS::ImmutableComputedValues const& computed_values() const { return static_cast<CSS::ImmutableComputedValues const&>(*m_computed_values); }
|
|
CSS::MutableComputedValues& mutable_computed_values() { return static_cast<CSS::MutableComputedValues&>(*m_computed_values); }
|
|
|
|
void apply_style(CSS::ComputedProperties const&);
|
|
|
|
Gfx::Font const& first_available_font() const;
|
|
Vector<CSS::BackgroundLayerData> const& background_layers() const { return computed_values().background_layers(); }
|
|
const CSS::AbstractImageStyleValue* list_style_image() const { return m_list_style_image; }
|
|
|
|
GC::Ref<NodeWithStyle> create_anonymous_wrapper() const;
|
|
|
|
void transfer_table_box_computed_values_to_wrapper_computed_values(CSS::ComputedValues& wrapper_computed_values);
|
|
|
|
bool is_body() const;
|
|
bool is_scroll_container() const;
|
|
|
|
virtual void visit_edges(Cell::Visitor& visitor) override;
|
|
|
|
void set_computed_values(NonnullOwnPtr<CSS::ComputedValues>);
|
|
|
|
protected:
|
|
NodeWithStyle(DOM::Document&, DOM::Node*, GC::Ref<CSS::ComputedProperties>);
|
|
NodeWithStyle(DOM::Document&, DOM::Node*, NonnullOwnPtr<CSS::ComputedValues>);
|
|
|
|
private:
|
|
virtual bool is_node_with_style() const final { return true; }
|
|
|
|
void reset_table_box_computed_values_used_by_wrapper_to_init_values();
|
|
void propagate_style_to_anonymous_wrappers();
|
|
|
|
NonnullOwnPtr<CSS::ComputedValues> m_computed_values;
|
|
RefPtr<CSS::AbstractImageStyleValue const> m_list_style_image;
|
|
};
|
|
|
|
template<>
|
|
inline bool Node::fast_is<NodeWithStyle>() const { return is_node_with_style(); }
|
|
|
|
class NodeWithStyleAndBoxModelMetrics : public NodeWithStyle {
|
|
GC_CELL(NodeWithStyleAndBoxModelMetrics, NodeWithStyle);
|
|
|
|
public:
|
|
GC::Ptr<NodeWithStyleAndBoxModelMetrics> continuation_of_node() const { return m_continuation_of_node; }
|
|
void set_continuation_of_node(Badge<TreeBuilder>, GC::Ptr<NodeWithStyleAndBoxModelMetrics> node) { m_continuation_of_node = node; }
|
|
|
|
bool should_create_inline_continuation() const;
|
|
|
|
void propagate_style_along_continuation(CSS::ComputedProperties const&) const;
|
|
|
|
virtual void visit_edges(Cell::Visitor& visitor) override;
|
|
|
|
protected:
|
|
NodeWithStyleAndBoxModelMetrics(DOM::Document& document, DOM::Node* node, GC::Ref<CSS::ComputedProperties> style)
|
|
: NodeWithStyle(document, node, style)
|
|
{
|
|
}
|
|
|
|
NodeWithStyleAndBoxModelMetrics(DOM::Document& document, DOM::Node* node, NonnullOwnPtr<CSS::ComputedValues> computed_values)
|
|
: NodeWithStyle(document, node, move(computed_values))
|
|
{
|
|
}
|
|
|
|
private:
|
|
virtual bool is_node_with_style_and_box_model_metrics() const final { return true; }
|
|
|
|
GC::Ptr<NodeWithStyleAndBoxModelMetrics> m_continuation_of_node;
|
|
};
|
|
|
|
template<>
|
|
inline bool Node::fast_is<NodeWithStyleAndBoxModelMetrics>() const { return is_node_with_style_and_box_model_metrics(); }
|
|
|
|
inline bool Node::has_style_or_parent_with_style() const
|
|
{
|
|
return m_has_style || (parent() != nullptr && parent()->has_style_or_parent_with_style());
|
|
}
|
|
|
|
inline Gfx::Font const& Node::first_available_font() const
|
|
{
|
|
VERIFY(has_style_or_parent_with_style());
|
|
if (m_has_style)
|
|
return static_cast<NodeWithStyle const*>(this)->first_available_font();
|
|
return parent()->first_available_font();
|
|
}
|
|
|
|
inline Gfx::Font const& Node::font(DisplayListRecordingContext& context) const
|
|
{
|
|
return font(context.device_pixels_per_css_pixel());
|
|
}
|
|
|
|
inline Gfx::Font const& Node::font(float scale_factor) const
|
|
{
|
|
auto const& font = first_available_font();
|
|
return font.with_size(font.point_size() * scale_factor);
|
|
}
|
|
|
|
inline const CSS::ImmutableComputedValues& Node::computed_values() const
|
|
{
|
|
VERIFY(has_style_or_parent_with_style());
|
|
|
|
if (m_has_style)
|
|
return static_cast<NodeWithStyle const*>(this)->computed_values();
|
|
return parent()->computed_values();
|
|
}
|
|
|
|
inline NodeWithStyle const* Node::parent() const
|
|
{
|
|
return static_cast<NodeWithStyle const*>(TreeNode<Node>::parent());
|
|
}
|
|
|
|
inline NodeWithStyle* Node::parent()
|
|
{
|
|
return static_cast<NodeWithStyle*>(TreeNode<Node>::parent());
|
|
}
|
|
|
|
inline Gfx::Font const& NodeWithStyle::first_available_font() const
|
|
{
|
|
// https://drafts.csswg.org/css-fonts/#first-available-font
|
|
// First font for which the character U+0020 (space) is not excluded by a unicode-range
|
|
return computed_values().font_list().font_for_code_point(' ');
|
|
}
|
|
|
|
}
|