/* * Copyright (c) 2022, the SerenityOS developers. * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include namespace Web::Bindings { OptionConstructor::OptionConstructor(JS::Realm& realm) : NativeFunction(realm.intrinsics().function_prototype()) { } JS::ThrowCompletionOr OptionConstructor::initialize(JS::Realm& realm) { auto& vm = this->vm(); MUST_OR_THROW_OOM(NativeFunction::initialize(realm)); define_direct_property(vm.names.prototype, &ensure_web_prototype(realm, "HTMLOptionElement"), 0); define_direct_property(vm.names.length, JS::Value(0), JS::Attribute::Configurable); return {}; } JS::ThrowCompletionOr OptionConstructor::call() { return vm().throw_completion(JS::ErrorType::ConstructorWithoutNew, "Option"); } // https://html.spec.whatwg.org/multipage/form-elements.html#dom-option JS::ThrowCompletionOr> OptionConstructor::construct(FunctionObject&) { auto& vm = this->vm(); auto& realm = *vm.current_realm(); // 1. Let document be the current global object's associated Document. auto& window = verify_cast(HTML::current_global_object()); auto& document = window.associated_document(); // 2. Let option be the result of creating an element given document, option, and the HTML namespace. auto element = TRY(Bindings::throw_dom_exception_if_needed(vm, [&]() { return DOM::create_element(document, HTML::TagNames::option, Namespace::HTML); })); JS::NonnullGCPtr option_element = verify_cast(*element); // 3. If text is not the empty string, then append to option a new Text node whose data is text. if (vm.argument_count() > 0) { auto text = TRY(vm.argument(0).to_deprecated_string(vm)); if (!text.is_empty()) { auto new_text_node = MUST_OR_THROW_OOM(vm.heap().allocate(realm, document, text)); MUST(option_element->append_child(*new_text_node)); } } // 4. If value is given, then set an attribute value for option using "value" and value. if (vm.argument_count() > 1) { auto value = TRY(vm.argument(1).to_deprecated_string(vm)); MUST(option_element->set_attribute(HTML::AttributeNames::value, value)); } // 5. If defaultSelected is true, then set an attribute value for option using "selected" and the empty string. if (vm.argument_count() > 2) { auto default_selected = vm.argument(2).to_boolean(); if (default_selected) { MUST(option_element->set_attribute(HTML::AttributeNames::selected, "")); } } // 6. If selected is true, then set option's selectedness to true; otherwise set its selectedness to false (even if defaultSelected is true). option_element->m_selected = vm.argument(3).to_boolean(); // 7. Return option. return option_element; } }