mirror of
https://github.com/RGBCube/serenity
synced 2025-10-23 19:32:31 +00:00

This allows us to improve the const-correctness in RadioNodeList, which
has been made possible as of: 5f0ccfb499
now that a GC-visit accepts a
const GC pointer.
105 lines
4 KiB
C++
105 lines
4 KiB
C++
/*
|
|
* Copyright (c) 2023, Shannon Booth <shannon@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <LibWeb/Bindings/Intrinsics.h>
|
|
#include <LibWeb/Bindings/RadioNodeListPrototype.h>
|
|
#include <LibWeb/DOM/Element.h>
|
|
#include <LibWeb/DOM/RadioNodeList.h>
|
|
#include <LibWeb/HTML/HTMLInputElement.h>
|
|
|
|
namespace Web::DOM {
|
|
|
|
JS_DEFINE_ALLOCATOR(RadioNodeList);
|
|
|
|
JS::NonnullGCPtr<RadioNodeList> RadioNodeList::create(JS::Realm& realm, Node const& root, Scope scope, Function<bool(Node const&)> filter)
|
|
{
|
|
return realm.heap().allocate<RadioNodeList>(realm, realm, root, scope, move(filter));
|
|
}
|
|
|
|
RadioNodeList::RadioNodeList(JS::Realm& realm, Node const& root, Scope scope, Function<bool(Node const&)> filter)
|
|
: LiveNodeList(realm, root, scope, move(filter))
|
|
{
|
|
}
|
|
|
|
RadioNodeList::~RadioNodeList() = default;
|
|
|
|
void RadioNodeList::initialize(JS::Realm& realm)
|
|
{
|
|
Base::initialize(realm);
|
|
set_prototype(&Bindings::ensure_web_prototype<Bindings::RadioNodeListPrototype>(realm, "RadioNodeList"_fly_string));
|
|
}
|
|
|
|
static HTML::HTMLInputElement const* radio_button(Node const& node)
|
|
{
|
|
if (!is<HTML::HTMLInputElement>(node))
|
|
return nullptr;
|
|
|
|
auto const& input_element = verify_cast<HTML::HTMLInputElement>(node);
|
|
if (input_element.type_state() != HTML::HTMLInputElement::TypeAttributeState::RadioButton)
|
|
return nullptr;
|
|
|
|
return &input_element;
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/common-dom-interfaces.html#dom-radionodelist-value
|
|
FlyString RadioNodeList::value() const
|
|
{
|
|
// 1. Let element be the first element in tree order represented by the RadioNodeList object that is an input element whose type
|
|
// attribute is in the Radio Button state and whose checkedness is true. Otherwise, let it be null.
|
|
auto* element = verify_cast<HTML::HTMLInputElement>(first_matching([](Node const& node) -> bool {
|
|
auto const* button = radio_button(node);
|
|
if (!button)
|
|
return false;
|
|
|
|
return button->checked();
|
|
}));
|
|
|
|
// 2. If element is null, return the empty string.
|
|
if (!element)
|
|
return String {};
|
|
|
|
// 3. If element is an element with no value attribute, return the string "on".
|
|
// 4. Otherwise, return the value of element's value attribute.
|
|
return element->get_attribute(HTML::AttributeNames::value).value_or("on"_string);
|
|
}
|
|
|
|
void RadioNodeList::set_value(FlyString const& value)
|
|
{
|
|
HTML::HTMLInputElement* element = nullptr;
|
|
|
|
// 1. If the new value is the string "on": let element be the first element in tree order represented by the RadioNodeList object
|
|
// that is an input element whose type attribute is in the Radio Button state and whose value content attribute is either absent,
|
|
// or present and equal to the new value, if any. If no such element exists, then instead let element be null.
|
|
if (value == "on"sv) {
|
|
element = verify_cast<HTML::HTMLInputElement>(first_matching([&value](auto const& node) {
|
|
auto const* button = radio_button(node);
|
|
if (!button)
|
|
return false;
|
|
|
|
auto const maybe_value = button->get_attribute(HTML::AttributeNames::value);
|
|
return !maybe_value.has_value() || maybe_value.value() == value;
|
|
}));
|
|
}
|
|
// 2. Otherwise: let element be the first element in tree order represented by the RadioNodeList object that is an input element whose
|
|
// type attribute is in the Radio Button state and whose value content attribute is present and equal to the new value, if any. If
|
|
// no such element exists, then instead let element be null.
|
|
else {
|
|
element = verify_cast<HTML::HTMLInputElement>(first_matching([&value](auto const& node) {
|
|
auto const* button = radio_button(node);
|
|
if (!button)
|
|
return false;
|
|
|
|
auto const maybe_value = button->get_attribute(HTML::AttributeNames::value);
|
|
return maybe_value.has_value() && maybe_value.value() == value;
|
|
}));
|
|
}
|
|
|
|
// 3. If element is not null, then set its checkedness to true.
|
|
if (element)
|
|
element->set_checked(true);
|
|
}
|
|
|
|
}
|