| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  | /*
 | 
					
						
							|  |  |  |  |  * Copyright (c) 2023, Andrew Kaster <akaster@serenityos.org> | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #pragma once
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include <LibJS/Runtime/Promise.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Bindings/NavigationPrototype.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/DOM/EventTarget.h>
 | 
					
						
							| 
									
										
										
										
											2023-08-28 18:03:45 +02:00
										 |  |  |  | #include <LibWeb/HTML/HistoryHandlingBehavior.h>
 | 
					
						
							| 
									
										
										
										
											2023-09-22 18:35:11 -06:00
										 |  |  |  | #include <LibWeb/HTML/Navigable.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/HTML/NavigationType.h>
 | 
					
						
							| 
									
										
										
										
											2023-08-24 16:20:38 -06:00
										 |  |  |  | #include <LibWeb/HTML/StructuredSerialize.h>
 | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | namespace Web::HTML { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigationupdatecurrententryoptions
 | 
					
						
							|  |  |  |  | struct NavigationUpdateCurrentEntryOptions { | 
					
						
							|  |  |  |  |     JS::Value state; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigationoptions
 | 
					
						
							|  |  |  |  | struct NavigationOptions { | 
					
						
							| 
									
										
										
										
											2023-08-24 16:20:38 -06:00
										 |  |  |  |     Optional<JS::Value> info; | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigationnavigateoptions
 | 
					
						
							|  |  |  |  | struct NavigationNavigateOptions : public NavigationOptions { | 
					
						
							| 
									
										
										
										
											2023-08-24 16:20:38 -06:00
										 |  |  |  |     Optional<JS::Value> state; | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  |     Bindings::NavigationHistoryBehavior history = Bindings::NavigationHistoryBehavior::Auto; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigationreloadoptions
 | 
					
						
							|  |  |  |  | struct NavigationReloadOptions : public NavigationOptions { | 
					
						
							| 
									
										
										
										
											2023-08-24 16:20:38 -06:00
										 |  |  |  |     Optional<JS::Value> state; | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigationresult
 | 
					
						
							|  |  |  |  | struct NavigationResult { | 
					
						
							| 
									
										
										
										
											2023-08-24 16:20:38 -06:00
										 |  |  |  |     // FIXME: Are we supposed to return a PromiseCapability (WebIDL::Promise) here?
 | 
					
						
							|  |  |  |  |     JS::NonnullGCPtr<JS::Object> committed; | 
					
						
							|  |  |  |  |     JS::NonnullGCPtr<JS::Object> finished; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigation-api-method-tracker
 | 
					
						
							|  |  |  |  | struct NavigationAPIMethodTracker final : public JS::Cell { | 
					
						
							|  |  |  |  |     JS_CELL(NavigationAPIMethodTracker, JS::Cell); | 
					
						
							| 
									
										
										
										
											2024-04-06 10:16:04 -07:00
										 |  |  |  |     JS_DECLARE_ALLOCATOR(NavigationAPIMethodTracker); | 
					
						
							| 
									
										
										
										
											2023-08-24 16:20:38 -06:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     NavigationAPIMethodTracker(JS::NonnullGCPtr<Navigation> navigation, | 
					
						
							|  |  |  |  |         Optional<String> key, | 
					
						
							|  |  |  |  |         JS::Value info, | 
					
						
							|  |  |  |  |         Optional<SerializationRecord> serialized_state, | 
					
						
							|  |  |  |  |         JS::GCPtr<NavigationHistoryEntry> commited_to_entry, | 
					
						
							|  |  |  |  |         JS::NonnullGCPtr<WebIDL::Promise> committed_promise, | 
					
						
							|  |  |  |  |         JS::NonnullGCPtr<WebIDL::Promise> finished_promise); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     virtual void visit_edges(Cell::Visitor&) override; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     JS::NonnullGCPtr<Navigation> navigation; | 
					
						
							|  |  |  |  |     Optional<String> key; | 
					
						
							|  |  |  |  |     JS::Value info; | 
					
						
							|  |  |  |  |     Optional<SerializationRecord> serialized_state; | 
					
						
							|  |  |  |  |     JS::GCPtr<NavigationHistoryEntry> commited_to_entry; | 
					
						
							|  |  |  |  |     JS::NonnullGCPtr<WebIDL::Promise> committed_promise; | 
					
						
							|  |  |  |  |     JS::NonnullGCPtr<WebIDL::Promise> finished_promise; | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigation-interface
 | 
					
						
							|  |  |  |  | class Navigation : public DOM::EventTarget { | 
					
						
							|  |  |  |  |     WEB_PLATFORM_OBJECT(Navigation, DOM::EventTarget); | 
					
						
							| 
									
										
										
										
											2023-11-19 19:47:52 +01:00
										 |  |  |  |     JS_DECLARE_ALLOCATOR(Navigation); | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | public: | 
					
						
							|  |  |  |  |     [[nodiscard]] static JS::NonnullGCPtr<Navigation> create(JS::Realm&); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // IDL properties and methods
 | 
					
						
							|  |  |  |  |     Vector<JS::NonnullGCPtr<NavigationHistoryEntry>> entries() const; | 
					
						
							|  |  |  |  |     JS::GCPtr<NavigationHistoryEntry> current_entry() const; | 
					
						
							|  |  |  |  |     WebIDL::ExceptionOr<void> update_current_entry(NavigationUpdateCurrentEntryOptions); | 
					
						
							| 
									
										
										
										
											2023-08-25 14:09:14 -06:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // https://html.spec.whatwg.org/multipage/nav-history-apis.html#dom-navigation-transition
 | 
					
						
							|  |  |  |  |     JS::GCPtr<NavigationTransition> transition() const { return m_transition; } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  |     bool can_go_back() const; | 
					
						
							|  |  |  |  |     bool can_go_forward() const; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-24 16:20:38 -06:00
										 |  |  |  |     WebIDL::ExceptionOr<NavigationResult> navigate(String url, NavigationNavigateOptions const&); | 
					
						
							| 
									
										
										
										
											2023-08-25 14:09:14 -06:00
										 |  |  |  |     WebIDL::ExceptionOr<NavigationResult> reload(NavigationReloadOptions const&); | 
					
						
							| 
									
										
										
										
											2023-08-23 15:34:02 -06:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-25 20:35:22 -06:00
										 |  |  |  |     WebIDL::ExceptionOr<NavigationResult> traverse_to(String key, NavigationOptions const&); | 
					
						
							|  |  |  |  |     WebIDL::ExceptionOr<NavigationResult> back(NavigationOptions const&); | 
					
						
							|  |  |  |  |     WebIDL::ExceptionOr<NavigationResult> forward(NavigationOptions const&); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  |     // Event Handlers
 | 
					
						
							|  |  |  |  |     void set_onnavigate(WebIDL::CallbackType*); | 
					
						
							|  |  |  |  |     WebIDL::CallbackType* onnavigate(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     void set_onnavigatesuccess(WebIDL::CallbackType*); | 
					
						
							|  |  |  |  |     WebIDL::CallbackType* onnavigatesuccess(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     void set_onnavigateerror(WebIDL::CallbackType*); | 
					
						
							|  |  |  |  |     WebIDL::CallbackType* onnavigateerror(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     void set_oncurrententrychange(WebIDL::CallbackType*); | 
					
						
							|  |  |  |  |     WebIDL::CallbackType* oncurrententrychange(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // Abstract Operations
 | 
					
						
							|  |  |  |  |     bool has_entries_and_events_disabled() const; | 
					
						
							|  |  |  |  |     i64 get_the_navigation_api_entry_index(SessionHistoryEntry const&) const; | 
					
						
							| 
									
										
										
										
											2024-04-07 15:40:20 +02:00
										 |  |  |  |     void abort_the_ongoing_navigation(JS::GCPtr<WebIDL::DOMException> error = {}); | 
					
						
							| 
									
										
										
										
											2023-09-22 19:10:23 -06:00
										 |  |  |  |     bool fire_a_traverse_navigate_event(JS::NonnullGCPtr<SessionHistoryEntry> destination_she, UserNavigationInvolvement = UserNavigationInvolvement::None); | 
					
						
							| 
									
										
										
										
											2023-09-22 19:18:50 -06:00
										 |  |  |  |     bool fire_a_push_replace_reload_navigate_event( | 
					
						
							|  |  |  |  |         Bindings::NavigationType, | 
					
						
							| 
									
										
										
										
											2024-03-18 16:22:27 +13:00
										 |  |  |  |         URL::URL destination_url, | 
					
						
							| 
									
										
										
										
											2023-09-22 19:18:50 -06:00
										 |  |  |  |         bool is_same_document, | 
					
						
							|  |  |  |  |         UserNavigationInvolvement = UserNavigationInvolvement::None, | 
					
						
							|  |  |  |  |         Optional<Vector<XHR::FormDataEntry>&> form_data_entry_list = {}, | 
					
						
							|  |  |  |  |         Optional<SerializationRecord> navigation_api_state = {}, | 
					
						
							|  |  |  |  |         Optional<SerializationRecord> classic_history_api_state = {}); | 
					
						
							| 
									
										
										
										
											2024-03-18 16:22:27 +13:00
										 |  |  |  |     bool fire_a_download_request_navigate_event(URL::URL destination_url, UserNavigationInvolvement user_involvement, String filename); | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-27 22:56:11 -06:00
										 |  |  |  |     void initialize_the_navigation_api_entries_for_a_new_document(Vector<JS::NonnullGCPtr<SessionHistoryEntry>> const& new_shes, JS::NonnullGCPtr<SessionHistoryEntry> initial_she); | 
					
						
							|  |  |  |  |     void update_the_navigation_api_entries_for_a_same_document_navigation(JS::NonnullGCPtr<SessionHistoryEntry> destination_she, Bindings::NavigationType); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  |     virtual ~Navigation() override; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-22 18:24:28 -06:00
										 |  |  |  |     // Internal Getters/Setters
 | 
					
						
							| 
									
										
										
										
											2023-09-20 23:21:16 -06:00
										 |  |  |  |     JS::GCPtr<NavigateEvent> ongoing_navigate_event() const { return m_ongoing_navigate_event; } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-22 18:24:28 -06:00
										 |  |  |  |     bool focus_changed_during_ongoing_navigation() const { return m_focus_changed_during_ongoing_navigation; } | 
					
						
							|  |  |  |  |     void set_focus_changed_during_ongoing_navigation(bool b) { m_focus_changed_during_ongoing_navigation = b; } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  | private: | 
					
						
							|  |  |  |  |     explicit Navigation(JS::Realm&); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     virtual void initialize(JS::Realm&) override; | 
					
						
							|  |  |  |  |     virtual void visit_edges(Visitor&) override; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-24 16:20:38 -06:00
										 |  |  |  |     using AnyException = decltype(declval<WebIDL::ExceptionOr<void>>().exception()); | 
					
						
							|  |  |  |  |     NavigationResult early_error_result(AnyException); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     JS::NonnullGCPtr<NavigationAPIMethodTracker> maybe_set_the_upcoming_non_traverse_api_method_tracker(JS::Value info, Optional<SerializationRecord>); | 
					
						
							| 
									
										
										
										
											2023-08-25 20:35:22 -06:00
										 |  |  |  |     JS::NonnullGCPtr<NavigationAPIMethodTracker> add_an_upcoming_traverse_api_method_tracker(String destination_key, JS::Value info); | 
					
						
							|  |  |  |  |     WebIDL::ExceptionOr<NavigationResult> perform_a_navigation_api_traversal(String key, NavigationOptions const&); | 
					
						
							| 
									
										
										
										
											2023-09-22 18:30:40 -06:00
										 |  |  |  |     void promote_an_upcoming_api_method_tracker_to_ongoing(Optional<String> destination_key); | 
					
						
							|  |  |  |  |     void resolve_the_finished_promise(JS::NonnullGCPtr<NavigationAPIMethodTracker>); | 
					
						
							|  |  |  |  |     void reject_the_finished_promise(JS::NonnullGCPtr<NavigationAPIMethodTracker>, JS::Value exception); | 
					
						
							|  |  |  |  |     void clean_up(JS::NonnullGCPtr<NavigationAPIMethodTracker>); | 
					
						
							| 
									
										
										
										
											2023-09-27 22:56:11 -06:00
										 |  |  |  |     void notify_about_the_committed_to_entry(JS::NonnullGCPtr<NavigationAPIMethodTracker>, JS::NonnullGCPtr<NavigationHistoryEntry>); | 
					
						
							| 
									
										
										
										
											2024-05-25 14:21:02 +01:00
										 |  |  |  |     void consume_history_action_user_activation(Window& window); | 
					
						
							| 
									
										
										
										
											2023-08-24 16:20:38 -06:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-22 18:35:11 -06:00
										 |  |  |  |     bool inner_navigate_event_firing_algorithm( | 
					
						
							|  |  |  |  |         Bindings::NavigationType, | 
					
						
							|  |  |  |  |         JS::NonnullGCPtr<NavigationDestination>, | 
					
						
							|  |  |  |  |         UserNavigationInvolvement, | 
					
						
							|  |  |  |  |         Optional<Vector<XHR::FormDataEntry>&> form_data_entry_list, | 
					
						
							|  |  |  |  |         Optional<String> download_request_filename, | 
					
						
							|  |  |  |  |         Optional<SerializationRecord> classic_history_api_state); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  |     // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigation-entry-list
 | 
					
						
							|  |  |  |  |     // Each Navigation has an associated entry list, a list of NavigationHistoryEntry objects, initially empty.
 | 
					
						
							|  |  |  |  |     Vector<JS::NonnullGCPtr<NavigationHistoryEntry>> m_entry_list; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigation-current-entry-index
 | 
					
						
							|  |  |  |  |     // Each Navigation has an associated current entry index, an integer, initially −1.
 | 
					
						
							|  |  |  |  |     i64 m_current_entry_index { -1 }; | 
					
						
							| 
									
										
										
										
											2023-08-23 15:34:02 -06:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // https://html.spec.whatwg.org/multipage/nav-history-apis.html#concept-navigation-transition
 | 
					
						
							|  |  |  |  |     // Each Navigation has a transition, which is a NavigationTransition or null, initially null.
 | 
					
						
							|  |  |  |  |     JS::GCPtr<NavigationTransition> m_transition { nullptr }; | 
					
						
							| 
									
										
										
										
											2023-08-24 16:20:38 -06:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // https://html.spec.whatwg.org/multipage/nav-history-apis.html#ongoing-navigate-event
 | 
					
						
							|  |  |  |  |     JS::GCPtr<NavigateEvent> m_ongoing_navigate_event { nullptr }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // https://html.spec.whatwg.org/multipage/nav-history-apis.html#focus-changed-during-ongoing-navigation
 | 
					
						
							|  |  |  |  |     bool m_focus_changed_during_ongoing_navigation { false }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // https://html.spec.whatwg.org/multipage/nav-history-apis.html#suppress-normal-scroll-restoration-during-ongoing-navigation
 | 
					
						
							|  |  |  |  |     bool m_suppress_scroll_restoration_during_ongoing_navigation { false }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // https://html.spec.whatwg.org/multipage/nav-history-apis.html#ongoing-api-method-tracker
 | 
					
						
							|  |  |  |  |     JS::GCPtr<NavigationAPIMethodTracker> m_ongoing_api_method_tracker = nullptr; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // https://html.spec.whatwg.org/multipage/nav-history-apis.html#upcoming-non-traverse-api-method-tracker
 | 
					
						
							|  |  |  |  |     JS::GCPtr<NavigationAPIMethodTracker> m_upcoming_non_traverse_api_method_tracker = nullptr; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // https://html.spec.whatwg.org/multipage/nav-history-apis.html#upcoming-non-traverse-api-method-tracker
 | 
					
						
							|  |  |  |  |     HashMap<String, JS::NonnullGCPtr<NavigationAPIMethodTracker>> m_upcoming_traverse_api_method_trackers; | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-28 18:03:45 +02:00
										 |  |  |  | HistoryHandlingBehavior to_history_handling_behavior(Bindings::NavigationHistoryBehavior); | 
					
						
							| 
									
										
										
										
											2024-03-28 12:58:43 +01:00
										 |  |  |  | Bindings::NavigationHistoryBehavior to_navigation_history_behavior(HistoryHandlingBehavior); | 
					
						
							| 
									
										
										
										
											2023-08-28 18:03:45 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-23 10:57:12 -06:00
										 |  |  |  | } |