/* * Copyright (c) 2018-2022, Andreas Kling * Copyright (c) 2022, Adam Hodgen * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include namespace Web::HTML { // https://html.spec.whatwg.org/multipage/input.html#attr-input-type #define ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTES \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(hidden, Hidden) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(text, Text) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(search, Search) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(tel, Telephone) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(url, URL) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(email, Email) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(password, Password) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(date, Date) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(month, Month) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(week, Week) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(time, Time) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE("datetime-local", LocalDateAndTime) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(number, Number) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(range, Range) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(color, Color) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(checkbox, Checkbox) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(radio, RadioButton) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(file, FileUpload) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(submit, SubmitButton) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(image, ImageButton) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(reset, ResetButton) \ __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(button, Button) class HTMLInputElement final : public HTMLElement , public FormAssociatedElement , public DOM::EditableTextNodeOwner { WEB_PLATFORM_OBJECT(HTMLInputElement, HTMLElement); JS_DECLARE_ALLOCATOR(HTMLInputElement); FORM_ASSOCIATED_ELEMENT(HTMLElement, HTMLInputElement) public: virtual ~HTMLInputElement() override; virtual JS::GCPtr create_layout_node(NonnullRefPtr) override; enum class TypeAttributeState { #define __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE(_, state) state, ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTES #undef __ENUMERATE_HTML_INPUT_TYPE_ATTRIBUTE }; StringView type() const; TypeAttributeState type_state() const { return m_type; } WebIDL::ExceptionOr set_type(String const&); DeprecatedString default_value() const { return deprecated_attribute(HTML::AttributeNames::value); } DeprecatedString name() const { return deprecated_attribute(HTML::AttributeNames::name); } virtual String value() const override; WebIDL::ExceptionOr set_value(String const&); void commit_pending_changes(); Optional placeholder_value() const; bool checked() const { return m_checked; } enum class ChangeSource { Programmatic, User, }; void set_checked(bool, ChangeSource = ChangeSource::Programmatic); bool checked_binding() const { return checked(); } void set_checked_binding(bool); bool indeterminate() const { return m_indeterminate; } void set_indeterminate(bool); bool is_mutable() const { return m_is_mutable; } void did_pick_color(Optional picked_color); JS::GCPtr files(); void set_files(JS::GCPtr); // NOTE: User interaction // https://html.spec.whatwg.org/multipage/input.html#update-the-file-selection void update_the_file_selection(JS::NonnullGCPtr); unsigned size() const; WebIDL::ExceptionOr set_size(unsigned value); WebIDL::ExceptionOr value_as_number() const; WebIDL::ExceptionOr set_value_as_number(double value); WebIDL::ExceptionOr step_up(long n = 1); WebIDL::ExceptionOr step_down(long n = 1); WebIDL::ExceptionOr check_validity(); WebIDL::ExceptionOr report_validity(); void set_custom_validity(String const&); WebIDL::ExceptionOr select(); WebIDL::ExceptionOr set_selection_range(u32 start, u32 end, Optional const& direction = {}); WebIDL::ExceptionOr show_picker(); // ^DOM::EditableTextNodeOwner virtual void did_edit_text_node(Badge) override; // ^EventTarget // https://html.spec.whatwg.org/multipage/interaction.html#the-tabindex-attribute:the-input-element virtual bool is_focusable() const override { return m_type != TypeAttributeState::Hidden; } // ^HTMLElement virtual void attribute_changed(FlyString const&, Optional const&) override; // ^FormAssociatedElement // https://html.spec.whatwg.org/multipage/forms.html#category-listed virtual bool is_listed() const override { return true; } // https://html.spec.whatwg.org/multipage/forms.html#category-submit virtual bool is_submittable() const override { return true; } // https://html.spec.whatwg.org/multipage/forms.html#category-reset virtual bool is_resettable() const override { return true; } // https://html.spec.whatwg.org/multipage/forms.html#category-autocapitalize virtual bool is_auto_capitalize_inheriting() const override { return true; } // https://html.spec.whatwg.org/multipage/forms.html#concept-button virtual bool is_button() const override; // https://html.spec.whatwg.org/multipage/forms.html#concept-submit-button virtual bool is_submit_button() const override; virtual void reset_algorithm() override; virtual void form_associated_element_was_inserted() override; virtual void form_associated_element_was_removed(DOM::Node*) override; // ^HTMLElement // https://html.spec.whatwg.org/multipage/forms.html#category-label virtual bool is_labelable() const override { return type_state() != TypeAttributeState::Hidden; } virtual Optional default_role() const override; JS::GCPtr placeholder_element() { return m_placeholder_element; } JS::GCPtr placeholder_element() const { return m_placeholder_element; } virtual bool has_activation_behavior() const override; virtual void activation_behavior(DOM::Event const&) override; bool has_input_activation_behavior() const; bool change_event_applies() const; bool value_as_number_applies() const; bool step_applies() const; bool step_up_or_down_applies() const; private: HTMLInputElement(DOM::Document&, DOM::QualifiedName); // ^DOM::Node virtual bool is_html_input_element() const final { return true; } // ^DOM::EventTarget virtual void did_lose_focus() override; virtual void did_receive_focus() override; virtual void legacy_pre_activation_behavior() override; virtual void legacy_cancelled_activation_behavior() override; virtual void legacy_cancelled_activation_behavior_was_not_called() override; // ^DOM::Element virtual i32 default_tab_index_value() const override; virtual void initialize(JS::Realm&) override; virtual void visit_edges(Cell::Visitor&) override; Optional convert_string_to_number(StringView input) const; String covert_number_to_string(double input) const; Optional min() const; Optional max() const; double default_step() const; double step_scale_factor() const; Optional allowed_value_step() const; double step_base() const; WebIDL::ExceptionOr step_up_or_down(bool is_down, long n); static TypeAttributeState parse_type_attribute(StringView); void create_shadow_tree_if_needed(); void create_text_input_shadow_tree(); void create_color_input_shadow_tree(); WebIDL::ExceptionOr run_input_activation_behavior(); void set_checked_within_group(); void handle_readonly_attribute(Optional const& value); // https://html.spec.whatwg.org/multipage/input.html#value-sanitization-algorithm String value_sanitization_algorithm(String const&) const; void update_placeholder_visibility(); JS::GCPtr m_placeholder_element; JS::GCPtr m_placeholder_text_node; JS::GCPtr m_inner_text_element; JS::GCPtr m_color_well_element; JS::GCPtr m_text_node; bool m_checked { false }; // https://html.spec.whatwg.org/multipage/input.html#dom-input-indeterminate bool m_indeterminate { false }; // https://html.spec.whatwg.org/multipage/input.html#concept-input-checked-dirty-flag bool m_dirty_checkedness { false }; // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#concept-fe-dirty bool m_dirty_value { false }; // https://html.spec.whatwg.org/multipage/input.html#the-input-element:concept-fe-mutable bool m_is_mutable { true }; // https://html.spec.whatwg.org/multipage/input.html#the-input-element:legacy-pre-activation-behavior bool m_before_legacy_pre_activation_behavior_checked { false }; bool m_before_legacy_pre_activation_behavior_indeterminate { false }; JS::GCPtr m_legacy_pre_activation_behavior_checked_element_in_group; // https://html.spec.whatwg.org/multipage/input.html#concept-input-type-file-selected JS::GCPtr m_selected_files; TypeAttributeState m_type { TypeAttributeState::Text }; String m_value; bool m_has_uncommitted_changes { false }; }; } namespace Web::DOM { template<> inline bool Node::fast_is() const { return is_html_input_element(); } }