mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-21 23:50:06 +00:00 
			
		
		
		
	
		
			Some checks are pending
		
		
	
	CI / macOS, arm64, Sanitizer, Clang (push) Waiting to run
				
			CI / Linux, x86_64, Fuzzers, Clang (push) Waiting to run
				
			CI / Linux, x86_64, Sanitizer, GNU (push) Waiting to run
				
			CI / Linux, x86_64, Sanitizer, Clang (push) Waiting to run
				
			Package the js repl as a binary artifact / Linux, arm64 (push) Waiting to run
				
			Package the js repl as a binary artifact / macOS, arm64 (push) Waiting to run
				
			Package the js repl as a binary artifact / Linux, x86_64 (push) Waiting to run
				
			Run test262 and test-wasm / run_and_update_results (push) Waiting to run
				
			Lint Code / lint (push) Waiting to run
				
			Label PRs with merge conflicts / auto-labeler (push) Waiting to run
				
			Push notes / build (push) Waiting to run
				
			This porting effort makes it pretty clear we will want a UTF-16-aware GenericLexer. But for now, we can actually make ASCII assumptions about what we are parsing, and act accordingly.
		
			
				
	
	
		
			268 lines
		
	
	
	
		
			16 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
	
		
			16 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2021, Andreas Kling <andreas@ladybird.org>
 | |
|  * Copyright (c) 2024, Jelle Raaijmakers <jelle@ladybird.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/String.h>
 | |
| #include <AK/WeakPtr.h>
 | |
| #include <LibWeb/Bindings/HTMLFormElementPrototype.h>
 | |
| #include <LibWeb/DOM/InputEventsTarget.h>
 | |
| #include <LibWeb/Forward.h>
 | |
| #include <LibWeb/WebIDL/Types.h>
 | |
| 
 | |
| namespace Web::HTML {
 | |
| 
 | |
| // Form-associated elements should invoke this macro to inject overridden FormAssociatedElement and HTMLElement
 | |
| // methods as needed. If your class wished to override an HTMLElement method that is overridden here, use the
 | |
| // following methods instead:
 | |
| //
 | |
| //    HTMLElement::inserted() -> Use form_associated_element_was_inserted()
 | |
| //    HTMLElement::removed_from() -> Use form_associated_element_was_removed()
 | |
| //
 | |
| #define FORM_ASSOCIATED_ELEMENT(ElementBaseClass, ElementClass)                                                                                                             \
 | |
| private:                                                                                                                                                                    \
 | |
|     virtual HTMLElement& form_associated_element_to_html_element() override                                                                                                 \
 | |
|     {                                                                                                                                                                       \
 | |
|         static_assert(IsBaseOf<HTMLElement, ElementClass>);                                                                                                                 \
 | |
|         return *this;                                                                                                                                                       \
 | |
|     }                                                                                                                                                                       \
 | |
|                                                                                                                                                                             \
 | |
|     virtual void inserted() override                                                                                                                                        \
 | |
|     {                                                                                                                                                                       \
 | |
|         ElementBaseClass::inserted();                                                                                                                                       \
 | |
|         form_node_was_inserted();                                                                                                                                           \
 | |
|         form_associated_element_was_inserted();                                                                                                                             \
 | |
|     }                                                                                                                                                                       \
 | |
|                                                                                                                                                                             \
 | |
|     virtual void removed_from(DOM::Node* old_parent, DOM::Node& old_root) override                                                                                          \
 | |
|     {                                                                                                                                                                       \
 | |
|         ElementBaseClass::removed_from(old_parent, old_root);                                                                                                               \
 | |
|         form_node_was_removed();                                                                                                                                            \
 | |
|         form_associated_element_was_removed(old_parent);                                                                                                                    \
 | |
|     }                                                                                                                                                                       \
 | |
|                                                                                                                                                                             \
 | |
|     virtual void moved_from(GC::Ptr<DOM::Node> old_parent) override                                                                                                         \
 | |
|     {                                                                                                                                                                       \
 | |
|         ElementBaseClass::moved_from(old_parent);                                                                                                                           \
 | |
|         form_node_was_moved();                                                                                                                                              \
 | |
|         form_associated_element_was_moved(old_parent);                                                                                                                      \
 | |
|     }                                                                                                                                                                       \
 | |
|                                                                                                                                                                             \
 | |
|     virtual void attribute_changed(FlyString const& name, Optional<String> const& old_value, Optional<String> const& value, Optional<FlyString> const& namespace_) override \
 | |
|     {                                                                                                                                                                       \
 | |
|         ElementBaseClass::attribute_changed(name, old_value, value, namespace_);                                                                                            \
 | |
|         form_node_attribute_changed(name, value);                                                                                                                           \
 | |
|         form_associated_element_attribute_changed(name, old_value, value, namespace_);                                                                                      \
 | |
|     }
 | |
| 
 | |
| // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#selection-direction
 | |
| enum class SelectionDirection {
 | |
|     Forward,
 | |
|     Backward,
 | |
|     None,
 | |
| };
 | |
| 
 | |
| class FormAssociatedElement {
 | |
| public:
 | |
|     HTMLFormElement* form() { return m_form; }
 | |
|     HTMLFormElement const* form() const { return m_form; }
 | |
| 
 | |
|     void set_form(HTMLFormElement*);
 | |
| 
 | |
|     void element_id_changed(Badge<DOM::Document>);
 | |
|     void element_with_id_was_added_or_removed(Badge<DOM::Document>);
 | |
| 
 | |
|     bool enabled() const;
 | |
| 
 | |
|     void set_parser_inserted(Badge<HTMLParser>);
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/forms.html#category-listed
 | |
|     virtual bool is_listed() const { return false; }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/forms.html#category-submit
 | |
|     virtual bool is_submittable() const { return false; }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/forms.html#category-reset
 | |
|     virtual bool is_resettable() const { return false; }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/forms.html#category-autocapitalize
 | |
|     virtual bool is_auto_capitalize_inheriting() const { return false; }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/forms.html#concept-button
 | |
|     virtual bool is_button() const { return false; }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/forms.html#concept-submit-button
 | |
|     virtual bool is_submit_button() const { return false; }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#check-validity-steps
 | |
|     bool check_validity_steps();
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#report-validity-steps
 | |
|     bool report_validity_steps();
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#candidate-for-constraint-validation
 | |
|     bool is_candidate_for_constraint_validation() const;
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#concept-fv-valid
 | |
|     bool satisfies_its_constraints() const;
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#concept-fs-novalidate
 | |
|     bool novalidate_state() const;
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure/#definitions
 | |
|     virtual bool suffering_from_being_missing() const { return false; }
 | |
|     virtual bool suffering_from_a_type_mismatch() const { return false; }
 | |
|     virtual bool suffering_from_a_pattern_mismatch() const { return false; }
 | |
|     bool suffering_from_being_too_long() const;
 | |
|     bool suffering_from_being_too_short() const;
 | |
|     virtual bool suffering_from_an_underflow() const { return false; }
 | |
|     virtual bool suffering_from_an_overflow() const { return false; }
 | |
|     virtual bool suffering_from_a_step_mismatch() const { return false; }
 | |
|     virtual bool suffering_from_bad_input() const { return false; }
 | |
|     bool suffering_from_a_custom_error() const;
 | |
| 
 | |
|     virtual Utf16String value() const { return {}; }
 | |
|     virtual Optional<String> optional_value() const { VERIFY_NOT_REACHED(); }
 | |
| 
 | |
|     virtual HTMLElement& form_associated_element_to_html_element() = 0;
 | |
|     HTMLElement const& form_associated_element_to_html_element() const { return const_cast<FormAssociatedElement&>(*this).form_associated_element_to_html_element(); }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#concept-form-reset-control
 | |
|     virtual void reset_algorithm() { }
 | |
| 
 | |
|     virtual void clear_algorithm();
 | |
| 
 | |
|     String form_action() const;
 | |
|     WebIDL::ExceptionOr<void> set_form_action(String const&);
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#dom-cva-validity
 | |
|     GC::Ref<ValidityState const> validity() const;
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#dom-cva-setcustomvalidity
 | |
|     void set_custom_validity(String& error);
 | |
| 
 | |
| protected:
 | |
|     FormAssociatedElement() = default;
 | |
|     virtual ~FormAssociatedElement() = default;
 | |
| 
 | |
|     virtual void form_associated_element_was_inserted() { }
 | |
|     virtual void form_associated_element_was_removed(DOM::Node*) { }
 | |
|     virtual void form_associated_element_was_moved(GC::Ptr<DOM::Node>) { }
 | |
|     virtual void form_associated_element_attribute_changed(FlyString const&, Optional<String> const&, Optional<String> const&, Optional<FlyString> const&) { }
 | |
| 
 | |
|     void form_node_was_inserted();
 | |
|     void form_node_was_removed();
 | |
|     void form_node_was_moved();
 | |
|     void form_node_attribute_changed(FlyString const&, Optional<String> const&);
 | |
| 
 | |
| private:
 | |
|     void reset_form_owner();
 | |
| 
 | |
|     WeakPtr<HTMLFormElement> m_form;
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#parser-inserted-flag
 | |
|     bool m_parser_inserted { false };
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#custom-validity-error-message
 | |
|     String m_custom_validity_error_message;
 | |
| };
 | |
| 
 | |
| enum class SelectionSource {
 | |
|     UI,
 | |
|     DOM,
 | |
| };
 | |
| 
 | |
| class FormAssociatedTextControlElement
 | |
|     : public FormAssociatedElement
 | |
|     , public InputEventsTarget {
 | |
| public:
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#concept-textarea/input-relevant-value
 | |
|     virtual Utf16String relevant_value() = 0;
 | |
|     virtual WebIDL::ExceptionOr<void> set_relevant_value(Utf16String const&) = 0;
 | |
| 
 | |
|     virtual void set_dirty_value_flag(bool flag) = 0;
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#dom-textarea/input-select
 | |
|     WebIDL::ExceptionOr<void> select();
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#dom-textarea/input-selectionstart
 | |
|     Optional<WebIDL::UnsignedLong> selection_start_binding() const;
 | |
|     WebIDL::ExceptionOr<void> set_selection_start_binding(Optional<WebIDL::UnsignedLong> const&);
 | |
|     WebIDL::UnsignedLong selection_start() const;
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#dom-textarea/input-selectionend
 | |
|     Optional<WebIDL::UnsignedLong> selection_end_binding() const;
 | |
|     WebIDL::ExceptionOr<void> set_selection_end_binding(Optional<WebIDL::UnsignedLong> const&);
 | |
|     WebIDL::UnsignedLong selection_end() const;
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#dom-textarea/input-selectiondirection
 | |
|     Optional<String> selection_direction() const;
 | |
|     void set_selection_direction(Optional<String> direction);
 | |
|     WebIDL::ExceptionOr<void> set_selection_direction_binding(Optional<String> direction);
 | |
|     SelectionDirection selection_direction_state() const { return m_selection_direction; }
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#dom-textarea/input-setrangetext
 | |
|     WebIDL::ExceptionOr<void> set_range_text_binding(Utf16String const& replacement);
 | |
|     WebIDL::ExceptionOr<void> set_range_text_binding(Utf16String const& replacement, WebIDL::UnsignedLong start, WebIDL::UnsignedLong end, Bindings::SelectionMode = Bindings::SelectionMode::Preserve);
 | |
|     WebIDL::ExceptionOr<void> set_range_text(Utf16String const& replacement, WebIDL::UnsignedLong start, WebIDL::UnsignedLong end, Bindings::SelectionMode = Bindings::SelectionMode::Preserve);
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#dom-textarea/input-setselectionrange
 | |
|     void set_the_selection_range(Optional<WebIDL::UnsignedLong> start, Optional<WebIDL::UnsignedLong> end, SelectionDirection direction = SelectionDirection::None, SelectionSource source = SelectionSource::DOM);
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#dom-textarea/input-setselectionrange
 | |
|     WebIDL::ExceptionOr<void> set_selection_range(Optional<WebIDL::UnsignedLong> start, Optional<WebIDL::UnsignedLong> end, Optional<String> direction);
 | |
| 
 | |
|     // https://w3c.github.io/selection-api/#dfn-has-scheduled-selectionchange-event
 | |
|     bool has_scheduled_selectionchange_event() const { return m_has_scheduled_selectionchange_event; }
 | |
|     void set_scheduled_selectionchange_event(bool value) { m_has_scheduled_selectionchange_event = value; }
 | |
| 
 | |
|     bool is_mutable() const { return m_is_mutable; }
 | |
|     void set_is_mutable(bool is_mutable) { m_is_mutable = is_mutable; }
 | |
| 
 | |
|     virtual void did_edit_text_node() = 0;
 | |
| 
 | |
|     virtual GC::Ptr<DOM::Text> form_associated_element_to_text_node() = 0;
 | |
|     virtual GC::Ptr<DOM::Text const> form_associated_element_to_text_node() const { return const_cast<FormAssociatedTextControlElement&>(*this).form_associated_element_to_text_node(); }
 | |
| 
 | |
|     virtual void handle_insert(Utf16String const&) override;
 | |
|     virtual void handle_delete(DeleteDirection) override;
 | |
|     virtual EventResult handle_return_key(FlyString const& ui_input_type) override;
 | |
|     virtual void select_all() override;
 | |
|     virtual void set_selection_anchor(GC::Ref<DOM::Node>, size_t offset) override;
 | |
|     virtual void set_selection_focus(GC::Ref<DOM::Node>, size_t offset) override;
 | |
|     virtual void move_cursor_to_start(CollapseSelection) override;
 | |
|     virtual void move_cursor_to_end(CollapseSelection) override;
 | |
|     virtual void increment_cursor_position_offset(CollapseSelection) override;
 | |
|     virtual void decrement_cursor_position_offset(CollapseSelection) override;
 | |
|     virtual void increment_cursor_position_to_next_word(CollapseSelection) override;
 | |
|     virtual void decrement_cursor_position_to_previous_word(CollapseSelection) override;
 | |
| 
 | |
|     GC::Ptr<DOM::Position> cursor_position() const;
 | |
| 
 | |
| protected:
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#concept-textarea/input-relevant-value
 | |
|     void relevant_value_was_changed();
 | |
| 
 | |
| private:
 | |
|     virtual GC::Ref<JS::Cell> as_cell() override;
 | |
| 
 | |
|     void collapse_selection_to_offset(size_t);
 | |
|     void selection_was_changed();
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#concept-textarea/input-selection
 | |
|     WebIDL::UnsignedLong m_selection_start { 0 };
 | |
|     WebIDL::UnsignedLong m_selection_end { 0 };
 | |
|     SelectionDirection m_selection_direction { SelectionDirection::None };
 | |
| 
 | |
|     // https://w3c.github.io/selection-api/#dfn-has-scheduled-selectionchange-event
 | |
|     bool m_has_scheduled_selectionchange_event { false };
 | |
| 
 | |
|     // https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#mutability
 | |
|     bool m_is_mutable { true };
 | |
| };
 | |
| 
 | |
| }
 |