mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-11-03 23:00:58 +00:00 
			
		
		
		
	`Element::ordinal_value` is called for every `li` element in
a list (ul, ol, menu).
Before:
  `ordinal_value` iterates through all of the children of the list
  owner. It is called once for each element: complexity $O(n^2)$.
After:
  - Save the result of the first calculation in `m_ordinal_value`
  then return it in subsequent calls.
  - Tree modifications are intercepted and trigger invalidation
    of the first node's `m_ordinal_value`:
    - insert_before
    - append
    - remove
  Results in noticeable performance improvement rendering' large
  lists: from 20s to 4s for 20K elements.
		
	
			
		
			
				
	
	
		
			90 lines
		
	
	
	
		
			4.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			90 lines
		
	
	
	
		
			4.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2020, the SerenityOS developers.
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier: BSD-2-Clause
 | 
						|
 */
 | 
						|
 | 
						|
#include <LibWeb/Bindings/HTMLLIElementPrototype.h>
 | 
						|
#include <LibWeb/CSS/StyleValues/CSSKeywordValue.h>
 | 
						|
#include <LibWeb/HTML/HTMLLIElement.h>
 | 
						|
#include <LibWeb/HTML/Numbers.h>
 | 
						|
#include <LibWeb/HTML/Window.h>
 | 
						|
 | 
						|
namespace Web::HTML {
 | 
						|
 | 
						|
GC_DEFINE_ALLOCATOR(HTMLLIElement);
 | 
						|
 | 
						|
HTMLLIElement::HTMLLIElement(DOM::Document& document, DOM::QualifiedName qualified_name)
 | 
						|
    : HTMLElement(document, move(qualified_name))
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
HTMLLIElement::~HTMLLIElement() = default;
 | 
						|
 | 
						|
void HTMLLIElement::initialize(JS::Realm& realm)
 | 
						|
{
 | 
						|
    WEB_SET_PROTOTYPE_FOR_INTERFACE(HTMLLIElement);
 | 
						|
    Base::initialize(realm);
 | 
						|
}
 | 
						|
 | 
						|
void HTMLLIElement::attribute_changed(FlyString const& local_name, Optional<String> const& old_value, Optional<String> const& value, Optional<FlyString> const& namespace_)
 | 
						|
{
 | 
						|
    Base::attribute_changed(local_name, old_value, value, namespace_);
 | 
						|
 | 
						|
    if (local_name == HTML::AttributeNames::value) {
 | 
						|
        if (auto* owner = list_owner()) {
 | 
						|
            owner->set_needs_layout_tree_update(true, DOM::SetNeedsLayoutTreeUpdateReason::HTMLOListElementOrdinalValues);
 | 
						|
            maybe_invalidate_ordinals_for_list_owner();
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
// https://html.spec.whatwg.org/multipage/grouping-content.html#dom-li-value
 | 
						|
WebIDL::Long HTMLLIElement::value()
 | 
						|
{
 | 
						|
    // The value IDL attribute must reflect the value of the value content attribute.
 | 
						|
    // NOTE: This is equivalent to the code that would be generated by the IDL generator if we used [Reflect] on the value attribute.
 | 
						|
    //       We don't do that in this case, since this method is used elsewhere.
 | 
						|
    auto content_attribute_value = get_attribute(AttributeNames::value).value_or("0"_string);
 | 
						|
    if (auto maybe_number = HTML::parse_integer(content_attribute_value); maybe_number.has_value())
 | 
						|
        return *maybe_number;
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
bool HTMLLIElement::is_presentational_hint(FlyString const& name) const
 | 
						|
{
 | 
						|
    if (Base::is_presentational_hint(name))
 | 
						|
        return true;
 | 
						|
 | 
						|
    return name == HTML::AttributeNames::type;
 | 
						|
}
 | 
						|
 | 
						|
void HTMLLIElement::apply_presentational_hints(GC::Ref<CSS::CascadedProperties> cascaded_properties) const
 | 
						|
{
 | 
						|
    // https://html.spec.whatwg.org/multipage/rendering.html#lists
 | 
						|
    for_each_attribute([&](auto& name, auto& value) {
 | 
						|
        if (name == HTML::AttributeNames::type) {
 | 
						|
            if (value == "1"sv) {
 | 
						|
                cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::ListStyleType, CSS::CSSKeywordValue::create(CSS::Keyword::Decimal));
 | 
						|
            } else if (value == "a"sv) {
 | 
						|
                cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::ListStyleType, CSS::CSSKeywordValue::create(CSS::Keyword::LowerAlpha));
 | 
						|
            } else if (value == "A"sv) {
 | 
						|
                cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::ListStyleType, CSS::CSSKeywordValue::create(CSS::Keyword::UpperAlpha));
 | 
						|
            } else if (value == "i"sv) {
 | 
						|
                cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::ListStyleType, CSS::CSSKeywordValue::create(CSS::Keyword::LowerRoman));
 | 
						|
            } else if (value == "I"sv) {
 | 
						|
                cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::ListStyleType, CSS::CSSKeywordValue::create(CSS::Keyword::UpperRoman));
 | 
						|
            } else if (value.equals_ignoring_ascii_case("none"sv)) {
 | 
						|
                cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::ListStyleType, CSS::CSSKeywordValue::create(CSS::Keyword::None));
 | 
						|
            } else if (value.equals_ignoring_ascii_case("disc"sv)) {
 | 
						|
                cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::ListStyleType, CSS::CSSKeywordValue::create(CSS::Keyword::Disc));
 | 
						|
            } else if (value.equals_ignoring_ascii_case("circle"sv)) {
 | 
						|
                cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::ListStyleType, CSS::CSSKeywordValue::create(CSS::Keyword::Circle));
 | 
						|
            } else if (value.equals_ignoring_ascii_case("square"sv)) {
 | 
						|
                cascaded_properties->set_property_from_presentational_hint(CSS::PropertyID::ListStyleType, CSS::CSSKeywordValue::create(CSS::Keyword::Square));
 | 
						|
            }
 | 
						|
        }
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
}
 |