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/Position.cpp
DOM/ProcessingInstruction.cpp
DOM/PseudoElement.cpp
DOM/QualifiedName.cpp
DOM/Range.cpp
DOM/ShadowRoot.cpp

View file

@ -91,9 +91,6 @@
namespace Web::DOM {
GC_DEFINE_ALLOCATOR(Element::PseudoElement);
GC_DEFINE_ALLOCATOR(Element::PseudoElementTreeNode);
Element::Element(Document& document, DOM::QualifiedName qualified_name)
: ParentNode(document, NodeType::ELEMENT_NODE)
, 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
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++) {
auto pseudo_element_type = static_cast<CSS::PseudoElement>(i);
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;
auto pseudo_element_style = pseudo_element_computed_properties(pseudo_element_type);
if (!pseudo_element_style)
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);
if (invalidation.repaint && node_with_style->first_paintable())
node_with_style->first_paintable()->set_needs_display();
@ -1392,13 +1380,13 @@ void Element::set_pseudo_element_node(Badge<Layout::TreeBuilder>, CSS::PseudoEle
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
{
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;
}
@ -1409,9 +1397,9 @@ bool Element::affected_by_pseudo_class(CSS::PseudoClass pseudo_class) const
}
if (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;
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;
}
}
@ -1578,7 +1566,7 @@ bool Element::has_pseudo_elements() const
{
if (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;
}
}
@ -1589,7 +1577,7 @@ void Element::clear_pseudo_element_nodes(Badge<Layout::TreeBuilder>)
{
if (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()) {
auto pseudo_element_data = get_pseudo_element(pseudo_element.value());
if (pseudo_element_data.has_value())
return pseudo_element_data->cascaded_properties;
return pseudo_element_data->cascaded_properties();
return nullptr;
}
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.value() >= CSS::PseudoElement::KnownPseudoElementCount)
return;
ensure_pseudo_element(pseudo_element.value()).cascaded_properties = cascaded_properties;
ensure_pseudo_element(pseudo_element.value()).set_cascaded_properties(cascaded_properties);
} else {
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))
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)
{
auto pseudo_element = get_pseudo_element(type);
if (pseudo_element.has_value())
return pseudo_element->computed_properties;
return pseudo_element->computed_properties();
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)
return {};
@ -2961,7 +2949,7 @@ Optional<Element::PseudoElement&> Element::get_pseudo_element(CSS::PseudoElement
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)
m_pseudo_element_data = make<PseudoElementData>();
@ -2990,7 +2978,7 @@ void Element::set_custom_properties(Optional<CSS::PseudoElement> pseudo_element,
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
@ -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()));
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

View file

@ -23,6 +23,7 @@
#include <LibWeb/DOM/ChildNode.h>
#include <LibWeb/DOM/NonDocumentTypeChildNode.h>
#include <LibWeb/DOM/ParentNode.h>
#include <LibWeb/DOM/PseudoElement.h>
#include <LibWeb/DOM/QualifiedName.h>
#include <LibWeb/DOM/Slottable.h>
#include <LibWeb/HTML/AttributeNames.h>
@ -564,25 +565,6 @@ private:
GC::Ptr<CSS::ComputedProperties> m_computed_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>>;
mutable OwnPtr<PseudoElementData> m_pseudo_element_data;
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);
if (!pseudo_element.has_value())
return false;
return pseudo_element.value()->layout_node;
return pseudo_element.value()->layout_node();
}
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 Position;
class ProcessingInstruction;
class PseudoElement;
class Range;
class RegisteredObserver;
class ShadowRoot;