mirror of
https://github.com/LadybirdBrowser/ladybird
synced 2026-05-02 04:27:12 +02:00
This unifies the implementations of the element resize and the scroll mouse inputs, so the actual code involved in handling the events can be simplified, and it only should require one object per event. The cursor override is now part of this ChromeWidget class as well, so that the hit test's cursor doesn't need to be passed around as much.
493 lines
23 KiB
C++
493 lines
23 KiB
C++
/*
|
|
* Copyright (c) 2020-2022, Andreas Kling <andreas@ladybird.org>
|
|
* Copyright (c) 2022, Sam Atkins <atkinssj@serenityos.org>
|
|
* Copyright (c) 2024-2026, Aliaksandr Kalenik <kalenik.aliaksandr@gmail.com>
|
|
* Copyright (c) 2025, Jelle Raaijmakers <jelle@ladybird.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/QuickSort.h>
|
|
#include <AK/TemporaryChange.h>
|
|
#include <LibGfx/Rect.h>
|
|
#include <LibWeb/DOM/Document.h>
|
|
#include <LibWeb/Layout/ReplacedBox.h>
|
|
#include <LibWeb/Layout/Viewport.h>
|
|
#include <LibWeb/Page/Page.h>
|
|
#include <LibWeb/Painting/DisplayList.h>
|
|
#include <LibWeb/Painting/DisplayListRecorder.h>
|
|
#include <LibWeb/Painting/PaintableBox.h>
|
|
#include <LibWeb/Painting/PaintableWithLines.h>
|
|
#include <LibWeb/Painting/SVGSVGPaintable.h>
|
|
#include <LibWeb/Painting/StackingContext.h>
|
|
#include <LibWeb/Painting/ViewportPaintable.h>
|
|
#include <LibWeb/SVG/SVGMaskElement.h>
|
|
|
|
namespace Web::Painting {
|
|
|
|
GC_DEFINE_ALLOCATOR(StackingContext);
|
|
|
|
static void paint_node(Paintable const& paintable, DisplayListRecordingContext& context, PaintPhase phase)
|
|
{
|
|
TemporaryChange save_nesting_level(context.display_list_recorder().m_save_nesting_level, 0);
|
|
|
|
auto const* paintable_box = as_if<PaintableBox>(paintable);
|
|
|
|
if (paintable_box) {
|
|
// Text fragments in a PaintableWithLines are content of the block container.
|
|
// They need the descendants' visual context, not the element's own visual context.
|
|
if (is<PaintableWithLines>(paintable) && phase == PaintPhase::Foreground)
|
|
context.display_list_recorder().set_accumulated_visual_context(paintable_box->accumulated_visual_context_for_descendants_index());
|
|
else
|
|
context.display_list_recorder().set_accumulated_visual_context(paintable_box->accumulated_visual_context_index());
|
|
}
|
|
|
|
bool const skip_cache = !paintable_box || context.should_show_line_box_borders();
|
|
if (!skip_cache && paintable_box->has_cached_commands(phase)) {
|
|
context.display_list_recorder().replay_cached_commands(paintable_box->cached_commands(phase));
|
|
} else if (!skip_cache) {
|
|
auto capture = context.display_list_recorder().begin_command_capture();
|
|
paintable.paint(context, phase);
|
|
paintable_box->set_cached_commands(phase, capture.take());
|
|
} else {
|
|
paintable.paint(context, phase);
|
|
}
|
|
|
|
context.display_list_recorder().set_accumulated_visual_context({});
|
|
|
|
VERIFY(context.display_list_recorder().m_save_nesting_level == 0);
|
|
}
|
|
|
|
StackingContext::StackingContext(PaintableBox& paintable, StackingContext* parent, size_t index_in_tree_order)
|
|
: m_paintable(paintable)
|
|
, m_parent(parent)
|
|
, m_index_in_tree_order(index_in_tree_order)
|
|
{
|
|
VERIFY(m_parent != this);
|
|
if (m_parent)
|
|
m_parent->m_children.append(*this);
|
|
}
|
|
|
|
void StackingContext::sort()
|
|
{
|
|
quick_sort(m_children, [](auto& a, auto& b) {
|
|
auto a_z_index = a->paintable_box().computed_values().z_index().value_or(0);
|
|
auto b_z_index = b->paintable_box().computed_values().z_index().value_or(0);
|
|
if (a_z_index == b_z_index)
|
|
return a->m_index_in_tree_order < b->m_index_in_tree_order;
|
|
return a_z_index < b_z_index;
|
|
});
|
|
|
|
for (auto child : m_children)
|
|
child->sort();
|
|
}
|
|
|
|
void StackingContext::visit_edges(Visitor& visitor)
|
|
{
|
|
Base::visit_edges(visitor);
|
|
visitor.visit(m_paintable);
|
|
visitor.visit(m_non_positioned_floating_descendants);
|
|
visitor.visit(m_positioned_descendants_and_stacking_contexts_with_stack_level_0);
|
|
visitor.visit(m_parent);
|
|
visitor.visit(m_children);
|
|
}
|
|
|
|
void StackingContext::set_last_paint_generation_id(u64 generation_id)
|
|
{
|
|
if (m_last_paint_generation_id.has_value() && m_last_paint_generation_id.value() >= generation_id) {
|
|
dbgln("FIXME: Painting commands are recorded twice for stacking context: {}", m_paintable->layout_node().debug_description());
|
|
}
|
|
m_last_paint_generation_id = generation_id;
|
|
}
|
|
|
|
static PaintPhase to_paint_phase(StackingContext::StackingContextPaintPhase phase)
|
|
{
|
|
// There are not a fully correct mapping since some stacking context phases are combined.
|
|
switch (phase) {
|
|
case StackingContext::StackingContextPaintPhase::Floats:
|
|
case StackingContext::StackingContextPaintPhase::BackgroundAndBordersForInlineLevelAndReplaced:
|
|
case StackingContext::StackingContextPaintPhase::BackgroundAndBorders:
|
|
return PaintPhase::Background;
|
|
case StackingContext::StackingContextPaintPhase::Foreground:
|
|
return PaintPhase::Foreground;
|
|
default:
|
|
VERIFY_NOT_REACHED();
|
|
}
|
|
}
|
|
|
|
void StackingContext::paint_node_as_stacking_context(Paintable const& paintable, DisplayListRecordingContext& context)
|
|
{
|
|
if (paintable.is_svg_svg_paintable()) {
|
|
paint_svg(context, static_cast<PaintableBox const&>(paintable), PaintPhase::Foreground);
|
|
return;
|
|
}
|
|
|
|
paint_node(paintable, context, PaintPhase::Background);
|
|
paint_node(paintable, context, PaintPhase::Border);
|
|
paint_descendants(context, paintable, StackingContextPaintPhase::BackgroundAndBorders);
|
|
paint_descendants(context, paintable, StackingContextPaintPhase::Floats);
|
|
paint_descendants(context, paintable, StackingContextPaintPhase::BackgroundAndBordersForInlineLevelAndReplaced);
|
|
paint_node(paintable, context, PaintPhase::Foreground);
|
|
paint_descendants(context, paintable, StackingContextPaintPhase::Foreground);
|
|
paint_node(paintable, context, PaintPhase::Outline);
|
|
paint_node(paintable, context, PaintPhase::Overlay);
|
|
}
|
|
|
|
void StackingContext::paint_svg(DisplayListRecordingContext& context, PaintableBox const& paintable, PaintPhase phase)
|
|
{
|
|
if (phase != PaintPhase::Foreground)
|
|
return;
|
|
|
|
paint_node(paintable, context, PaintPhase::Background);
|
|
paint_node(paintable, context, PaintPhase::Border);
|
|
SVGSVGPaintable::paint_svg_box(context, paintable, phase);
|
|
}
|
|
|
|
void StackingContext::paint_descendants(DisplayListRecordingContext& context, Paintable const& paintable, StackingContextPaintPhase phase)
|
|
{
|
|
paintable.for_each_child([&context, phase](auto& child) {
|
|
if (child.has_stacking_context())
|
|
return IterationDecision::Continue;
|
|
|
|
if (child.is_svg_svg_paintable()) {
|
|
paint_svg(context, static_cast<PaintableBox const&>(child), to_paint_phase(phase));
|
|
return IterationDecision::Continue;
|
|
}
|
|
|
|
// NOTE: Grid specification https://www.w3.org/TR/css-grid-2/#z-order says that grid items should be treated
|
|
// the same way as CSS2 defines for inline-blocks:
|
|
// "For each one of these, treat the element as if it created a new stacking context, but any positioned
|
|
// descendants and descendants which actually create a new stacking context should be considered part of
|
|
// the parent stacking context, not this new one."
|
|
auto const& z_index = [&] { return child.computed_values().z_index(); };
|
|
if (child.layout_node().is_grid_item() && !z_index().has_value()) {
|
|
// FIXME: This may not be fully correct with respect to the paint phases.
|
|
if (phase == StackingContextPaintPhase::Foreground)
|
|
paint_node_as_stacking_context(child, context);
|
|
return IterationDecision::Continue;
|
|
}
|
|
|
|
// https://drafts.csswg.org/css2/#painting-order
|
|
// All non-positioned floating descendants, in tree order. For each one of these, treat the
|
|
// element as if it created a new stacking context, but any positioned descendants and
|
|
// descendants which actually create a new stacking context should be considered part of the
|
|
// parent stacking context, not this new one.
|
|
if (child.is_floating() && !child.is_positioned() && !z_index().has_value()) {
|
|
if (phase == StackingContextPaintPhase::Floats) {
|
|
paint_node_as_stacking_context(child, context);
|
|
}
|
|
return IterationDecision::Continue;
|
|
}
|
|
|
|
if (child.is_positioned() && z_index().value_or(0) == 0)
|
|
return IterationDecision::Continue;
|
|
|
|
bool child_is_inline_or_replaced = child.is_inline() || is<Layout::ReplacedBox>(child.layout_node());
|
|
switch (phase) {
|
|
case StackingContextPaintPhase::BackgroundAndBorders:
|
|
if (!child_is_inline_or_replaced && !child.is_floating()) {
|
|
paint_node(child, context, PaintPhase::Background);
|
|
paint_node(child, context, PaintPhase::Border);
|
|
paint_descendants(context, child, phase);
|
|
paint_node(child, context, PaintPhase::TableCollapsedBorder);
|
|
}
|
|
break;
|
|
case StackingContextPaintPhase::Floats:
|
|
if (child.is_floating()) {
|
|
paint_node(child, context, PaintPhase::Background);
|
|
paint_node(child, context, PaintPhase::Border);
|
|
paint_descendants(context, child, StackingContextPaintPhase::BackgroundAndBorders);
|
|
}
|
|
paint_descendants(context, child, phase);
|
|
break;
|
|
case StackingContextPaintPhase::BackgroundAndBordersForInlineLevelAndReplaced:
|
|
if (child_is_inline_or_replaced) {
|
|
paint_node(child, context, PaintPhase::Background);
|
|
paint_node(child, context, PaintPhase::Border);
|
|
paint_node(child, context, PaintPhase::TableCollapsedBorder);
|
|
paint_descendants(context, child, StackingContextPaintPhase::BackgroundAndBorders);
|
|
}
|
|
paint_descendants(context, child, phase);
|
|
break;
|
|
case StackingContextPaintPhase::Foreground:
|
|
paint_node(child, context, PaintPhase::Foreground);
|
|
paint_descendants(context, child, phase);
|
|
paint_node(child, context, PaintPhase::Outline);
|
|
paint_node(child, context, PaintPhase::Overlay);
|
|
break;
|
|
}
|
|
|
|
return IterationDecision::Continue;
|
|
});
|
|
}
|
|
|
|
void StackingContext::paint_child(DisplayListRecordingContext& context, StackingContext const& child)
|
|
{
|
|
VERIFY(!child.paintable_box().is_svg_paintable());
|
|
const_cast<StackingContext&>(child).set_last_paint_generation_id(context.paint_generation_id());
|
|
child.paint(context);
|
|
}
|
|
|
|
void StackingContext::paint_internal(DisplayListRecordingContext& context) const
|
|
{
|
|
VERIFY(!paintable_box().is_svg_paintable());
|
|
if (paintable_box().is_svg_svg_paintable()) {
|
|
auto const& svg_svg_paintable = static_cast<SVGSVGPaintable const&>(paintable_box());
|
|
paint_node(svg_svg_paintable, context, PaintPhase::Background);
|
|
paint_node(svg_svg_paintable, context, PaintPhase::Border);
|
|
|
|
SVGSVGPaintable::paint_descendants(context, svg_svg_paintable, PaintPhase::Foreground);
|
|
|
|
paint_node(svg_svg_paintable, context, PaintPhase::Outline);
|
|
if (context.should_paint_overlay()) {
|
|
paint_node(svg_svg_paintable, context, PaintPhase::Overlay);
|
|
}
|
|
return;
|
|
}
|
|
|
|
// For a more elaborate description of the algorithm, see CSS 2.1 Appendix E
|
|
// Draw the background and borders for the context root (steps 1, 2)
|
|
paint_node(paintable_box(), context, PaintPhase::Background);
|
|
paint_node(paintable_box(), context, PaintPhase::Border);
|
|
|
|
// Stacking contexts formed by positioned descendants with negative z-indices (excluding 0) in z-index order
|
|
// (most negative first) then tree order. (step 3)
|
|
// Here, we treat non-positioned stacking contexts as if they were positioned, because CSS 2.0 spec does not
|
|
// account for new properties like `transform` and `opacity` that can create stacking contexts.
|
|
// https://github.com/w3c/csswg-drafts/issues/2717
|
|
for (auto child : m_children) {
|
|
if (child->paintable_box().computed_values().z_index().has_value() && child->paintable_box().computed_values().z_index().value() < 0)
|
|
paint_child(context, *child);
|
|
}
|
|
|
|
// Draw the background and borders for block-level children (step 4)
|
|
paint_descendants(context, paintable_box(), StackingContextPaintPhase::BackgroundAndBorders);
|
|
// Draw the non-positioned floats (step 5)
|
|
paint_descendants(context, paintable_box(), StackingContextPaintPhase::Floats);
|
|
// Draw inline content, replaced content, etc. (steps 6, 7)
|
|
paint_descendants(context, paintable_box(), StackingContextPaintPhase::BackgroundAndBordersForInlineLevelAndReplaced);
|
|
paint_node(paintable_box(), context, PaintPhase::Foreground);
|
|
paint_descendants(context, paintable_box(), StackingContextPaintPhase::Foreground);
|
|
|
|
// Draw positioned descendants with z-index `0` or `auto` in tree order. (step 8)
|
|
// Here, we treat non-positioned stacking contexts as if they were positioned, because CSS 2.0 spec does not
|
|
// account for new properties like `transform` and `opacity` that can create stacking contexts.
|
|
// https://github.com/w3c/csswg-drafts/issues/2717
|
|
for (auto const& paintable : m_positioned_descendants_and_stacking_contexts_with_stack_level_0) {
|
|
// At this point, `paintable_box` is a positioned descendant with z-index: auto.
|
|
// FIXME: This is basically duplicating logic found elsewhere in this same function. Find a way to make this more elegant.
|
|
if (auto* child = paintable->stacking_context()) {
|
|
paint_child(context, *child);
|
|
} else {
|
|
paint_node_as_stacking_context(paintable, context);
|
|
}
|
|
};
|
|
|
|
// Stacking contexts formed by positioned descendants with z-indices greater than or equal to 1 in z-index order
|
|
// (smallest first) then tree order. (Step 9)
|
|
// Here, we treat non-positioned stacking contexts as if they were positioned, because CSS 2.0 spec does not
|
|
// account for new properties like `transform` and `opacity` that can create stacking contexts.
|
|
// https://github.com/w3c/csswg-drafts/issues/2717
|
|
for (auto child : m_children) {
|
|
if (child->paintable_box().computed_values().z_index().has_value() && child->paintable_box().computed_values().z_index().value() >= 1)
|
|
paint_child(context, *child);
|
|
}
|
|
|
|
paint_node(paintable_box(), context, PaintPhase::Outline);
|
|
|
|
if (context.should_paint_overlay()) {
|
|
paint_node(paintable_box(), context, PaintPhase::Overlay);
|
|
}
|
|
}
|
|
|
|
void StackingContext::paint(DisplayListRecordingContext& context) const
|
|
{
|
|
// https://drafts.csswg.org/css-transforms-1/#transform-function-lists
|
|
// If a transform function causes the current transformation matrix of an object to be non-invertible, the object
|
|
// and its content do not get displayed.
|
|
if (paintable_box().has_non_invertible_css_transform())
|
|
return;
|
|
|
|
if (paintable_box().computed_values().opacity() == 0.0f)
|
|
return;
|
|
|
|
TemporaryChange save_nesting_level(context.display_list_recorder().m_save_nesting_level, 0);
|
|
ScopeGuard verify_save_and_restore_are_balanced([&] {
|
|
VERIFY(context.display_list_recorder().m_save_nesting_level == 0);
|
|
});
|
|
|
|
auto const& computed_values = paintable_box().computed_values();
|
|
auto mask_image = computed_values.mask_image();
|
|
|
|
if (mask_image) {
|
|
mask_image->resolve_for_size(paintable_box().layout_node_with_style_and_box_metrics(), paintable_box().absolute_padding_box_rect().size());
|
|
}
|
|
|
|
auto effective_context_index = paintable_box().accumulated_visual_context_index();
|
|
context.display_list_recorder().set_accumulated_visual_context(effective_context_index);
|
|
|
|
// For elements with SVG filters, emit a transparent FillRect to trigger filter application.
|
|
// This ensures content-generating filters (feFlood, feImage) work even with empty source.
|
|
if (auto const& bounds = paintable_box().filter().svg_filter_bounds; bounds.has_value()) {
|
|
auto device_rect = context.enclosing_device_rect(*bounds).to_type<int>();
|
|
context.display_list_recorder().fill_rect_transparent(device_rect);
|
|
}
|
|
|
|
// Collect all masks (CSS mask-image, SVG <mask>, SVG <clipPath>).
|
|
Vector<DisplayListRecorder::MaskInfo> masks;
|
|
|
|
if (mask_image) {
|
|
auto mask_display_list = DisplayList::create(AccumulatedVisualContextTree::create());
|
|
DisplayListRecorder display_list_recorder(*mask_display_list);
|
|
auto mask_painting_context = context.clone(display_list_recorder);
|
|
auto mask_rect_in_device_pixels = context.enclosing_device_rect(paintable_box().absolute_padding_box_rect());
|
|
mask_image->paint(mask_painting_context, { {}, mask_rect_in_device_pixels.size() }, CSS::ImageRendering::Auto);
|
|
masks.append({ mask_display_list, mask_rect_in_device_pixels.to_type<int>(), Gfx::MaskKind::Alpha });
|
|
}
|
|
|
|
if (auto mask_area = paintable_box().get_mask_area(); mask_area.has_value()) {
|
|
if (auto mask_display_list = paintable_box().calculate_mask(context, *mask_area)) {
|
|
auto rect = context.enclosing_device_rect(*mask_area).to_type<int>();
|
|
auto kind = paintable_box().get_mask_type().value_or(Gfx::MaskKind::Alpha);
|
|
masks.append({ mask_display_list, rect, kind });
|
|
}
|
|
}
|
|
|
|
if (auto clip_area = paintable_box().get_clip_area(); clip_area.has_value()) {
|
|
if (auto clip_display_list = paintable_box().calculate_clip(context, *clip_area)) {
|
|
auto rect = context.enclosing_device_rect(*clip_area).to_type<int>();
|
|
masks.append({ clip_display_list, rect, Gfx::MaskKind::Alpha });
|
|
}
|
|
}
|
|
|
|
context.display_list_recorder().begin_masks(masks);
|
|
|
|
auto context_before_children = context.display_list_recorder().accumulated_visual_context();
|
|
|
|
paint_internal(context);
|
|
|
|
context.display_list_recorder().set_accumulated_visual_context(context_before_children);
|
|
|
|
context.display_list_recorder().end_masks(masks);
|
|
}
|
|
|
|
TraversalDecision StackingContext::hit_test(CSSPixelPoint position, HitTestType type, Function<TraversalDecision(HitTestResult)> const& callback) const
|
|
{
|
|
// AD-HOC: Elements with non-invertible transforms are not displayed per the spec,
|
|
// so they should not be hit-testable either.
|
|
if (paintable_box().has_non_invertible_css_transform())
|
|
return TraversalDecision::Continue;
|
|
|
|
auto const is_visible = paintable_box().computed_values().visibility() == CSS::Visibility::Visible;
|
|
|
|
// NOTE: Hit testing basically happens in reverse painting order.
|
|
// https://drafts.csswg.org/css2/#z-index
|
|
|
|
// 7. the child stacking contexts with positive stack levels (least positive first).
|
|
// NOTE: Hit testing follows reverse painting order, that's why the conditions here are reversed.
|
|
for (auto const child : m_children.in_reverse()) {
|
|
if (child->paintable_box().computed_values().z_index().value_or(0) <= 0)
|
|
break;
|
|
if (child->hit_test(position, type, callback) == TraversalDecision::Break)
|
|
return TraversalDecision::Break;
|
|
}
|
|
|
|
// 6. the child stacking contexts with stack level 0 and the positioned descendants with stack level 0.
|
|
for (auto const& paintable_box : m_positioned_descendants_and_stacking_contexts_with_stack_level_0.in_reverse()) {
|
|
if (paintable_box->stacking_context()) {
|
|
if (paintable_box->stacking_context()->hit_test(position, type, callback) == TraversalDecision::Break)
|
|
return TraversalDecision::Break;
|
|
} else {
|
|
if (paintable_box->hit_test(position, type, callback) == TraversalDecision::Break)
|
|
return TraversalDecision::Break;
|
|
}
|
|
}
|
|
|
|
// 5. the in-flow, inline-level, non-positioned descendants, including inline tables and inline blocks.
|
|
if (paintable_box().layout_node().children_are_inline() && is<Layout::BlockContainer>(paintable_box().layout_node())) {
|
|
for (auto const* paintable = paintable_box().last_child(); paintable; paintable = paintable->previous_sibling()) {
|
|
if (paintable->is_inline() && !paintable->is_absolutely_positioned() && !paintable->has_stacking_context()) {
|
|
if (paintable->hit_test(position, type, callback) == TraversalDecision::Break)
|
|
return TraversalDecision::Break;
|
|
}
|
|
}
|
|
|
|
// Hit test the stacking context root's own fragments if it's a PaintableWithLines.
|
|
if (is<PaintableWithLines>(paintable_box())) {
|
|
auto const& paintable_with_lines = as<PaintableWithLines>(paintable_box());
|
|
auto local_position = paintable_box().transform_point_to_local(position);
|
|
|
|
if (local_position.has_value()) {
|
|
if (paintable_with_lines.hit_test_fragments(position, local_position.value(), type, callback) == TraversalDecision::Break)
|
|
return TraversalDecision::Break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// 4. the non-positioned floats.
|
|
for (auto const& paintable_box : m_non_positioned_floating_descendants.in_reverse()) {
|
|
if (paintable_box->hit_test(position, type, callback) == TraversalDecision::Break)
|
|
return TraversalDecision::Break;
|
|
}
|
|
|
|
// 3. the in-flow, non-inline-level, non-positioned descendants.
|
|
if (!paintable_box().layout_node().children_are_inline()) {
|
|
for (auto const* child = paintable_box().last_child(); child; child = child->previous_sibling()) {
|
|
if (!child->is_paintable_box())
|
|
continue;
|
|
|
|
auto const& paintable_box = as<PaintableBox>(*child);
|
|
if (!paintable_box.is_absolutely_positioned() && !paintable_box.is_floating() && !paintable_box.stacking_context()) {
|
|
if (paintable_box.hit_test(position, type, callback) == TraversalDecision::Break)
|
|
return TraversalDecision::Break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// 2. the child stacking contexts with negative stack levels (most negative first).
|
|
// NB: Hit testing follows reverse painting order, so we visit the least negative stack levels first.
|
|
for (auto const child : m_children.in_reverse()) {
|
|
// Skip positive/ zero index child stacking contexts, which have already been handled above.
|
|
if (child->paintable_box().computed_values().z_index().value_or(0) >= 0)
|
|
continue;
|
|
if (child->hit_test(position, type, callback) == TraversalDecision::Break)
|
|
return TraversalDecision::Break;
|
|
}
|
|
|
|
// Hidden elements and elements with pointer-events: none shouldn't be hit.
|
|
if (!is_visible || !paintable_box().visible_for_hit_testing())
|
|
return TraversalDecision::Continue;
|
|
|
|
auto local_position = paintable_box().transform_point_to_local(position);
|
|
|
|
if (local_position.has_value() && paintable_box().absolute_border_box_rect().contains(local_position.value())) {
|
|
if (callback({ .paintable = const_cast<PaintableBox&>(paintable_box()) }) == TraversalDecision::Break)
|
|
return TraversalDecision::Break;
|
|
}
|
|
|
|
return TraversalDecision::Continue;
|
|
}
|
|
|
|
void StackingContext::dump(StringBuilder& builder, int indent) const
|
|
{
|
|
for (int i = 0; i < indent; ++i)
|
|
builder.append(' ');
|
|
CSSPixelRect rect = paintable_box().absolute_rect();
|
|
builder.appendff("SC for {} {} [children: {}] (z-index: ", paintable_box().layout_node().debug_description(), rect, m_children.size());
|
|
|
|
if (paintable_box().computed_values().z_index().has_value())
|
|
builder.appendff("{}", paintable_box().computed_values().z_index().value());
|
|
else
|
|
builder.append("auto"sv);
|
|
builder.append(')');
|
|
|
|
if (paintable_box().has_css_transform())
|
|
builder.append(", has_transform"sv);
|
|
|
|
builder.append('\n');
|
|
for (auto& child : m_children)
|
|
child->dump(builder, indent + 1);
|
|
}
|
|
|
|
}
|