/* * Copyright (c) 2023, Andrew Kaster * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Web::HTML { static NavigationResult navigation_api_method_tracker_derived_result(JS::NonnullGCPtr api_method_tracker); NavigationAPIMethodTracker::NavigationAPIMethodTracker(JS::NonnullGCPtr navigation, Optional key, JS::Value info, Optional serialized_state, JS::GCPtr commited_to_entry, JS::NonnullGCPtr committed_promise, JS::NonnullGCPtr finished_promise) : navigation(navigation) , key(move(key)) , info(info) , serialized_state(move(serialized_state)) , commited_to_entry(commited_to_entry) , committed_promise(committed_promise) , finished_promise(finished_promise) { } void NavigationAPIMethodTracker::visit_edges(Cell::Visitor& visitor) { Base::visit_edges(visitor); visitor.visit(navigation); visitor.visit(info); visitor.visit(commited_to_entry); visitor.visit(committed_promise); visitor.visit(finished_promise); } JS::NonnullGCPtr Navigation::create(JS::Realm& realm) { return realm.heap().allocate(realm, realm); } Navigation::Navigation(JS::Realm& realm) : DOM::EventTarget(realm) { } Navigation::~Navigation() = default; void Navigation::initialize(JS::Realm& realm) { Base::initialize(realm); set_prototype(&Bindings::ensure_web_prototype(realm, "Navigation")); } void Navigation::visit_edges(JS::Cell::Visitor& visitor) { Base::visit_edges(visitor); for (auto& entry : m_entry_list) visitor.visit(entry); visitor.visit(m_transition); visitor.visit(m_ongoing_navigate_event); visitor.visit(m_ongoing_api_method_tracker); visitor.visit(m_upcoming_non_traverse_api_method_tracker); for (auto& key_and_tracker : m_upcoming_traverse_api_method_trackers) visitor.visit(key_and_tracker.value); } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#dom-navigation-entries Vector> Navigation::entries() const { // The entries() method steps are: // 1. If this has entries and events disabled, then return the empty list. if (has_entries_and_events_disabled()) return {}; // 2. Return this's entry list. // NOTE: Recall that because of Web IDL's sequence type conversion rules, // this will create a new JavaScript array object on each call. // That is, navigation.entries() !== navigation.entries(). return m_entry_list; } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigation-current-entry JS::GCPtr Navigation::current_entry() const { // The current entry of a Navigation navigation is the result of running the following steps: // 1. If navigation has entries and events disabled, then return null. if (has_entries_and_events_disabled()) return nullptr; // FIXME 2. Assert: navigation's current entry index is not −1. // FIXME: This should not happen once Navigable's algorithms properly set up NHEs. if (m_current_entry_index == -1) return nullptr; // 3. Return navigation's entry list[navigation's current entry index]. return m_entry_list[m_current_entry_index]; } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#dom-navigation-updatecurrententry WebIDL::ExceptionOr Navigation::update_current_entry(NavigationUpdateCurrentEntryOptions options) { // The updateCurrentEntry(options) method steps are: // 1. Let current be the current entry of this. auto current = current_entry(); // 2. If current is null, then throw an "InvalidStateError" DOMException. if (current == nullptr) return WebIDL::InvalidStateError::create(realm(), "Cannot update current NavigationHistoryEntry when there is no current entry"sv); // 3. Let serializedState be StructuredSerializeForStorage(options["state"]), rethrowing any exceptions. auto serialized_state = TRY(structured_serialize_for_storage(vm(), options.state)); // 4. Set current's session history entry's navigation API state to serializedState. current->session_history_entry().navigation_api_state = serialized_state; // 5. Fire an event named currententrychange at this using NavigationCurrentEntryChangeEvent, // with its navigationType attribute initialized to null and its from initialized to current. NavigationCurrentEntryChangeEventInit event_init = {}; event_init.navigation_type = {}; event_init.from = current; dispatch_event(HTML::NavigationCurrentEntryChangeEvent::create(realm(), HTML::EventNames::currententrychange, event_init)); return {}; } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#dom-navigation-cangoback bool Navigation::can_go_back() const { // The canGoBack getter steps are: // 1. If this has entries and events disabled, then return false. if (has_entries_and_events_disabled()) return false; // FIXME 2. Assert: navigation's current entry index is not −1. // FIXME: This should not happen once Navigable's algorithms properly set up NHEs. if (m_current_entry_index == -1) return false; // 3. If this's current entry index is 0, then return false. // 4. Return true. return (m_current_entry_index != 0); } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#dom-navigation-cangoforward bool Navigation::can_go_forward() const { // The canGoForward getter steps are: // 1. If this has entries and events disabled, then return false. if (has_entries_and_events_disabled()) return false; // FIXME 2. Assert: navigation's current entry index is not −1. // FIXME: This should not happen once Navigable's algorithms properly set up NHEs. if (m_current_entry_index == -1) return false; // 3. If this's current entry index is equal to this's entry list's size, then return false. // 4. Return true. return (m_current_entry_index != static_cast(m_entry_list.size())); } static HistoryHandlingBehavior to_history_handling_behavior(Bindings::NavigationHistoryBehavior b) { switch (b) { case Bindings::NavigationHistoryBehavior::Auto: return HistoryHandlingBehavior::Default; case Bindings::NavigationHistoryBehavior::Push: return HistoryHandlingBehavior::Push; case Bindings::NavigationHistoryBehavior::Replace: return HistoryHandlingBehavior::Replace; }; VERIFY_NOT_REACHED(); } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#dom-navigation-navigate WebIDL::ExceptionOr Navigation::navigate(String url, NavigationNavigateOptions const& options) { auto& realm = this->realm(); auto& vm = this->vm(); // The navigate(options) method steps are: // 1. Parse url relative to this's relevant settings object. // If that returns failure, then return an early error result for a "SyntaxError" DOMException. // Otherwise, let urlRecord be the resulting URL record. auto url_record = relevant_settings_object(*this).parse_url(url); if (!url_record.is_valid()) return early_error_result(WebIDL::SyntaxError::create(realm, "Cannot navigate to Invalid URL")); // 2. Let document be this's relevant global object's associated Document. auto& document = verify_cast(relevant_global_object(*this)).associated_document(); // 3. If options["history"] is "push", and the navigation must be a replace given urlRecord and document, // then return an early error result for a "NotSupportedError" DOMException. if (options.history == Bindings::NavigationHistoryBehavior::Push && navigation_must_be_a_replace(url_record, document)) return early_error_result(WebIDL::NotSupportedError::create(realm, "Navigation must be a replace, but push was requested")); // 4. Let state be options["state"], if it exists; otherwise, undefined. auto state = options.state.value_or(JS::js_undefined()); // 5. Let serializedState be StructuredSerializeForStorage(state). // If this throws an exception, then return an early error result for that exception. // FIXME: Fix this spec grammaro in the note // NOTE: It is importantly to perform this step early, since serialization can invoke web developer code, // which in turn might change various things we check in later steps. auto serialized_state_or_error = structured_serialize_for_storage(vm, state); if (serialized_state_or_error.is_error()) { return early_error_result(serialized_state_or_error.release_error()); } auto serialized_state = serialized_state_or_error.release_value(); // 6. If document is not fully active, then return an early error result for an "InvalidStateError" DOMException. if (!document.is_fully_active()) return early_error_result(WebIDL::InvalidStateError::create(realm, "Document is not fully active")); // 7. If document's unload counter is greater than 0, then return an early error result for an "InvalidStateError" DOMException. if (document.unload_counter() > 0) return early_error_result(WebIDL::InvalidStateError::create(realm, "Document already unloaded")); // 8. Let info be options["info"], if it exists; otherwise, undefined. auto info = options.info.value_or(JS::js_undefined()); // 9. Let apiMethodTracker be the result of maybe setting the upcoming non-traverse API method tracker for this // given info and serializedState. auto api_method_tracker = maybe_set_the_upcoming_non_traverse_api_method_tracker(info, serialized_state); // 10. Navigate document's node navigable to urlRecord using document, // with historyHandling set to options["history"] and navigationAPIState set to serializedState. // FIXME: Fix spec typo here // NOTE: Unlike location.assign() and friends, which are exposed across origin-domain boundaries, // navigation.navigate() can only be accessed by code with direct synchronous access to the /// window.navigation property. Thus, we avoid the complications about attributing the source document // of the navigation, and we don't need to deal with the allowed by sandboxing to navigate check and its // acccompanying exceptionsEnabled flag. We just treat all navigations as if they come from the Document // corresponding to this Navigation object itself (i.e., document). [[maybe_unused]] auto history_handling_behavior = to_history_handling_behavior(options.history); // FIXME: Actually call navigate once Navigables are implemented enough to guarantee a node navigable on // an active document that's not being unloaded. // document.navigable().navigate(url, document, history behavior, state) // 11. If this's upcoming non-traverse API method tracker is apiMethodTracker, then: // NOTE: If the upcoming non-traverse API method tracker is still apiMethodTracker, this means that the navigate // algorithm bailed out before ever getting to the inner navigate event firing algorithm which would promote // that upcoming API method tracker to ongoing. if (m_upcoming_non_traverse_api_method_tracker == api_method_tracker) { m_upcoming_non_traverse_api_method_tracker = nullptr; return early_error_result(WebIDL::AbortError::create(realm, "Navigation aborted")); } // 12. Return a navigation API method tracker-derived result for apiMethodTracker. return navigation_api_method_tracker_derived_result(api_method_tracker); } void Navigation::set_onnavigate(WebIDL::CallbackType* event_handler) { set_event_handler_attribute(HTML::EventNames::navigate, event_handler); } WebIDL::CallbackType* Navigation::onnavigate() { return event_handler_attribute(HTML::EventNames::navigate); } void Navigation::set_onnavigatesuccess(WebIDL::CallbackType* event_handler) { set_event_handler_attribute(HTML::EventNames::navigatesuccess, event_handler); } WebIDL::CallbackType* Navigation::onnavigatesuccess() { return event_handler_attribute(HTML::EventNames::navigatesuccess); } void Navigation::set_onnavigateerror(WebIDL::CallbackType* event_handler) { set_event_handler_attribute(HTML::EventNames::navigateerror, event_handler); } WebIDL::CallbackType* Navigation::onnavigateerror() { return event_handler_attribute(HTML::EventNames::navigateerror); } void Navigation::set_oncurrententrychange(WebIDL::CallbackType* event_handler) { set_event_handler_attribute(HTML::EventNames::currententrychange, event_handler); } WebIDL::CallbackType* Navigation::oncurrententrychange() { return event_handler_attribute(HTML::EventNames::currententrychange); } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#has-entries-and-events-disabled bool Navigation::has_entries_and_events_disabled() const { // A Navigation navigation has entries and events disabled if the following steps return true: // 1. Let document be navigation's relevant global object's associated Document. auto const& document = verify_cast(relevant_global_object(*this)).associated_document(); // 2. If document is not fully active, then return true. if (!document.is_fully_active()) return true; // 3. If document's is initial about:blank is true, then return true. if (document.is_initial_about_blank()) return true; // 4. If document's origin is opaque, then return true. if (document.origin().is_opaque()) return true; // 5. Return false. return false; } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#getting-the-navigation-api-entry-index i64 Navigation::get_the_navigation_api_entry_index(SessionHistoryEntry const& she) const { // To get the navigation API entry index of a session history entry she within a Navigation navigation: // 1. Let index be 0. i64 index = 0; // 2. For each nhe of navigation's entry list: for (auto const& nhe : m_entry_list) { // 1. If nhe's session history entry is equal to she, then return index. if (&nhe->session_history_entry() == &she) return index; // 2. Increment index by 1. ++index; } // 3. Return −1. return -1; } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigation-api-early-error-result NavigationResult Navigation::early_error_result(AnyException e) { auto& vm = this->vm(); // An early error result for an exception e is a NavigationResult dictionary instance given by // «[ "committed" → a promise rejected with e, "finished" → a promise rejected with e ]». auto throw_completion = Bindings::dom_exception_to_throw_completion(vm, e); return { .committed = WebIDL::create_rejected_promise(realm(), *throw_completion.value())->promise(), .finished = WebIDL::create_rejected_promise(realm(), *throw_completion.value())->promise(), }; } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#navigation-api-method-tracker-derived-result NavigationResult navigation_api_method_tracker_derived_result(JS::NonnullGCPtr api_method_tracker) { // A navigation API method tracker-derived result for a navigation API method tracker is a NavigationResult /// dictionary instance given by «[ "committed" apiMethodTracker's committed promise, "finished" → apiMethodTracker's finished promise ]». return { api_method_tracker->committed_promise->promise(), api_method_tracker->finished_promise->promise(), }; } // https://html.spec.whatwg.org/multipage/nav-history-apis.html#upcoming-non-traverse-api-method-tracker JS::NonnullGCPtr Navigation::maybe_set_the_upcoming_non_traverse_api_method_tracker(JS::Value info, Optional serialized_state) { auto& realm = relevant_realm(*this); auto& vm = this->vm(); // To maybe set the upcoming non-traverse API method tracker given a Navigation navigation, // a JavaScript value info, and a serialized state-or-null serializedState: // 1. Let committedPromise and finishedPromise be new promises created in navigation's relevant realm. auto committed_promise = WebIDL::create_promise(realm); auto finished_promise = WebIDL::create_promise(realm); // 2. Mark as handled finishedPromise. // NOTE: The web developer doesn’t necessarily care about finishedPromise being rejected: // - They might only care about committedPromise. // - They could be doing multiple synchronous navigations within the same task, // in which case all but the last will be aborted (causing their finishedPromise to reject). // This could be an application bug, but also could just be an emergent feature of disparate // parts of the application overriding each others' actions. // - They might prefer to listen to other transition-failure signals instead of finishedPromise, e.g., // the navigateerror event, or the navigation.transition.finished promise. // As such, we mark it as handled to ensure that it never triggers unhandledrejection events. WebIDL::mark_promise_as_handled(finished_promise); // 3. Let apiMethodTracker be a new navigation API method tracker with: // navigation object: navigation // key: null // info: info // serialized state: serializedState // comitted-to entry: null // comitted promise: committedPromise // finished promise: finishedPromise auto api_method_tracker = vm.heap().allocate_without_realm( /* .navigation = */ *this, /* .key = */ OptionalNone {}, /* .info = */ info, /* .serialized_state = */ move(serialized_state), /* .commited_to_entry = */ nullptr, /* .committed_promise = */ committed_promise, /* .finished_promise = */ finished_promise); // 4. Assert: navigation's upcoming non-traverse API method tracker is null. VERIFY(m_upcoming_non_traverse_api_method_tracker == nullptr); // 5. If navigation does not have entries and events disabled, // then set navigation's upcoming non-traverse API method tracker to apiMethodTracker. // NOTE: If navigation has entries and events disabled, then committedPromise and finishedPromise will never fulfill // (since we never create a NavigationHistoryEntry object for such Documents, and so we have nothing to resolve them with); // there is no NavigationHistoryEntry to apply serializedState to; and there is no navigate event to include info with. // So, we don't need to track this API method call after all. if (!has_entries_and_events_disabled()) m_upcoming_non_traverse_api_method_tracker = api_method_tracker; // 6. Return apiMethodTracker. return api_method_tracker; } }