mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 02:02:45 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			85 lines
		
	
	
	
		
			3.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			85 lines
		
	
	
	
		
			3.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2023, Shannon Booth <shannon@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <LibWeb/Bindings/Intrinsics.h>
 | |
| #include <LibWeb/DOM/Element.h>
 | |
| #include <LibWeb/DOM/HTMLCollection.h>
 | |
| #include <LibWeb/DOM/HTMLFormControlsCollection.h>
 | |
| #include <LibWeb/DOM/ParentNode.h>
 | |
| 
 | |
| namespace Web::DOM {
 | |
| 
 | |
| JS_DEFINE_ALLOCATOR(HTMLFormControlsCollection);
 | |
| 
 | |
| JS::NonnullGCPtr<HTMLFormControlsCollection> HTMLFormControlsCollection::create(ParentNode& root, Scope scope, Function<bool(Element const&)> filter)
 | |
| {
 | |
|     return root.heap().allocate<HTMLFormControlsCollection>(root.realm(), root, scope, move(filter));
 | |
| }
 | |
| 
 | |
| HTMLFormControlsCollection::HTMLFormControlsCollection(ParentNode& root, Scope scope, Function<bool(Element const&)> filter)
 | |
|     : HTMLCollection(root, scope, move(filter))
 | |
| {
 | |
| }
 | |
| 
 | |
| HTMLFormControlsCollection::~HTMLFormControlsCollection() = default;
 | |
| 
 | |
| void HTMLFormControlsCollection::initialize(JS::Realm& realm)
 | |
| {
 | |
|     Base::initialize(realm);
 | |
|     set_prototype(&Bindings::ensure_web_prototype<Bindings::HTMLFormControlsCollectionPrototype>(realm, "HTMLFormControlsCollection"_fly_string));
 | |
| }
 | |
| 
 | |
| // https://html.spec.whatwg.org/multipage/common-dom-interfaces.html#dom-htmlformcontrolscollection-nameditem
 | |
| Variant<Empty, Element*, JS::Handle<RadioNodeList>> HTMLFormControlsCollection::named_item_or_radio_node_list(FlyString const& name) const
 | |
| {
 | |
|     // 1. If name is the empty string, return null and stop the algorithm.
 | |
|     if (name.is_empty())
 | |
|         return {};
 | |
| 
 | |
|     // 2. If, at the time the method is called, there is exactly one node in the collection that has either an id attribute or a name attribute equal to name, then return that node and stop the algorithm.
 | |
|     // 3. Otherwise, if there are no nodes in the collection that have either an id attribute or a name attribute equal to name, then return null and stop the algorithm.
 | |
|     Element* matching_element = nullptr;
 | |
|     bool multiple_matching = false;
 | |
| 
 | |
|     auto collection = collect_matching_elements();
 | |
|     for (auto const& element : collection) {
 | |
|         if (element->id() != name && element->name() != name)
 | |
|             continue;
 | |
| 
 | |
|         if (matching_element) {
 | |
|             multiple_matching = true;
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         matching_element = element;
 | |
|     }
 | |
| 
 | |
|     if (!matching_element)
 | |
|         return {};
 | |
| 
 | |
|     if (!multiple_matching)
 | |
|         return matching_element;
 | |
| 
 | |
|     // 4. Otherwise, create a new RadioNodeList object representing a live view of the HTMLFormControlsCollection object, further filtered so that the only nodes in the
 | |
|     //    RadioNodeList object are those that have either an id attribute or a name attribute equal to name. The nodes in the RadioNodeList object must be sorted in tree
 | |
|     //    order. Return that RadioNodeList object.
 | |
|     return JS::make_handle(RadioNodeList::create(realm(), root(), LiveNodeList::Scope::Descendants, [name](Node const& node) {
 | |
|         if (!is<Element>(node))
 | |
|             return false;
 | |
| 
 | |
|         auto const& element = verify_cast<Element>(node);
 | |
|         return element.id() == name || element.name() == name;
 | |
|     }));
 | |
| }
 | |
| 
 | |
| WebIDL::ExceptionOr<JS::Value> HTMLFormControlsCollection::named_item_value(FlyString const& name) const
 | |
| {
 | |
|     return named_item_or_radio_node_list(name).visit(
 | |
|         [](Empty) -> JS::Value { return JS::js_undefined(); },
 | |
|         [](auto const& value) -> JS::Value { return value; });
 | |
| }
 | |
| 
 | |
| }
 | 
