summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAndi Gallo <andigallo@proton.me>2023-05-24 06:57:38 +0000
committerAndreas Kling <kling@serenityos.org>2023-05-26 14:53:53 +0200
commit5cec5171538a003d2880d412d3bcae79e7bd88ca (patch)
tree47966cc9f36532d01e79e41c445270a8da695a0a
parent706a20c4d4acbfc9eb9d980969f29e4b1c7a4d1c (diff)
downloadserenity-5cec5171538a003d2880d412d3bcae79e7bd88ca.zip
LibWeb: Get reference height from closest non-anonymous ancestor
Ignore anonymous block boxes when resolving percentage weights that would refer to them, per the CSS 2 visual formatting model specification. This fixes the case when we create an anonymous block between an image which uses a percentage height relative to a parent which specifies a definite height. Fixes #19052.
-rw-r--r--Tests/LibWeb/Layout/expected/resolve-height-of-containing-block.txt26
-rw-r--r--Tests/LibWeb/Layout/input/resolve-height-of-containing-block.html33
-rw-r--r--Userland/Libraries/LibWeb/Layout/FormattingContext.cpp6
-rw-r--r--Userland/Libraries/LibWeb/Layout/Node.cpp11
-rw-r--r--Userland/Libraries/LibWeb/Layout/Node.h6
5 files changed, 79 insertions, 3 deletions
diff --git a/Tests/LibWeb/Layout/expected/resolve-height-of-containing-block.txt b/Tests/LibWeb/Layout/expected/resolve-height-of-containing-block.txt
new file mode 100644
index 0000000000..d8d1dedc68
--- /dev/null
+++ b/Tests/LibWeb/Layout/expected/resolve-height-of-containing-block.txt
@@ -0,0 +1,26 @@
+Viewport <#document> at (0,0) content-size 800x600 children: not-inline
+ BlockContainer <html> at (0,0) content-size 800x824 [BFC] children: not-inline
+ BlockContainer <(anonymous)> at (0,0) content-size 800x0 children: inline
+ TextNode <#text>
+ BlockContainer <body> at (8,16) content-size 784x800 children: not-inline
+ BlockContainer <(anonymous)> at (8,16) content-size 784x0 children: inline
+ TextNode <#text>
+ BlockContainer <div.foo> at (8,16) content-size 1280x800 children: not-inline
+ BlockContainer <(anonymous)> at (8,16) content-size 1280x0 children: inline
+ TextNode <#text>
+ BlockContainer <div> at (8,16) content-size 1280x600 children: not-inline
+ BlockContainer <(anonymous)> at (8,16) content-size 1280x0 children: inline
+ TextNode <#text>
+ ImageBox <img> at (88,16) content-size 1200x600 floating children: not-inline
+ TextNode <#text>
+ BlockContainer <p> at (8,16) content-size 1280x17.46875 children: inline
+ line 0 width: 37.21875, height: 17.46875, bottom: 17.46875, baseline: 13.53125
+ frag 0 from TextNode start: 0, length: 4, rect: [8,16 37.21875x17.46875]
+ "Test"
+ TextNode <#text>
+ BlockContainer <(anonymous)> at (8,49.46875) content-size 1280x0 children: inline
+ TextNode <#text>
+ BlockContainer <(anonymous)> at (8,616) content-size 1280x0 children: inline
+ TextNode <#text>
+ BlockContainer <(anonymous)> at (8,816) content-size 784x0 children: inline
+ TextNode <#text>
diff --git a/Tests/LibWeb/Layout/input/resolve-height-of-containing-block.html b/Tests/LibWeb/Layout/input/resolve-height-of-containing-block.html
new file mode 100644
index 0000000000..b8aa340683
--- /dev/null
+++ b/Tests/LibWeb/Layout/input/resolve-height-of-containing-block.html
@@ -0,0 +1,33 @@
+<!DOCTYPE html>
+
+<head>
+ <style>
+ .foo {
+ width: 1280px;
+ height: 800px;
+ }
+
+ .foo div {
+ height: 600px;
+ }
+
+ .foo div img {
+ max-height: 100%;
+ float: right;
+ }
+ </style>
+</head>
+
+<body>
+
+ <div class="foo">
+ <div>
+ <img
+ src="">
+ <p>Test</p>
+ </div>
+ </div>
+
+</body>
+
+</html> \ No newline at end of file
diff --git a/Userland/Libraries/LibWeb/Layout/FormattingContext.cpp b/Userland/Libraries/LibWeb/Layout/FormattingContext.cpp
index 5682585212..f05884f469 100644
--- a/Userland/Libraries/LibWeb/Layout/FormattingContext.cpp
+++ b/Userland/Libraries/LibWeb/Layout/FormattingContext.cpp
@@ -262,7 +262,7 @@ static CSSPixelSize solve_replaced_size_constraint(LayoutState const& state, CSS
{
// 10.4 Minimum and maximum widths: 'min-width' and 'max-width'
- auto const& containing_block = *box.containing_block();
+ auto const& containing_block = *box.non_anyonymous_containing_block();
auto const& containing_block_state = state.get(containing_block);
auto width_of_containing_block = containing_block_state.content_width();
auto height_of_containing_block = containing_block_state.content_height();
@@ -509,7 +509,7 @@ CSSPixels FormattingContext::compute_height_for_replaced_element(LayoutState con
// 10.6.6 Floating replaced elements
// 10.6.10 'inline-block' replaced elements in normal flow
- auto height_of_containing_block = state.get(*box.containing_block()).content_height();
+ auto height_of_containing_block = state.get(*box.non_anyonymous_containing_block()).content_height();
auto computed_width = should_treat_width_as_auto(box, available_space) ? CSS::Size::make_auto() : box.computed_values().width();
auto computed_height = should_treat_height_as_auto(box, available_space) ? CSS::Size::make_auto() : box.computed_values().height();
@@ -1358,7 +1358,7 @@ CSS::Length FormattingContext::calculate_inner_width(Layout::Box const& box, Ava
CSS::Length FormattingContext::calculate_inner_height(Layout::Box const& box, AvailableSize const&, CSS::Size const& height) const
{
- auto height_of_containing_block = m_state.get(*box.containing_block()).content_height();
+ auto height_of_containing_block = m_state.get(*box.non_anyonymous_containing_block()).content_height();
auto height_of_containing_block_as_length_for_resolve = CSS::Length::make_px(height_of_containing_block);
if (height.is_auto()) {
return height.resolved(box, height_of_containing_block_as_length_for_resolve);
diff --git a/Userland/Libraries/LibWeb/Layout/Node.cpp b/Userland/Libraries/LibWeb/Layout/Node.cpp
index 60d4c00485..956e56120f 100644
--- a/Userland/Libraries/LibWeb/Layout/Node.cpp
+++ b/Userland/Libraries/LibWeb/Layout/Node.cpp
@@ -114,6 +114,17 @@ Box const* Node::containing_block() const
return nearest_ancestor_capable_of_forming_a_containing_block(*this);
}
+Box const* Node::non_anyonymous_containing_block() const
+{
+ 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;
+}
+
// https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Positioning/Understanding_z_index/The_stacking_context
bool Node::establishes_stacking_context() const
{
diff --git a/Userland/Libraries/LibWeb/Layout/Node.h b/Userland/Libraries/LibWeb/Layout/Node.h
index 70b1ebd267..0919077882 100644
--- a/Userland/Libraries/LibWeb/Layout/Node.h
+++ b/Userland/Libraries/LibWeb/Layout/Node.h
@@ -112,6 +112,12 @@ public:
Box const* containing_block() const;
Box* containing_block() { return const_cast<Box*>(const_cast<Node const*>(this)->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_anyonymous_containing_block() const;
+
bool establishes_stacking_context() const;
bool can_contain_boxes_with_position_absolute() const;