mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-08-08 01:00:05 +00:00
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:
parent
e56146edec
commit
e7c2f0dd52
Notes:
github-actions[bot]
2025-06-19 11:37:51 +00:00
Author: https://github.com/AtkinsSJ
Commit: e7c2f0dd52
Pull-request: https://github.com/LadybirdBrowser/ladybird/pull/5132
Reviewed-by: https://github.com/tcl3
6 changed files with 96 additions and 48 deletions
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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&);
|
||||||
|
|
26
Libraries/LibWeb/DOM/PseudoElement.cpp
Normal file
26
Libraries/LibWeb/DOM/PseudoElement.cpp
Normal 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);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
50
Libraries/LibWeb/DOM/PseudoElement.h
Normal file
50
Libraries/LibWeb/DOM/PseudoElement.h
Normal 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);
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
|
@ -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;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue