/* * Copyright (c) 2024-2025, Sam Atkins * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include namespace Web::DOM { AbstractElement::AbstractElement(GC::Ref element, Optional pseudo_element) : m_element(element) , m_pseudo_element(move(pseudo_element)) { } void AbstractElement::visit(GC::Cell::Visitor& visitor) const { visitor.visit(m_element); visitor.visit(m_inheritance_override); } Document& AbstractElement::document() const { return m_element->document(); } CSS::TreeCountingFunctionResolutionContext AbstractElement::tree_counting_function_resolution_context() const { // FIXME: When used on an element-backed pseudo-element which is also a real element, the tree counting functions // resolve for that real element. For other pseudo elements, they resolve as if they were resolved against // the originating element. It follows that for nested pseudo elements the resolution will recursively walk // the originating elements until a real element is found. // FIXME: A tree counting function is a tree-scoped reference where it references an implicit tree-scoped name for // the element it resolves against. This is done to not leak tree information to an outer tree. A tree // counting function that is scoped to an outer tree relative to the element it resolves against, will alway // resolve to 0. auto const& element_to_resolve_tree_counting_function_against = element(); // The sibling-count() functional notation represents, as an , the total number of child elements in the // parent of the element on which the notation is used. auto const& parent = element_to_resolve_tree_counting_function_against.parent_element(); // If there is no parent we are the root node if (!parent) return { .sibling_count = 1, .sibling_index = 1 }; size_t count = 0; size_t index = 0; for (auto const* child = parent->first_child_of_type(); child; child = child->next_element_sibling()) { ++count; if (child == &element_to_resolve_tree_counting_function_against) index = count; } return { .sibling_count = count, .sibling_index = index }; } GC::Ptr AbstractElement::layout_node() { if (m_pseudo_element.has_value()) return m_element->get_pseudo_element_node(*m_pseudo_element); return m_element->layout_node(); } GC::Ptr AbstractElement::parent_element() const { if (m_pseudo_element.has_value()) return m_element; return m_element->parent_element(); } Optional AbstractElement::element_to_inherit_style_from() const { if (m_inheritance_override) return AbstractElement { *m_inheritance_override }; GC::Ptr element = m_element->element_to_inherit_style_from(m_pseudo_element); if (!element) return OptionalNone {}; return AbstractElement { const_cast(*element) }; } Optional AbstractElement::walk_layout_tree(WalkMethod walk_method) { GC::Ptr node = layout_node(); if (!node) return OptionalNone {}; while (true) { switch (walk_method) { case WalkMethod::Previous: node = node->previous_in_pre_order(); break; case WalkMethod::PreviousSibling: node = node->previous_sibling(); break; } if (!node) return OptionalNone {}; if (auto* previous_element = as_if(node->dom_node())) return AbstractElement { *previous_element }; if (node->is_generated_for_pseudo_element()) return AbstractElement { *node->pseudo_element_generator(), node->generated_for_pseudo_element() }; } } bool AbstractElement::is_before(AbstractElement const& other) const { auto this_node = layout_node(); auto other_node = other.layout_node(); return this_node && other_node && this_node->is_before(*other_node); } GC::Ptr AbstractElement::computed_properties() const { return m_element->computed_properties(m_pseudo_element); } OrderedHashMap const& AbstractElement::custom_properties() const { return m_element->custom_properties(m_pseudo_element); } void AbstractElement::set_custom_properties(OrderedHashMap&& custom_properties) { m_element->set_custom_properties(m_pseudo_element, move(custom_properties)); } RefPtr AbstractElement::get_custom_property(FlyString const& name) const { // FIXME: We should be producing computed values for custom properties, just like regular properties. if (m_pseudo_element.has_value()) { auto const& custom_properties = m_element->custom_properties(*m_pseudo_element); if (auto it = custom_properties.find(name); it != custom_properties.end()) { return it->value.value; } } for (auto const* current_element = m_element.ptr(); current_element; current_element = current_element->parent_or_shadow_host_element()) { auto const& custom_properties = current_element->custom_properties({}); if (auto it = custom_properties.find(name); it != custom_properties.end()) { return it->value.value; } } return nullptr; } GC::Ptr AbstractElement::cascaded_properties() const { return m_element->cascaded_properties(m_pseudo_element); } void AbstractElement::set_cascaded_properties(GC::Ptr cascaded_properties) { m_element->set_cascaded_properties(m_pseudo_element, cascaded_properties); } bool AbstractElement::has_non_empty_counters_set() const { if (m_pseudo_element.has_value()) return m_element->get_pseudo_element(*m_pseudo_element)->has_non_empty_counters_set(); return m_element->has_non_empty_counters_set(); } Optional AbstractElement::counters_set() const { if (m_pseudo_element.has_value()) return m_element->get_pseudo_element(*m_pseudo_element)->counters_set(); return m_element->counters_set(); } CSS::CountersSet& AbstractElement::ensure_counters_set() { if (m_pseudo_element.has_value()) return m_element->get_pseudo_element(*m_pseudo_element)->ensure_counters_set(); return m_element->ensure_counters_set(); } void AbstractElement::set_counters_set(OwnPtr&& counters_set) { if (m_pseudo_element.has_value()) { m_element->get_pseudo_element(*m_pseudo_element)->set_counters_set(move(counters_set)); } else { m_element->set_counters_set(move(counters_set)); } } String AbstractElement::debug_description() const { if (m_pseudo_element.has_value()) { StringBuilder builder; builder.append(m_element->debug_description()); builder.append("::"sv); builder.append(CSS::pseudo_element_name(*m_pseudo_element)); return builder.to_string_without_validation(); } return m_element->debug_description(); } }