LibWeb: Make PseudoElement a class in its own right

It's getting a bit large and complicated to be a struct hidden in
DOM::Element.
This commit is contained in:
Sam Atkins 2025-05-23 17:22:48 +01:00
commit e7c2f0dd52
Notes: github-actions[bot] 2025-06-19 11:37:51 +00:00
6 changed files with 96 additions and 48 deletions

View file

@ -250,6 +250,7 @@ set(SOURCES
DOM/ParentNode.cpp DOM/ParentNode.cpp
DOM/Position.cpp DOM/Position.cpp
DOM/ProcessingInstruction.cpp DOM/ProcessingInstruction.cpp
DOM/PseudoElement.cpp
DOM/QualifiedName.cpp DOM/QualifiedName.cpp
DOM/Range.cpp DOM/Range.cpp
DOM/ShadowRoot.cpp DOM/ShadowRoot.cpp

View file

@ -91,9 +91,6 @@
namespace Web::DOM { namespace Web::DOM {
GC_DEFINE_ALLOCATOR(Element::PseudoElement);
GC_DEFINE_ALLOCATOR(Element::PseudoElementTreeNode);
Element::Element(Document& document, DOM::QualifiedName qualified_name) Element::Element(Document& document, DOM::QualifiedName qualified_name)
: ParentNode(document, NodeType::ELEMENT_NODE) : ParentNode(document, NodeType::ELEMENT_NODE)
, m_qualified_name(move(qualified_name)) , m_qualified_name(move(qualified_name))
@ -133,15 +130,6 @@ void Element::visit_edges(Cell::Visitor& visitor)
} }
} }
void Element::PseudoElement::visit_edges(Cell::Visitor& visitor)
{
Base::visit_edges(visitor);
visitor.visit(cascaded_properties);
visitor.visit(computed_properties);
visitor.visit(layout_node);
}
// https://dom.spec.whatwg.org/#dom-element-getattribute // https://dom.spec.whatwg.org/#dom-element-getattribute
Optional<String> Element::get_attribute(FlyString const& name) const Optional<String> Element::get_attribute(FlyString const& name) const
{ {
@ -780,14 +768,14 @@ CSS::RequiredInvalidationAfterStyleChange Element::recompute_style()
for (auto i = 0; i < to_underlying(CSS::PseudoElement::KnownPseudoElementCount); i++) { for (auto i = 0; i < to_underlying(CSS::PseudoElement::KnownPseudoElementCount); i++) {
auto pseudo_element_type = static_cast<CSS::PseudoElement>(i); auto pseudo_element_type = static_cast<CSS::PseudoElement>(i);
auto pseudo_element = get_pseudo_element(pseudo_element_type); auto pseudo_element = get_pseudo_element(pseudo_element_type);
if (!pseudo_element.has_value() || !pseudo_element->layout_node) if (!pseudo_element.has_value() || !pseudo_element->layout_node())
continue; continue;
auto pseudo_element_style = pseudo_element_computed_properties(pseudo_element_type); auto pseudo_element_style = pseudo_element_computed_properties(pseudo_element_type);
if (!pseudo_element_style) if (!pseudo_element_style)
continue; continue;
if (auto* node_with_style = dynamic_cast<Layout::NodeWithStyle*>(pseudo_element->layout_node.ptr())) { if (auto node_with_style = pseudo_element->layout_node()) {
node_with_style->apply_style(*pseudo_element_style); node_with_style->apply_style(*pseudo_element_style);
if (invalidation.repaint && node_with_style->first_paintable()) if (invalidation.repaint && node_with_style->first_paintable())
node_with_style->first_paintable()->set_needs_display(); node_with_style->first_paintable()->set_needs_display();
@ -1392,13 +1380,13 @@ void Element::set_pseudo_element_node(Badge<Layout::TreeBuilder>, CSS::PseudoEle
return; return;
} }
ensure_pseudo_element(pseudo_element).layout_node = move(pseudo_element_node); ensure_pseudo_element(pseudo_element).set_layout_node(move(pseudo_element_node));
} }
GC::Ptr<Layout::NodeWithStyle> Element::get_pseudo_element_node(CSS::PseudoElement pseudo_element) const GC::Ptr<Layout::NodeWithStyle> Element::get_pseudo_element_node(CSS::PseudoElement pseudo_element) const
{ {
if (auto element_data = get_pseudo_element(pseudo_element); element_data.has_value()) if (auto element_data = get_pseudo_element(pseudo_element); element_data.has_value())
return element_data->layout_node; return element_data->layout_node();
return nullptr; return nullptr;
} }
@ -1409,9 +1397,9 @@ bool Element::affected_by_pseudo_class(CSS::PseudoClass pseudo_class) const
} }
if (m_pseudo_element_data) { if (m_pseudo_element_data) {
for (auto& pseudo_element : *m_pseudo_element_data) { for (auto& pseudo_element : *m_pseudo_element_data) {
if (!pseudo_element.value->computed_properties) if (!pseudo_element.value->computed_properties())
continue; continue;
if (pseudo_element.value->computed_properties->has_attempted_match_against_pseudo_class(pseudo_class)) if (pseudo_element.value->computed_properties()->has_attempted_match_against_pseudo_class(pseudo_class))
return true; return true;
} }
} }
@ -1578,7 +1566,7 @@ bool Element::has_pseudo_elements() const
{ {
if (m_pseudo_element_data) { if (m_pseudo_element_data) {
for (auto& pseudo_element : *m_pseudo_element_data) { for (auto& pseudo_element : *m_pseudo_element_data) {
if (pseudo_element.value->layout_node) if (pseudo_element.value->layout_node())
return true; return true;
} }
} }
@ -1589,7 +1577,7 @@ void Element::clear_pseudo_element_nodes(Badge<Layout::TreeBuilder>)
{ {
if (m_pseudo_element_data) { if (m_pseudo_element_data) {
for (auto& pseudo_element : *m_pseudo_element_data) { for (auto& pseudo_element : *m_pseudo_element_data) {
pseudo_element.value->layout_node = nullptr; pseudo_element.value->set_layout_node(nullptr);
} }
} }
} }
@ -2903,7 +2891,7 @@ GC::Ptr<CSS::CascadedProperties> Element::cascaded_properties(Optional<CSS::Pseu
if (pseudo_element.has_value()) { if (pseudo_element.has_value()) {
auto pseudo_element_data = get_pseudo_element(pseudo_element.value()); auto pseudo_element_data = get_pseudo_element(pseudo_element.value());
if (pseudo_element_data.has_value()) if (pseudo_element_data.has_value())
return pseudo_element_data->cascaded_properties; return pseudo_element_data->cascaded_properties();
return nullptr; return nullptr;
} }
return m_cascaded_properties; return m_cascaded_properties;
@ -2914,7 +2902,7 @@ void Element::set_cascaded_properties(Optional<CSS::PseudoElement> pseudo_elemen
if (pseudo_element.has_value()) { if (pseudo_element.has_value()) {
if (pseudo_element.value() >= CSS::PseudoElement::KnownPseudoElementCount) if (pseudo_element.value() >= CSS::PseudoElement::KnownPseudoElementCount)
return; return;
ensure_pseudo_element(pseudo_element.value()).cascaded_properties = cascaded_properties; ensure_pseudo_element(pseudo_element.value()).set_cascaded_properties(cascaded_properties);
} else { } else {
m_cascaded_properties = cascaded_properties; m_cascaded_properties = cascaded_properties;
} }
@ -2934,18 +2922,18 @@ void Element::set_pseudo_element_computed_properties(CSS::PseudoElement pseudo_e
if (!CSS::Selector::PseudoElementSelector::is_known_pseudo_element_type(pseudo_element)) if (!CSS::Selector::PseudoElementSelector::is_known_pseudo_element_type(pseudo_element))
return; return;
ensure_pseudo_element(pseudo_element).computed_properties = style; ensure_pseudo_element(pseudo_element).set_computed_properties(style);
} }
GC::Ptr<CSS::ComputedProperties> Element::pseudo_element_computed_properties(CSS::PseudoElement type) GC::Ptr<CSS::ComputedProperties> Element::pseudo_element_computed_properties(CSS::PseudoElement type)
{ {
auto pseudo_element = get_pseudo_element(type); auto pseudo_element = get_pseudo_element(type);
if (pseudo_element.has_value()) if (pseudo_element.has_value())
return pseudo_element->computed_properties; return pseudo_element->computed_properties();
return {}; return {};
} }
Optional<Element::PseudoElement&> Element::get_pseudo_element(CSS::PseudoElement type) const Optional<PseudoElement&> Element::get_pseudo_element(CSS::PseudoElement type) const
{ {
if (!m_pseudo_element_data) if (!m_pseudo_element_data)
return {}; return {};
@ -2961,7 +2949,7 @@ Optional<Element::PseudoElement&> Element::get_pseudo_element(CSS::PseudoElement
return *(pseudo_element.value()); return *(pseudo_element.value());
} }
Element::PseudoElement& Element::ensure_pseudo_element(CSS::PseudoElement type) const PseudoElement& Element::ensure_pseudo_element(CSS::PseudoElement type) const
{ {
if (!m_pseudo_element_data) if (!m_pseudo_element_data)
m_pseudo_element_data = make<PseudoElementData>(); m_pseudo_element_data = make<PseudoElementData>();
@ -2990,7 +2978,7 @@ void Element::set_custom_properties(Optional<CSS::PseudoElement> pseudo_element,
return; return;
} }
ensure_pseudo_element(pseudo_element.value()).custom_properties = move(custom_properties); ensure_pseudo_element(pseudo_element.value()).set_custom_properties(move(custom_properties));
} }
HashMap<FlyString, CSS::StyleProperty> const& Element::custom_properties(Optional<CSS::PseudoElement> pseudo_element) const HashMap<FlyString, CSS::StyleProperty> const& Element::custom_properties(Optional<CSS::PseudoElement> pseudo_element) const
@ -3000,7 +2988,7 @@ HashMap<FlyString, CSS::StyleProperty> const& Element::custom_properties(Optiona
VERIFY(CSS::Selector::PseudoElementSelector::is_known_pseudo_element_type(pseudo_element.value())); VERIFY(CSS::Selector::PseudoElementSelector::is_known_pseudo_element_type(pseudo_element.value()));
return ensure_pseudo_element(pseudo_element.value()).custom_properties; return ensure_pseudo_element(pseudo_element.value()).custom_properties();
} }
// https://drafts.csswg.org/cssom-view/#dom-element-scroll // https://drafts.csswg.org/cssom-view/#dom-element-scroll

View file

@ -23,6 +23,7 @@
#include <LibWeb/DOM/ChildNode.h> #include <LibWeb/DOM/ChildNode.h>
#include <LibWeb/DOM/NonDocumentTypeChildNode.h> #include <LibWeb/DOM/NonDocumentTypeChildNode.h>
#include <LibWeb/DOM/ParentNode.h> #include <LibWeb/DOM/ParentNode.h>
#include <LibWeb/DOM/PseudoElement.h>
#include <LibWeb/DOM/QualifiedName.h> #include <LibWeb/DOM/QualifiedName.h>
#include <LibWeb/DOM/Slottable.h> #include <LibWeb/DOM/Slottable.h>
#include <LibWeb/HTML/AttributeNames.h> #include <LibWeb/HTML/AttributeNames.h>
@ -564,25 +565,6 @@ private:
GC::Ptr<CSS::ComputedProperties> m_computed_properties; GC::Ptr<CSS::ComputedProperties> m_computed_properties;
HashMap<FlyString, CSS::StyleProperty> m_custom_properties; HashMap<FlyString, CSS::StyleProperty> m_custom_properties;
struct PseudoElement : public JS::Cell {
GC_CELL(PseudoElement, JS::Cell);
GC_DECLARE_ALLOCATOR(PseudoElement);
GC::Ptr<Layout::NodeWithStyle> layout_node;
GC::Ptr<CSS::CascadedProperties> cascaded_properties;
GC::Ptr<CSS::ComputedProperties> computed_properties;
HashMap<FlyString, CSS::StyleProperty> custom_properties;
private:
virtual void visit_edges(JS::Cell::Visitor&) override;
};
// https://drafts.csswg.org/css-view-transitions/#pseudo-element-tree
struct PseudoElementTreeNode
: public PseudoElement
, TreeNode<PseudoElementTreeNode> {
GC_CELL(PseudoElementTreeNode, PseudoElement);
GC_DECLARE_ALLOCATOR(PseudoElementTreeNode);
};
using PseudoElementData = HashMap<CSS::PseudoElement, GC::Ref<PseudoElement>>; using PseudoElementData = HashMap<CSS::PseudoElement, GC::Ref<PseudoElement>>;
mutable OwnPtr<PseudoElementData> m_pseudo_element_data; mutable OwnPtr<PseudoElementData> m_pseudo_element_data;
Optional<PseudoElement&> get_pseudo_element(CSS::PseudoElement) const; Optional<PseudoElement&> get_pseudo_element(CSS::PseudoElement) const;
@ -673,7 +655,7 @@ inline bool Element::has_pseudo_element(CSS::PseudoElement type) const
auto pseudo_element = m_pseudo_element_data->get(type); auto pseudo_element = m_pseudo_element_data->get(type);
if (!pseudo_element.has_value()) if (!pseudo_element.has_value())
return false; return false;
return pseudo_element.value()->layout_node; return pseudo_element.value()->layout_node();
} }
bool is_valid_namespace_prefix(FlyString const&); bool is_valid_namespace_prefix(FlyString const&);

View file

@ -0,0 +1,26 @@
/*
* Copyright (c) 2025, Sam Atkins <sam@ladybird.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <LibWeb/CSS/CascadedProperties.h>
#include <LibWeb/CSS/ComputedProperties.h>
#include <LibWeb/DOM/PseudoElement.h>
#include <LibWeb/Layout/Node.h>
namespace Web::DOM {
GC_DEFINE_ALLOCATOR(PseudoElement);
GC_DEFINE_ALLOCATOR(PseudoElementTreeNode);
void PseudoElement::visit_edges(JS::Cell::Visitor& visitor)
{
Base::visit_edges(visitor);
visitor.visit(m_cascaded_properties);
visitor.visit(m_computed_properties);
visitor.visit(m_layout_node);
}
}

View file

@ -0,0 +1,50 @@
/*
* Copyright (c) 2025, Sam Atkins <sam@ladybird.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#pragma once
#include <LibGC/CellAllocator.h>
#include <LibJS/Heap/Cell.h>
#include <LibWeb/CSS/StyleProperty.h>
#include <LibWeb/Forward.h>
#include <LibWeb/TreeNode.h>
namespace Web::DOM {
class PseudoElement : public JS::Cell {
GC_CELL(PseudoElement, JS::Cell);
GC_DECLARE_ALLOCATOR(PseudoElement);
GC::Ptr<Layout::NodeWithStyle> layout_node() const { return m_layout_node; }
void set_layout_node(GC::Ptr<Layout::NodeWithStyle> value) { m_layout_node = value; }
GC::Ptr<CSS::CascadedProperties> cascaded_properties() const { return m_cascaded_properties; }
void set_cascaded_properties(GC::Ptr<CSS::CascadedProperties> value) { m_cascaded_properties = value; }
GC::Ptr<CSS::ComputedProperties> computed_properties() const { return m_computed_properties; }
void set_computed_properties(GC::Ptr<CSS::ComputedProperties> value) { m_computed_properties = value; }
HashMap<FlyString, CSS::StyleProperty> const& custom_properties() const { return m_custom_properties; }
void set_custom_properties(HashMap<FlyString, CSS::StyleProperty> value) { m_custom_properties = move(value); }
virtual void visit_edges(JS::Cell::Visitor&) override;
private:
GC::Ptr<Layout::NodeWithStyle> m_layout_node;
GC::Ptr<CSS::CascadedProperties> m_cascaded_properties;
GC::Ptr<CSS::ComputedProperties> m_computed_properties;
HashMap<FlyString, CSS::StyleProperty> m_custom_properties;
};
// https://drafts.csswg.org/css-view-transitions/#pseudo-element-tree
class PseudoElementTreeNode final
: public PseudoElement
, TreeNode<PseudoElementTreeNode> {
GC_CELL(PseudoElementTreeNode, PseudoElement);
GC_DECLARE_ALLOCATOR(PseudoElementTreeNode);
};
}

View file

@ -373,6 +373,7 @@ class NodeList;
class ParentNode; class ParentNode;
class Position; class Position;
class ProcessingInstruction; class ProcessingInstruction;
class PseudoElement;
class Range; class Range;
class RegisteredObserver; class RegisteredObserver;
class ShadowRoot; class ShadowRoot;