/* * Copyright (c) 2021, Tim Flynn * Copyright (c) 2022, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include namespace { // https://infra.spec.whatwg.org/#set-append inline void append_to_ordered_set(Vector& set, String item) { if (!set.contains_slow(item)) set.append(move(item)); } // https://infra.spec.whatwg.org/#list-remove inline void remove_from_ordered_set(Vector& set, StringView item) { set.remove_first_matching([&](auto const& value) { return value == item; }); } // https://infra.spec.whatwg.org/#set-replace inline void replace_in_ordered_set(Vector& set, String const& item, String replacement) { auto item_index = set.find_first_index(item); VERIFY(item_index.has_value()); auto replacement_index = set.find_first_index(replacement); if (!replacement_index.has_value()) { set[*item_index] = move(replacement); return; } auto index_to_set = min(*item_index, *replacement_index); auto index_to_remove = max(*item_index, *replacement_index); if (index_to_set == index_to_remove) return; set[index_to_set] = move(replacement); set.remove(index_to_remove); } } namespace Web::DOM { JS_DEFINE_ALLOCATOR(DOMTokenList); JS::NonnullGCPtr DOMTokenList::create(Element& associated_element, FlyString associated_attribute) { auto& realm = associated_element.realm(); return realm.heap().allocate(realm, associated_element, move(associated_attribute)); } // https://dom.spec.whatwg.org/#ref-for-domtokenlist%E2%91%A0%E2%91%A2 DOMTokenList::DOMTokenList(Element& associated_element, FlyString associated_attribute) : Bindings::PlatformObject(associated_element.realm()) , m_associated_element(associated_element) , m_associated_attribute(move(associated_attribute)) { m_legacy_platform_object_flags = LegacyPlatformObjectFlags { .supports_indexed_properties = 1 }; associated_attribute_changed(associated_element.get_attribute_value(m_associated_attribute)); } void DOMTokenList::initialize(JS::Realm& realm) { Base::initialize(realm); WEB_SET_PROTOTYPE_FOR_INTERFACE(DOMTokenList); } void DOMTokenList::visit_edges(Cell::Visitor& visitor) { Base::visit_edges(visitor); visitor.visit(m_associated_element); } // https://dom.spec.whatwg.org/#ref-for-domtokenlist%E2%91%A0%E2%91%A1 void DOMTokenList::associated_attribute_changed(StringView value) { m_token_set.clear(); if (value.is_empty()) return; auto split_values = value.split_view_if(Infra::is_ascii_whitespace); for (auto const& split_value : split_values) append_to_ordered_set(m_token_set, String::from_utf8(split_value).release_value_but_fixme_should_propagate_errors()); } // https://dom.spec.whatwg.org/#ref-for-dfn-supported-property-indices%E2%91%A3 bool DOMTokenList::is_supported_property_index(u32 index) const { return index < m_token_set.size(); } // https://dom.spec.whatwg.org/#dom-domtokenlist-item Optional DOMTokenList::item(size_t index) const { // 1. If index is equal to or greater than this’s token set’s size, then return null. if (index >= m_token_set.size()) return {}; // 2. Return this’s token set[index]. return m_token_set[index]; } // https://dom.spec.whatwg.org/#dom-domtokenlist-contains bool DOMTokenList::contains(String const& token) { return m_token_set.contains_slow(token); } // https://dom.spec.whatwg.org/#dom-domtokenlist-add WebIDL::ExceptionOr DOMTokenList::add(Vector const& tokens) { // 1. For each token in tokens: for (auto const& token : tokens) { // a. If token is the empty string, then throw a "SyntaxError" DOMException. // b. If token contains any ASCII whitespace, then throw an "InvalidCharacterError" DOMException. TRY(validate_token(token)); // 2. For each token in tokens, append token to this’s token set. append_to_ordered_set(m_token_set, token); } // 3. Run the update steps. run_update_steps(); return {}; } // https://dom.spec.whatwg.org/#dom-domtokenlist-remove WebIDL::ExceptionOr DOMTokenList::remove(Vector const& tokens) { // 1. For each token in tokens: for (auto const& token : tokens) { // a. If token is the empty string, then throw a "SyntaxError" DOMException. // b. If token contains any ASCII whitespace, then throw an "InvalidCharacterError" DOMException. TRY(validate_token(token)); // 2. For each token in tokens, remove token from this’s token set. remove_from_ordered_set(m_token_set, token); } // 3. Run the update steps. run_update_steps(); return {}; } // https://dom.spec.whatwg.org/#dom-domtokenlist-toggle WebIDL::ExceptionOr DOMTokenList::toggle(String const& token, Optional force) { // 1. If token is the empty string, then throw a "SyntaxError" DOMException. // 2. If token contains any ASCII whitespace, then throw an "InvalidCharacterError" DOMException. TRY(validate_token(token)); // 3. If this’s token set[token] exists, then: if (contains(token)) { // a. If force is either not given or is false, then remove token from this’s token set, run the update steps and return false. if (!force.has_value() || !force.value()) { remove_from_ordered_set(m_token_set, token); run_update_steps(); return false; } // b. Return true. return true; } // 4. Otherwise, if force not given or is true, append token to this’s token set, run the update steps, and return true. if (!force.has_value() || force.value()) { append_to_ordered_set(m_token_set, token); run_update_steps(); return true; } // 5. Return false. return false; } // https://dom.spec.whatwg.org/#dom-domtokenlist-replace WebIDL::ExceptionOr DOMTokenList::replace(String const& token, String const& new_token) { // 1. If either token or newToken is the empty string, then throw a "SyntaxError" DOMException. // 2. If either token or newToken contains any ASCII whitespace, then throw an "InvalidCharacterError" DOMException. TRY(validate_token(token)); TRY(validate_token(new_token)); // 3. If this’s token set does not contain token, then return false. if (!contains(token)) return false; // 4. Replace token in this’s token set with newToken. replace_in_ordered_set(m_token_set, token, new_token); // 5. Run the update steps. run_update_steps(); // 6. Return true. return true; } // https://dom.spec.whatwg.org/#dom-domtokenlist-supports // https://dom.spec.whatwg.org/#concept-domtokenlist-validation WebIDL::ExceptionOr DOMTokenList::supports(StringView token) { static HashMap> supported_tokens_map = { // NOTE: The supported values for rel were taken from HTMLLinkElement::Relationship { HTML::AttributeNames::rel, { "alternate"sv, "stylesheet"sv, "preload"sv, "dns-prefetch"sv, "preconnect"sv, "icon"sv } }, }; // 1. If the associated attribute’s local name does not define supported tokens, throw a TypeError. auto supported_tokens = supported_tokens_map.get(m_associated_attribute); if (!supported_tokens.has_value()) return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, MUST(String::formatted("Attribute {} does not define any supported tokens", m_associated_attribute)) }; // AD-HOC: Other browsers return false for rel attributes on non-link elements for all attribute values we currently support. if (m_associated_attribute == HTML::AttributeNames::rel && !is(*m_associated_element)) return false; // 2. Let lowercase token be a copy of token, in ASCII lowercase. auto lowercase_token = token.to_lowercase_string(); // 3. If lowercase token is present in supported tokens, return true. if (supported_tokens->contains_slow(lowercase_token)) return true; // 4. Return false. return false; } // https://dom.spec.whatwg.org/#dom-domtokenlist-value String DOMTokenList::value() const { StringBuilder builder; builder.join(' ', m_token_set); return MUST(builder.to_string()); } // https://dom.spec.whatwg.org/#ref-for-concept-element-attributes-set-value%E2%91%A2 void DOMTokenList::set_value(String const& value) { JS::GCPtr associated_element = m_associated_element.ptr(); if (!associated_element) return; MUST(associated_element->set_attribute(m_associated_attribute, value)); } WebIDL::ExceptionOr DOMTokenList::validate_token(StringView token) const { if (token.is_empty()) return WebIDL::SyntaxError::create(realm(), "Non-empty DOM tokens are not allowed"_fly_string); if (any_of(token, Infra::is_ascii_whitespace)) return WebIDL::InvalidCharacterError::create(realm(), "DOM tokens containing ASCII whitespace are not allowed"_fly_string); return {}; } // https://dom.spec.whatwg.org/#concept-dtl-update void DOMTokenList::run_update_steps() { JS::GCPtr associated_element = m_associated_element.ptr(); if (!associated_element) return; // 1. If the associated element does not have an associated attribute and token set is empty, then return. if (!associated_element->has_attribute(m_associated_attribute) && m_token_set.is_empty()) return; // 2. Set an attribute value for the associated element using associated attribute’s local name and the result of running the ordered set serializer for token set. MUST(associated_element->set_attribute(m_associated_attribute, value())); } WebIDL::ExceptionOr DOMTokenList::item_value(size_t index) const { auto string = item(index); if (!string.has_value()) return JS::js_undefined(); return JS::PrimitiveString::create(vm(), string.release_value()); } }