ladybird/Libraries/LibWeb/Layout/Box.cpp
Andreas Kling 9340d2d1a3 LibWeb: Make layout nodes refcounted
Move the layout tree from GC allocation to refcounted ownership so
removed layout and paint subtrees are destroyed synchronously instead
of waiting for the next GC sweep. This dramatically reduces GC memory
usage peaks after layout tree churn and makes it easier for memory use
to fall back after large document updates.

Update layout factories, tree traversal, SVG layout node creation,
paintable back-pointers, and pseudo-element layout links to use RefPtr
ownership.

Make display: contents follow the same shape as Blink and WebKit: the
element itself does not create a layout node, and its children are
flattened into the nearest layout parent. Wrap direct non-whitespace
text in an anonymous inline node when the boxless element contributes
inherited style to that text.

Use an internal inline wrapper for display: contents pseudo-elements
so generated content can still participate in layout, painting, hit
testing, and pseudo-element queries. Keep CSSOM reporting the computed
display value from the pseudo style, not the internal wrapper.

Remove the retained out-of-tree layout node list and its testing hook,
since the flattened model does not need a side owner for boxless
elements. Add coverage for inherited text style, dynamic insertion
order, pseudo-element hit testing, and computed style queries.
2026-06-07 20:52:49 +02:00

87 lines
2.5 KiB
C++

/*
* Copyright (c) 2018-2020, Andreas Kling <andreas@ladybird.org>
* Copyright (c) 2021-2022, Sam Atkins <atkinssj@serenityos.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <LibWeb/DOM/Document.h>
#include <LibWeb/HTML/HTMLHtmlElement.h>
#include <LibWeb/Layout/BlockContainer.h>
#include <LibWeb/Layout/Box.h>
#include <LibWeb/Layout/FormattingContext.h>
#include <LibWeb/Layout/TableWrapper.h>
#include <LibWeb/Painting/PaintableBox.h>
namespace Web::Layout {
Box::Box(DOM::Document& document, DOM::Node* node, CSS::ComputedProperties const& style)
: NodeWithStyleAndBoxModelMetrics(document, node, style)
{
}
Box::Box(DOM::Document& document, DOM::Node* node, NonnullOwnPtr<CSS::ComputedValues> computed_values)
: NodeWithStyleAndBoxModelMetrics(document, node, move(computed_values))
{
}
Box::~Box()
{
}
CSS::SizeWithAspectRatio Box::auto_content_box_size() const
{
// https://drafts.csswg.org/css-contain-2/#containment-size
// Replaced elements must be treated as having a natural width and height of 0 and no natural aspect
// ratio.
if (has_size_containment())
return { 0, 0, {} };
return compute_auto_content_box_size();
}
RefPtr<Painting::Paintable> Box::create_paintable() const
{
return Painting::PaintableBox::create(*this);
}
RefPtr<Painting::PaintableBox> Box::paintable_box()
{
if (auto paintable = Node::first_paintable())
return static_cast<Painting::PaintableBox&>(*paintable);
return nullptr;
}
RefPtr<Painting::PaintableBox const> Box::paintable_box() const
{
if (auto paintable = Node::first_paintable())
return static_cast<Painting::PaintableBox const&>(*paintable);
return nullptr;
}
Optional<CSSPixelFraction> Box::preferred_aspect_ratio() const
{
auto const& computed_aspect_ratio = computed_values().aspect_ratio();
// https://www.w3.org/TR/css-contain-2/#containment-size
if (!has_size_containment() && computed_aspect_ratio.use_natural_aspect_ratio_if_available) {
if (auto auto_size = auto_content_box_size(); auto_size.has_aspect_ratio())
return auto_size.aspect_ratio;
}
if (!computed_aspect_ratio.preferred_ratio.has_value())
return {};
auto ratio = computed_aspect_ratio.preferred_ratio.value();
if (ratio.is_degenerate())
return {};
auto fraction = CSSPixelFraction(ratio.numerator(), ratio.denominator());
// ratio.is_degenerate() operates on doubles while CSSPixelFraction uses CSSPixels, so we need to check again here.
if (fraction == 0)
return {};
return fraction;
}
}