
The invariants for these property getters are supposed to be checked by the has_entries_and_events_disabled AO, but we don't have all the plumbing hooked up between Navigables and Navigation yet. Add a test to make sure that these methods don't assert when calling them on a fresh page.
443 lines
19 KiB
C++
443 lines
19 KiB
C++
/*
|
||
* Copyright (c) 2023, Andrew Kaster <akaster@serenityos.org>
|
||
*
|
||
* SPDX-License-Identifier: BSD-2-Clause
|
||
*/
|
||
|
||
#include <LibJS/Heap/Heap.h>
|
||
#include <LibJS/Runtime/Realm.h>
|
||
#include <LibJS/Runtime/VM.h>
|
||
#include <LibWeb/Bindings/ExceptionOrUtils.h>
|
||
#include <LibWeb/Bindings/Intrinsics.h>
|
||
#include <LibWeb/Bindings/NavigationPrototype.h>
|
||
#include <LibWeb/DOM/Document.h>
|
||
#include <LibWeb/HTML/NavigateEvent.h>
|
||
#include <LibWeb/HTML/Navigation.h>
|
||
#include <LibWeb/HTML/NavigationCurrentEntryChangeEvent.h>
|
||
#include <LibWeb/HTML/NavigationHistoryEntry.h>
|
||
#include <LibWeb/HTML/NavigationTransition.h>
|
||
#include <LibWeb/HTML/Window.h>
|
||
|
||
namespace Web::HTML {
|
||
|
||
static NavigationResult navigation_api_method_tracker_derived_result(JS::NonnullGCPtr<NavigationAPIMethodTracker> api_method_tracker);
|
||
|
||
NavigationAPIMethodTracker::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)
|
||
: 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> Navigation::create(JS::Realm& realm)
|
||
{
|
||
return realm.heap().allocate<Navigation>(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<Bindings::NavigationPrototype>(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<JS::NonnullGCPtr<NavigationHistoryEntry>> 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<NavigationHistoryEntry> 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<void> 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<i64>(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<NavigationResult> 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<HTML::Window>(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<HTML::Window>(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<NavigationAPIMethodTracker> 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<NavigationAPIMethodTracker> Navigation::maybe_set_the_upcoming_non_traverse_api_method_tracker(JS::Value info, Optional<SerializationRecord> 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<NavigationAPIMethodTracker>(
|
||
/* .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;
|
||
}
|
||
|
||
}
|