mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-11-03 23:00:58 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			162 lines
		
	
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			162 lines
		
	
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2024-2025, Sam Atkins <sam@ladybird.org>
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier: BSD-2-Clause
 | 
						|
 */
 | 
						|
 | 
						|
#include <LibWeb/DOM/AbstractElement.h>
 | 
						|
#include <LibWeb/DOM/Element.h>
 | 
						|
#include <LibWeb/Layout/Node.h>
 | 
						|
 | 
						|
namespace Web::DOM {
 | 
						|
 | 
						|
AbstractElement::AbstractElement(GC::Ref<Element> element, Optional<CSS::PseudoElement> pseudo_element)
 | 
						|
    : m_element(element)
 | 
						|
    , m_pseudo_element(move(pseudo_element))
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
void AbstractElement::visit(GC::Cell::Visitor& visitor) const
 | 
						|
{
 | 
						|
    visitor.visit(m_element);
 | 
						|
}
 | 
						|
 | 
						|
Document& AbstractElement::document() const
 | 
						|
{
 | 
						|
    return m_element->document();
 | 
						|
}
 | 
						|
 | 
						|
GC::Ptr<Layout::NodeWithStyle> 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<Element const> AbstractElement::parent_element() const
 | 
						|
{
 | 
						|
    if (m_pseudo_element.has_value())
 | 
						|
        return m_element;
 | 
						|
    return m_element->parent_element();
 | 
						|
}
 | 
						|
 | 
						|
Optional<AbstractElement> AbstractElement::element_to_inherit_style_from() const
 | 
						|
{
 | 
						|
    GC::Ptr<Element const> element = m_element->element_to_inherit_style_from(m_pseudo_element);
 | 
						|
 | 
						|
    if (!element)
 | 
						|
        return OptionalNone {};
 | 
						|
 | 
						|
    return AbstractElement { const_cast<DOM::Element&>(*element) };
 | 
						|
}
 | 
						|
 | 
						|
Optional<AbstractElement> AbstractElement::walk_layout_tree(WalkMethod walk_method)
 | 
						|
{
 | 
						|
    GC::Ptr<Layout::Node> 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<Element>(node->dom_node()))
 | 
						|
            return AbstractElement { *previous_element };
 | 
						|
 | 
						|
        if (node->is_generated())
 | 
						|
            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<CSS::ComputedProperties const> AbstractElement::computed_properties() const
 | 
						|
{
 | 
						|
    return m_element->computed_properties(m_pseudo_element);
 | 
						|
}
 | 
						|
 | 
						|
HashMap<FlyString, CSS::StyleProperty> const& AbstractElement::custom_properties() const
 | 
						|
{
 | 
						|
    return m_element->custom_properties(m_pseudo_element);
 | 
						|
}
 | 
						|
 | 
						|
void AbstractElement::set_custom_properties(HashMap<FlyString, CSS::StyleProperty>&& custom_properties)
 | 
						|
{
 | 
						|
    m_element->set_custom_properties(m_pseudo_element, move(custom_properties));
 | 
						|
}
 | 
						|
 | 
						|
RefPtr<CSS::StyleValue const> 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;
 | 
						|
}
 | 
						|
 | 
						|
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<CSS::CountersSet const&> 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<CSS::CountersSet>&& 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();
 | 
						|
}
 | 
						|
 | 
						|
}
 |