ladybird/Libraries/LibWeb/HTML/Navigable.cpp
Luke Wilde eeb5446c1b LibWeb: Avoid including Navigable.h in headers
This greatly reduces how much is recompiled when changing Navigable.h,
from >1000 to 82.
2025-10-20 10:16:55 +01:00

2769 lines
138 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* Copyright (c) 2022-2024, Andreas Kling <andreas@ladybird.org>
* Copyright (c) 2023-2025, Aliaksandr Kalenik <kalenik.aliaksandr@gmail.com>
* Copyright (c) 2025, Luke Wilde <luke@ladybird.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <LibWeb/CSS/SystemColor.h>
#include <LibWeb/CSS/VisualViewport.h>
#include <LibWeb/ContentSecurityPolicy/BlockingAlgorithms.h>
#include <LibWeb/ContentSecurityPolicy/Directives/DirectiveOperations.h>
#include <LibWeb/ContentSecurityPolicy/PolicyList.h>
#include <LibWeb/ContentSecurityPolicy/Violation.h>
#include <LibWeb/Crypto/Crypto.h>
#include <LibWeb/DOM/Document.h>
#include <LibWeb/DOM/DocumentLoading.h>
#include <LibWeb/DOM/Event.h>
#include <LibWeb/DOM/Range.h>
#include <LibWeb/DOM/Text.h>
#include <LibWeb/Fetch/Fetching/Fetching.h>
#include <LibWeb/Fetch/Infrastructure/FetchAlgorithms.h>
#include <LibWeb/Fetch/Infrastructure/FetchController.h>
#include <LibWeb/Fetch/Infrastructure/HTTP/Requests.h>
#include <LibWeb/Fetch/Infrastructure/URL.h>
#include <LibWeb/HTML/BrowsingContext.h>
#include <LibWeb/HTML/BrowsingContextGroup.h>
#include <LibWeb/HTML/DocumentState.h>
#include <LibWeb/HTML/HTMLIFrameElement.h>
#include <LibWeb/HTML/HistoryHandlingBehavior.h>
#include <LibWeb/HTML/Navigable.h>
#include <LibWeb/HTML/Navigation.h>
#include <LibWeb/HTML/NavigationObserver.h>
#include <LibWeb/HTML/NavigationParams.h>
#include <LibWeb/HTML/POSTResource.h>
#include <LibWeb/HTML/Parser/HTMLParser.h>
#include <LibWeb/HTML/SandboxingFlagSet.h>
#include <LibWeb/HTML/Scripting/ClassicScript.h>
#include <LibWeb/HTML/SessionHistoryEntry.h>
#include <LibWeb/HTML/StructuredSerialize.h>
#include <LibWeb/HTML/TraversableNavigable.h>
#include <LibWeb/HTML/Window.h>
#include <LibWeb/HTML/WindowProxy.h>
#include <LibWeb/Infra/Strings.h>
#include <LibWeb/Layout/Node.h>
#include <LibWeb/Layout/Viewport.h>
#include <LibWeb/Loader/GeneratedPagesLoader.h>
#include <LibWeb/Page/Page.h>
#include <LibWeb/Painting/DisplayListPlayerSkia.h>
#include <LibWeb/Painting/NavigableContainerViewportPaintable.h>
#include <LibWeb/Painting/Paintable.h>
#include <LibWeb/Painting/ViewportPaintable.h>
#include <LibWeb/Platform/EventLoopPlugin.h>
#include <LibWeb/Selection/Selection.h>
#include <LibWeb/XHR/FormData.h>
namespace Web::HTML {
GC_DEFINE_ALLOCATOR(Navigable);
class ResponseHolder : public JS::Cell {
GC_CELL(ResponseHolder, JS::Cell);
GC_DECLARE_ALLOCATOR(ResponseHolder);
public:
[[nodiscard]] static GC::Ref<ResponseHolder> create(JS::VM& vm)
{
return vm.heap().allocate<ResponseHolder>();
}
[[nodiscard]] GC::Ptr<Fetch::Infrastructure::Response> response() const { return m_response; }
void set_response(GC::Ptr<Fetch::Infrastructure::Response> response) { m_response = response; }
virtual void visit_edges(Cell::Visitor& visitor) override
{
Base::visit_edges(visitor);
visitor.visit(m_response);
}
private:
GC::Ptr<Fetch::Infrastructure::Response> m_response;
};
GC_DEFINE_ALLOCATOR(ResponseHolder);
HashTable<GC::RawRef<Navigable>>& all_navigables()
{
static HashTable<GC::RawRef<Navigable>> set;
return set;
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#child-navigable
Vector<GC::Root<Navigable>> Navigable::child_navigables() const
{
Vector<GC::Root<Navigable>> results;
for (auto& entry : all_navigables()) {
if (entry->current_session_history_entry()->step() == SessionHistoryEntry::Pending::Tag)
continue;
if (entry->parent() == this)
results.append(entry);
}
return results;
}
bool Navigable::is_ancestor_of(GC::Ref<Navigable> other) const
{
for (auto ancestor = other->parent(); ancestor; ancestor = ancestor->parent()) {
if (ancestor == this)
return true;
}
return false;
}
static RefPtr<Gfx::SkiaBackendContext> g_cached_skia_backend_context;
static RefPtr<Gfx::SkiaBackendContext> get_skia_backend_context()
{
if (!g_cached_skia_backend_context) {
#ifdef AK_OS_MACOS
auto metal_context = Gfx::get_metal_context();
g_cached_skia_backend_context = Gfx::SkiaBackendContext::create_metal_context(*metal_context);
#elif USE_VULKAN
auto maybe_vulkan_context = Gfx::create_vulkan_context();
if (maybe_vulkan_context.is_error()) {
dbgln("Vulkan context creation failed: {}", maybe_vulkan_context.error());
return {};
}
auto vulkan_context = maybe_vulkan_context.release_value();
g_cached_skia_backend_context = Gfx::SkiaBackendContext::create_vulkan_context(vulkan_context);
#endif
}
return g_cached_skia_backend_context;
}
Navigable::Navigable(GC::Ref<Page> page, bool is_svg_page)
: m_page(page)
, m_event_handler({}, *this)
, m_is_svg_page(is_svg_page)
, m_backing_store_manager(heap().allocate<Painting::BackingStoreManager>(*this))
{
all_navigables().set(*this);
if (!m_is_svg_page) {
auto display_list_player_type = page->client().display_list_player_type();
OwnPtr<Painting::DisplayListPlayerSkia> skia_player;
if (display_list_player_type == DisplayListPlayerType::SkiaGPUIfAvailable) {
m_skia_backend_context = get_skia_backend_context();
skia_player = make<Painting::DisplayListPlayerSkia>(m_skia_backend_context);
} else {
skia_player = make<Painting::DisplayListPlayerSkia>();
}
m_rendering_thread.set_skia_player(move(skia_player));
m_rendering_thread.start(display_list_player_type);
}
}
Navigable::~Navigable() = default;
void Navigable::finalize()
{
all_navigables().remove(*this);
Base::finalize();
}
void Navigable::visit_edges(Cell::Visitor& visitor)
{
Base::visit_edges(visitor);
visitor.visit(m_page);
visitor.visit(m_parent);
visitor.visit(m_current_session_history_entry);
visitor.visit(m_active_session_history_entry);
visitor.visit(m_container);
visitor.visit(m_navigation_observers);
visitor.visit(m_backing_store_manager);
m_event_handler.visit_edges(visitor);
for (auto& navigation_params : m_pending_navigations) {
navigation_params.visit_edges(visitor);
}
}
void Navigable::NavigateParams::visit_edges(Cell::Visitor& visitor)
{
visitor.visit(response);
visitor.visit(source_document);
visitor.visit(source_element);
}
// https://html.spec.whatwg.org/multipage/nav-history-apis.html#script-closable
bool Navigable::is_script_closable()
{
// A navigable is script-closable if it is a top-level traversable, and any of the following are true:
// - its is created by web content is true; or
// - its session history entries's size is 1.
if (!is_top_level_traversable())
return false;
return as<TraversableNavigable>(this)->is_created_by_web_content()
|| get_session_history_entries().size() == 1;
}
void Navigable::set_delaying_load_events(bool value)
{
if (value) {
auto document = container_document();
VERIFY(document);
m_delaying_the_load_event.emplace(*document);
} else {
m_delaying_the_load_event.clear();
}
}
GC::Ptr<Navigable> Navigable::navigable_with_active_document(GC::Ref<DOM::Document> document)
{
for (auto navigable : all_navigables()) {
if (navigable->active_document() == document)
return navigable;
}
return nullptr;
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#initialize-the-navigable
ErrorOr<void> Navigable::initialize_navigable(GC::Ref<DocumentState> document_state, GC::Ptr<Navigable> parent)
{
static int next_id = 0;
m_id = String::number(next_id++);
// 1. Assert: documentState's document is non-null.
VERIFY(document_state->document());
// 2. Let entry be a new session history entry, with
GC::Ref<SessionHistoryEntry> entry = *heap().allocate<SessionHistoryEntry>();
// URL: document's URL
entry->set_url(document_state->document()->url());
// document state: documentState
entry->set_document_state(document_state);
// 3. Set navigable's current session history entry to entry.
m_current_session_history_entry = entry;
// 4. Set navigable's active session history entry to entry.
m_active_session_history_entry = entry;
// 5. Set navigable's parent to parent.
m_parent = parent;
return {};
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#getting-the-target-history-entry
GC::Ptr<SessionHistoryEntry> Navigable::get_the_target_history_entry(int target_step) const
{
// 1. Let entries be the result of getting session history entries for navigable.
auto& entries = get_session_history_entries();
// 2. Return the item in entries that has the greatest step less than or equal to step.
GC::Ptr<SessionHistoryEntry> result = nullptr;
for (auto& entry : entries) {
auto entry_step = entry->step().get<int>();
if (entry_step <= target_step) {
if (!result || result->step().get<int>() < entry_step) {
result = entry;
}
}
}
return result;
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#activate-history-entry
void Navigable::activate_history_entry(GC::Ptr<SessionHistoryEntry> entry)
{
// FIXME: 1. Save persisted state to the navigable's active session history entry.
// 2. Let newDocument be entry's document.
GC::Ptr<DOM::Document> new_document = entry->document().ptr();
// 3. Assert: newDocument's is initial about:blank is false, i.e., we never traverse
// back to the initial about:blank Document because it always gets replaced when we
// navigate away from it.
VERIFY(!new_document->is_initial_about_blank());
// 4. Set navigable's active session history entry to entry.
m_active_session_history_entry = entry;
// 5. Make active newDocument.
new_document->make_active();
if (m_ongoing_navigation.has<Empty>()) {
for (auto navigation_observer : m_navigation_observers) {
if (navigation_observer->navigation_complete())
navigation_observer->navigation_complete()->function()();
}
}
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#nav-document
GC::Ptr<DOM::Document> Navigable::active_document() const
{
// A navigable's active document is its active session history entry's document.
return m_active_session_history_entry->document();
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#nav-bc
GC::Ptr<BrowsingContext> Navigable::active_browsing_context()
{
// A navigable's active browsing context is its active document's browsing context.
// If this navigable is a traversable navigable, then its active browsing context will be a top-level browsing context.
if (auto document = active_document())
return document->browsing_context();
return nullptr;
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#nav-wp
GC::Ptr<HTML::WindowProxy> Navigable::active_window_proxy()
{
// A navigable's active WindowProxy is its active browsing context's associated WindowProxy.
if (auto browsing_context = active_browsing_context())
return browsing_context->window_proxy();
return nullptr;
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#nav-window
GC::Ptr<HTML::Window> Navigable::active_window()
{
// A navigable's active window is its active WindowProxy's [[Window]].
if (auto window_proxy = active_window_proxy())
return window_proxy->window();
return nullptr;
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#nav-target
String Navigable::target_name() const
{
// A navigable's target name is its active session history entry's document state's navigable target name.
return active_session_history_entry()->document_state()->navigable_target_name();
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#nav-container
GC::Ptr<NavigableContainer> Navigable::container() const
{
// The container of a navigable navigable is the navigable container whose nested navigable is navigable, or null if there is no such element.
return NavigableContainer::navigable_container_with_content_navigable(const_cast<Navigable&>(*this));
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#nav-container-document
GC::Ptr<DOM::Document> Navigable::container_document() const
{
auto container = this->container();
// 1. If navigable's container is null, then return null.
if (!container)
return nullptr;
// 2. Return navigable's container's node document.
return container->document();
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#nav-traversable
GC::Ptr<TraversableNavigable> Navigable::traversable_navigable() const
{
// 1. Let navigable be inputNavigable.
auto navigable = const_cast<Navigable*>(this);
// 2. While navigable is not a traversable navigable, set navigable to navigable's parent.
while (navigable && !is<TraversableNavigable>(*navigable))
navigable = navigable->parent();
// 3. Return navigable.
return static_cast<TraversableNavigable*>(navigable);
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#nav-top
GC::Ptr<TraversableNavigable> Navigable::top_level_traversable()
{
// 1. Let navigable be inputNavigable.
auto navigable = this;
// 2. While navigable's parent is not null, set navigable to navigable's parent.
while (navigable->parent())
navigable = navigable->parent();
// 3. Return navigable.
return as<TraversableNavigable>(navigable);
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#set-the-ongoing-navigation
void Navigable::set_ongoing_navigation(Variant<Empty, Traversal, String> ongoing_navigation)
{
// 1. If navigable's ongoing navigation is equal to newValue, then return.
if (m_ongoing_navigation == ongoing_navigation)
return;
// 2. Inform the navigation API about aborting navigation given navigable.
inform_the_navigation_api_about_aborting_navigation();
// 3. Set navigable's ongoing navigation to newValue.
m_ongoing_navigation = ongoing_navigation;
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#the-rules-for-choosing-a-navigable
Navigable::ChosenNavigable Navigable::choose_a_navigable(StringView name, TokenizedFeature::NoOpener no_opener, ActivateTab activate_tab, Optional<TokenizedFeature::Map const&> window_features)
{
// 1. Let chosen be null.
GC::Ptr<Navigable> chosen = nullptr;
// 2. Let windowType be "existing or none".
auto window_type = WindowType::ExistingOrNone;
// 3. Let sandboxingFlagSet be currentNavigable's active document's active sandboxing flag set.
auto sandboxing_flag_set = active_document()->active_sandboxing_flag_set();
// 4. If name is the empty string or an ASCII case-insensitive match for "_self", then set chosen to currentNavigable.
if (name.is_empty() || name.equals_ignoring_ascii_case("_self"sv)) {
chosen = this;
}
// 5. Otherwise, if name is an ASCII case-insensitive match for "_parent",
// set chosen to currentNavigable's parent, if any, and currentNavigable otherwise.
else if (name.equals_ignoring_ascii_case("_parent"sv)) {
if (auto parent = this->parent())
chosen = parent;
else
chosen = this;
}
// 6. Otherwise, if name is an ASCII case-insensitive match for "_top",
// set chosen to currentNavigable's traversable navigable.
else if (name.equals_ignoring_ascii_case("_top"sv)) {
chosen = traversable_navigable();
}
// 7. Otherwise, if name is not an ASCII case-insensitive match for "_blank" and noopener is false, then set chosen
// to the result of finding a navigable by target name given name and currentNavigable.
else if (!name.equals_ignoring_ascii_case("_blank"sv) && no_opener == TokenizedFeature::NoOpener::No) {
chosen = find_a_navigable_by_target_name(name);
}
// 8. If chosen is null, then a new top-level traversable is being requested, and what happens depends on the user
// agent's configuration and abilities — it is determined by the rules given for the first applicable option
// from the following list:
if (!chosen) {
// --> If currentNavigable's active window does not have transient activation and the user agent has been configured to
// not show popups (i.e., the user agent has a "popup blocker" enabled)
if (active_window() && !active_window()->has_transient_activation() && traversable_navigable()->page().should_block_pop_ups()) {
// FIXME: The user agent may inform the user that a popup has been blocked.
dbgln("Pop-up blocked!");
}
// --> If sandboxingFlagSet has the sandboxed auxiliary navigation browsing context flag set
else if (has_flag(sandboxing_flag_set, SandboxingFlagSet::SandboxedAuxiliaryNavigation)) {
// FIXME: The user agent may report to a developer console that a popup has been blocked.
dbgln("Pop-up blocked!");
}
// --> If the user agent has been configured such that in this instance it will create a new top-level traversable
else if (true) { // FIXME: When is this the case?
// 1. Consume user activation of currentNavigable's active window.
active_window()->consume_user_activation();
// 2. Set windowType to "new and unrestricted".
window_type = WindowType::NewAndUnrestricted;
// 3. Let currentDocument be currentNavigable's active document.
auto current_document = active_document();
// 4. If currentDocument's opener policy's value is "same-origin" or "same-origin-plus-COEP",
// and currentDocument's origin is not same origin with currentDocument's relevant settings object's top-level origin, then:
if ((current_document->opener_policy().value == OpenerPolicyValue::SameOrigin || current_document->opener_policy().value == OpenerPolicyValue::SameOriginPlusCOEP)
&& !current_document->origin().is_same_origin(relevant_settings_object(*current_document).top_level_origin.value())) {
// 1. Set noopener to true.
no_opener = TokenizedFeature::NoOpener::Yes;
// 2. Set name to "_blank".
name = "_blank"sv;
// 3. Set windowType to "new with no opener".
window_type = WindowType::NewWithNoOpener;
}
// NOTE: In the presence of an opener policy,
// nested documents that are cross-origin with their top-level browsing context's active document always set noopener to true.
// 5. Let targetName be the empty string.
String target_name;
// 6. If name is not an ASCII case-insensitive match for "_blank", then set targetName to name.
if (!name.equals_ignoring_ascii_case("_blank"sv))
target_name = MUST(String::from_utf8(name));
auto create_new_traversable_closure = [this, no_opener, target_name, activate_tab, window_features](GC::Ptr<BrowsingContext> opener) -> GC::Ref<Navigable> {
auto hints = WebViewHints::from_tokenised_features(window_features.value_or({}), traversable_navigable()->page());
auto [page, window_handle] = traversable_navigable()->page().client().page_did_request_new_web_view(activate_tab, hints, no_opener);
auto traversable = TraversableNavigable::create_a_new_top_level_traversable(*page, opener, target_name).release_value_but_fixme_should_propagate_errors();
page->set_top_level_traversable(traversable);
traversable->set_window_handle(window_handle);
return traversable;
};
auto create_new_traversable = GC::create_function(heap(), move(create_new_traversable_closure));
// 7. If noopener is true, then set chosen to the result of creating a new top-level traversable given null and targetName.
if (no_opener == TokenizedFeature::NoOpener::Yes) {
chosen = create_new_traversable->function()(nullptr);
}
// 8. Otherwise:
else {
// 1. Set chosen to the result of creating a new top-level traversable given currentNavigable's active browsing context, targetName, and currentNavigable.
// FIXME: "and currentNavigable", which is the openerNavigableForWebDriver parameter.
chosen = create_new_traversable->function()(active_browsing_context());
// 2. If sandboxingFlagSet's sandboxed navigation browsing context flag is set,
// then set chosen's active browsing context's one permitted sandboxed navigator to currentNavigable's active browsing context.
if (has_flag(sandboxing_flag_set, SandboxingFlagSet::SandboxedNavigation))
chosen->active_browsing_context()->set_the_one_permitted_sandboxed_navigator(active_browsing_context());
}
// 9. If sandboxingFlagSet's sandbox propagates to auxiliary browsing contexts flag is set,
// then all the flags that are set in sandboxingFlagSet must be set in chosen's active browsing context's popup sandboxing flag set.
if (has_flag(sandboxing_flag_set, SandboxingFlagSet::SandboxPropagatesToAuxiliaryBrowsingContexts))
chosen->active_browsing_context()->set_popup_sandboxing_flag_set(chosen->active_browsing_context()->popup_sandboxing_flag_set() | sandboxing_flag_set);
// 10. Set chosen's is created by web content to true.
as<TraversableNavigable>(*chosen).set_is_created_by_web_content(true);
}
// --> If the user agent has been configured such that in this instance it will choose currentNavigable
else if (false) { // FIXME: When is this the case?
// Set chosen to current.
chosen = *this;
}
// --> If the user agent has been configured such that in this instance it will not find a navigable
else if (false) { // FIXME: When is this the case?
// Do nothing.
}
}
// 9. Return chosen and windowType
return { chosen.ptr(), window_type };
}
// https://html.spec.whatwg.org/multipage/document-sequences.html#find-a-navigable-by-target-name
GC::Ptr<Navigable> Navigable::find_a_navigable_by_target_name(StringView name)
{
// 1. Let currentDocument be currentNavigable's active document.
auto& current_document = *active_document();
// 2. Let sourceSnapshotParams be the result of snapshotting source snapshot params given currentDocument.
auto source_snapshot_params = current_document.snapshot_source_snapshot_params();
// 3. Let subtreesToSearch be an implementation-defined choice of one of the following:
// - « currentNavigable's traversable navigable, currentNavigable »
// - the inclusive ancestor navigables of currentDocument
// FIXME: Decide which to use, or wait until the spec picks one.
auto subtrees_to_search = current_document.inclusive_ancestor_navigables();
// 4. For each subtreeToSearch of subtreesToSearch, in reverse order:
for (auto const& subtree_to_search : subtrees_to_search.in_reverse()) {
// 1. Let documentToSearch be subtreeToSearch's active document.
auto& document_to_search = *subtree_to_search->active_document();
// 2. For each navigable of the inclusive descendant navigables of documentToSearch:
for (auto const& navigable : document_to_search.inclusive_descendant_navigables()) {
// 1. If currentNavigable is not allowed by sandboxing to navigate navigable given sourceSnapshotParams, then optionally continue.
if (!allowed_by_sandboxing_to_navigate(*navigable, source_snapshot_params))
continue;
// 2. If navigable's target name is name, then return navigable.
if (navigable->target_name() == name)
return *navigable;
}
}
// 5. Let currentTopLevelBrowsingContext be currentNavigable's active browsing context's top-level browsing context.
auto& current_top_level_browsing_context = *active_browsing_context()->top_level_browsing_context();
// 6. Let group be currentTopLevelBrowsingContext's group.
auto* group = current_top_level_browsing_context.group();
// 7. For each topLevelBrowsingContext of group's browsing context set, in an implementation-defined order (the user agent should pick a consistent ordering, such as the most recently opened, most recently focused, or more closely related):
for (auto const& top_level_browsing_context : group->browsing_context_set()) {
// 1. If currentTopLevelBrowsingContext is topLevelBrowsingContext, then continue.
if (&current_top_level_browsing_context == top_level_browsing_context)
continue;
// 2. Let documentToSearch be topLevelBrowsingContext's active document.
auto* document_to_search = top_level_browsing_context->active_document();
// 3. For each navigable of the inclusive descendant navigables of documentToSearch:
for (auto const& navigable : document_to_search->inclusive_descendant_navigables()) {
// 1. If currentNavigable's active browsing context is not familiar with navigable's active browsing context, then continue.
if (!active_browsing_context()->is_familiar_with(*navigable->active_browsing_context()))
continue;
// 2. If currentNavigable is not allowed by sandboxing to navigate navigable given sourceSnapshotParams, then optionally continue.
if (!allowed_by_sandboxing_to_navigate(*navigable, source_snapshot_params))
continue;
// 3. If navigable's target name is name, then return navigable.
if (navigable->target_name() == name)
return *navigable;
}
}
// 8. Return null.
return nullptr;
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#getting-session-history-entries
Vector<GC::Ref<SessionHistoryEntry>>& Navigable::get_session_history_entries() const
{
// 1. Let traversable be navigable's traversable navigable.
auto traversable = traversable_navigable();
// FIXME 2. Assert: this is running within traversable's session history traversal queue.
// 3. If navigable is traversable, return traversable's session history entries.
if (this == traversable)
return traversable->session_history_entries();
// 4. Let docStates be an empty ordered set of document states.
Vector<GC::Ptr<DocumentState>> doc_states;
// 5. For each entry of traversable's session history entries, append entry's document state to docStates.
for (auto& entry : traversable->session_history_entries())
doc_states.append(entry->document_state());
// 6. For each docState of docStates:
while (!doc_states.is_empty()) {
auto doc_state = doc_states.take_first();
// 1. For each nestedHistory of docState's nested histories:
for (auto& nested_history : doc_state->nested_histories()) {
// 1. If nestedHistory's id equals navigable's id, return nestedHistory's entries.
if (nested_history.id == id())
return nested_history.entries;
// 2. For each entry of nestedHistory's entries, append entry's document state to docStates.
for (auto& entry : nested_history.entries)
doc_states.append(entry->document_state());
}
}
VERIFY_NOT_REACHED();
}
// https://html.spec.whatwg.org/multipage/browsers.html#determining-navigation-params-policy-container
static GC::Ref<PolicyContainer> determine_navigation_params_policy_container(URL::URL const& response_url,
GC::Heap& heap,
GC::Ptr<PolicyContainer> history_policy_container,
GC::Ptr<PolicyContainer> initiator_policy_container,
GC::Ptr<PolicyContainer> parent_policy_container,
GC::Ptr<PolicyContainer> response_policy_container)
{
// 1. If historyPolicyContainer is not null, then:
if (history_policy_container) {
// FIXME: 1. Assert: responseURL requires storing the policy container in history.
// 2. Return a clone of historyPolicyContainer.
return history_policy_container->clone(heap);
}
// 2. If responseURL is about:srcdoc, then:
if (response_url == URL::about_srcdoc()) {
// 1. Assert: parentPolicyContainer is not null.
VERIFY(parent_policy_container);
// 2. Return a clone of parentPolicyContainer.
return parent_policy_container->clone(heap);
}
// 3. If responseURL is local and initiatorPolicyContainer is not null, then return a clone of initiatorPolicyContainer.
if (Fetch::Infrastructure::is_local_url(response_url) && initiator_policy_container)
return initiator_policy_container->clone(heap);
// 4. If responsePolicyContainer is not null, then return responsePolicyContainer.
// FIXME: File a spec issue to say "a clone of" here for consistency
if (response_policy_container)
return response_policy_container->clone(heap);
// 5. Return a new policy container.
return heap.allocate<PolicyContainer>(heap);
}
// https://html.spec.whatwg.org/multipage/browsers.html#obtain-coop
static OpenerPolicy obtain_an_opener_policy(GC::Ref<Fetch::Infrastructure::Response>, Fetch::Infrastructure::Request::ReservedClientType const& reserved_client)
{
// 1. Let policy be a new opener policy.
OpenerPolicy policy = {};
// AD-HOC: We don't yet setup environments in all cases
if (!reserved_client)
return policy;
auto& reserved_environment = *reserved_client;
// 2. If reservedEnvironment is a non-secure context, then return policy.
if (is_non_secure_context(reserved_environment))
return policy;
// FIXME: We don't yet have the technology to extract structured data from Fetch headers
// FIXME: 3. Let parsedItem be the result of getting a structured field value given `Cross-Origin-Opener-Policy` and "item" from response's header list.
// FIXME: 4. If parsedItem is not null, then:
// FIXME: nested steps...
// FIXME: 5. Set parsedItem to the result of getting a structured field value given `Cross-Origin-Opener-Policy-Report-Only` and "item" from response's header list.
// FIXME: 6. If parsedItem is not null, then:
// FIXME: nested steps...
// 7. Return policy.
return policy;
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#attempt-to-create-a-non-fetch-scheme-document
static GC::Ptr<DOM::Document> attempt_to_create_a_non_fetch_scheme_document(NonFetchSchemeNavigationParams const& params)
{
// 1. Let url be navigationParams's URL.
auto const& url = params.url;
// 2. Let navigable be navigationParams's navigable.
[[maybe_unused]] auto navigable = params.navigable;
// 3. FIXME: If url is to be handled using a mechanism that does not affect navigable, e.g., because url's scheme is
// handled externally, then:
if (false) {
// 1. FIXME: Hand-off to external software given url, navigable, navigationParams's target snapshot sandboxing flags,
// navigationParams's source snapshot has transient activation, and navigationParams's initiator origin.
// 2. Return null.
return {};
}
// 4. FIXME: Handle url by displaying some sort of inline content, e.g., an error message because the specified scheme is
// not one of the supported protocols, or an inline prompt to allow the user to select a registered handler for
// the given scheme. Return the result of displaying the inline content given navigable, navigationParams's id,
// navigationParams's navigation timing type, and navigationParams's user involvement.
dbgln("FIXME: Don't know how to navigate to {}", url);
return {};
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#create-navigation-params-from-a-srcdoc-resource
static GC::Ref<NavigationParams> create_navigation_params_from_a_srcdoc_resource(GC::Ptr<SessionHistoryEntry> entry, GC::Ptr<Navigable> navigable, TargetSnapshotParams const& target_snapshot_params, UserNavigationInvolvement user_involvement, Optional<String> navigation_id)
{
auto& vm = navigable->vm();
VERIFY(navigable->active_window());
auto& realm = navigable->active_window()->realm();
// 1. Let documentResource be entry's document state's resource.
auto document_resource = entry->document_state()->resource();
VERIFY(document_resource.has<String>());
// 2. Let response be a new response with
// URL: about:srcdoc
// header list: (`Content-Type`, `text/html`)
// body: the UTF-8 encoding of documentResource, as a body
auto response = Fetch::Infrastructure::Response::create(vm);
response->url_list().append(URL::about_srcdoc());
auto header = Fetch::Infrastructure::Header::from_string_pair("Content-Type"sv, "text/html"sv);
response->header_list()->append(move(header));
response->set_body(Fetch::Infrastructure::byte_sequence_as_body(realm, document_resource.get<String>().bytes()));
// 3. Let responseOrigin be the result of determining the origin given response's URL, targetSnapshotParams's sandboxing flags, and entry's document state's origin.
auto response_origin = determine_the_origin(response->url(), target_snapshot_params.sandboxing_flags, entry->document_state()->origin());
// 4. Let coop be a new opener policy.
OpenerPolicy coop = {};
// 5. Let coopEnforcementResult be a new opener policy enforcement result with
// url: response's URL
// origin: responseOrigin
// opener policy: coop
OpenerPolicyEnforcementResult coop_enforcement_result {
.url = *response->url(),
.origin = response_origin,
.opener_policy = coop
};
// 6. Let policyContainer be the result of determining navigation params policy container given response's URL,
// entry's document state's history policy container, null, navigable's container document's policy container, and null.
GC::Ptr<PolicyContainer> history_policy_container = entry->document_state()->history_policy_container().visit(
[](GC::Ref<PolicyContainer> const& c) -> GC::Ptr<PolicyContainer> { return c; },
[](DocumentState::Client) -> GC::Ptr<PolicyContainer> { return {}; });
GC::Ptr<PolicyContainer> policy_container;
if (navigable->container()) {
// NOTE: Specification assumes that only navigables corresponding to iframes can be navigated to about:srcdoc.
// We also use srcdoc to implement load_html() for top level navigables so we need to null check container
// because it might be null.
policy_container = determine_navigation_params_policy_container(*response->url(), realm.heap(), history_policy_container, {}, navigable->container_document()->policy_container(), {});
} else {
policy_container = realm.heap().allocate<PolicyContainer>(realm.heap());
}
// 7. Return a new navigation params, with
// id: navigationId
// navigable: navigable
// request: null
// response: response
// fetch controller: null
// commit early hints: null
// COOP enforcement result: coopEnforcementResult
// reserved environment: null
// origin: responseOrigin
// policy container: policyContainer
// final sandboxing flag set: targetSnapshotParams's sandboxing flags
// opener policy: coop
// FIXME: navigation timing type: navTimingType
// about base URL: entry's document state's about base URL
// user involvement: userInvolvement
return vm.heap().allocate<NavigationParams>(
move(navigation_id),
navigable,
nullptr,
response,
nullptr,
nullptr,
move(coop_enforcement_result),
nullptr,
move(response_origin),
*policy_container,
target_snapshot_params.sandboxing_flags,
move(coop),
entry->document_state()->about_base_url(),
user_involvement);
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#create-navigation-params-by-fetching
static WebIDL::ExceptionOr<Navigable::NavigationParamsVariant> create_navigation_params_by_fetching(GC::Ptr<SessionHistoryEntry> entry, GC::Ptr<Navigable> navigable, SourceSnapshotParams const& source_snapshot_params, TargetSnapshotParams const& target_snapshot_params, ContentSecurityPolicy::Directives::Directive::NavigationType csp_navigation_type, UserNavigationInvolvement user_involvement, Optional<String> navigation_id)
{
auto& vm = navigable->vm();
VERIFY(navigable->active_window());
auto& realm = navigable->active_window()->realm();
auto& active_document = *navigable->active_document();
// FIXME: 1. Assert: this is running in parallel.
// 2. Let documentResource be entry's document state's resource.
auto document_resource = entry->document_state()->resource();
// 3. Let request be a new request, with
// url: entry's URL
// client: sourceSnapshotParams's fetch client
// destination: "document"
// credentials mode: "include"
// use-URL-credentials flag: set
// redirect mode: "manual"
// replaces client id: navigable's active document's relevant settings object's id
// mode: "navigate"
// referrer: entry's document state's request referrer
// referrer policy: entry's document state's request referrer policy
// policy container: sourceSnapshotParams's source policy container
auto request = Fetch::Infrastructure::Request::create(vm);
request->set_url(entry->url());
request->set_client(source_snapshot_params.fetch_client);
request->set_destination(Fetch::Infrastructure::Request::Destination::Document);
request->set_credentials_mode(Fetch::Infrastructure::Request::CredentialsMode::Include);
request->set_use_url_credentials(true);
request->set_redirect_mode(Fetch::Infrastructure::Request::RedirectMode::Manual);
request->set_replaces_client_id(active_document.relevant_settings_object().id);
request->set_mode(Fetch::Infrastructure::Request::Mode::Navigate);
request->set_referrer(entry->document_state()->request_referrer());
request->set_policy_container(source_snapshot_params.source_policy_container);
// 4. If navigable is a top-level traversable, then set request's top-level navigation initiator origin to entry's
// document state's initiator origin.
if (navigable->top_level_traversable()->parent() == nullptr)
request->set_top_level_navigation_initiator_origin(entry->document_state()->origin());
// 5. If request's client is null:
if (request->client() == nullptr) {
// Note: This only occurs in the case of a browser UI-initiated navigation.
// 1. Set request's origin to a new opaque origin.
request->set_origin(URL::Origin::create_opaque());
// 2. Set request's service-workers mode to "all".
request->set_service_workers_mode(Fetch::Infrastructure::Request::ServiceWorkersMode::All);
// 3. Set request's referrer to "no-referrer".
request->set_referrer(Fetch::Infrastructure::Request::Referrer::NoReferrer);
}
// 6. If documentResource is a POST resource:
if (auto* post_resource = document_resource.get_pointer<POSTResource>()) {
// 1. Set request's method to `POST`.
request->set_method(TRY_OR_THROW_OOM(vm, ByteBuffer::copy("POST"sv.bytes())));
// 2. Set request's body to documentResource's request body.
request->set_body(document_resource.get<POSTResource>().request_body.value());
// 3. Set `Content-Type` to documentResource's request content-type in request's header list.
auto request_content_type = [&]() {
switch (post_resource->request_content_type) {
case POSTResource::RequestContentType::ApplicationXWWWFormUrlencoded:
return "application/x-www-form-urlencoded"sv;
case POSTResource::RequestContentType::MultipartFormData:
return "multipart/form-data"sv;
case POSTResource::RequestContentType::TextPlain:
return "text/plain"sv;
default:
VERIFY_NOT_REACHED();
}
}();
StringBuilder request_content_type_buffer;
if (!post_resource->request_content_type_directives.is_empty()) {
request_content_type_buffer.append(request_content_type);
for (auto const& directive : post_resource->request_content_type_directives)
request_content_type_buffer.appendff("; {}={}", directive.type, directive.value);
request_content_type = request_content_type_buffer.string_view();
}
auto header = Fetch::Infrastructure::Header::from_string_pair("Content-Type"sv, request_content_type);
request->header_list()->append(move(header));
}
// 7. If entry's document state's reload pending is true, then set request's reload-navigation flag.
if (entry->document_state()->reload_pending())
request->set_reload_navigation(true);
// 8. Otherwise, if entry's document state's ever populated is true, then set request's history-navigation flag.
else if (entry->document_state()->ever_populated())
request->set_history_navigation(true);
// 9. If sourceSnapshotParams's has transient activation is true, then set request's user-activation to true.
if (source_snapshot_params.has_transient_activation)
request->set_user_activation(true);
// 10. If navigable's container is non-null:
if (navigable->container() != nullptr) {
// 1. If the navigable's container has a browsing context scope origin, then set request's origin to that browsing context scope origin.
// FIXME: From "browsing context scope origin": This definition is broken and needs investigation to see what it was intended to express: see issue #4703.
// The referenced issue suggests that it is a no-op to retrieve the browsing context scope origin.
// 2. Set request's destination to navigable's container's local name.
// FIXME: Are there other container types? If so, we need a helper here
Web::Fetch::Infrastructure::Request::Destination destination = is<HTMLIFrameElement>(*navigable->container()) ? Web::Fetch::Infrastructure::Request::Destination::IFrame
: Web::Fetch::Infrastructure::Request::Destination::Object;
request->set_destination(destination);
// 3. If sourceSnapshotParams's fetch client is navigable's container document's relevant settings object,
// then set request's initiator type to navigable's container's local name.
// NOTE: This ensure that only container-initiated navigations are reported to resource timing.
if (source_snapshot_params.fetch_client == &navigable->container_document()->relevant_settings_object()) {
// FIXME: Are there other container types? If so, we need a helper here
Web::Fetch::Infrastructure::Request::InitiatorType initiator_type = is<HTMLIFrameElement>(*navigable->container()) ? Web::Fetch::Infrastructure::Request::InitiatorType::IFrame
: Web::Fetch::Infrastructure::Request::InitiatorType::Object;
request->set_initiator_type(initiator_type);
}
}
// 11. Let response be null.
// NOTE: We use a heap-allocated cell to hold the response pointer because the processResponse callback below
// might use it after this stack is freed.
auto response_holder = ResponseHolder::create(vm);
// 12. Let responseOrigin be null.
Optional<URL::Origin> response_origin;
// 13. Let fetchController be null.
GC::Ptr<Fetch::Infrastructure::FetchController> fetch_controller = nullptr;
// 14. Let coopEnforcementResult be a new opener policy enforcement result, with
// - url: navigable's active document's URL
// - origin: navigable's active document's origin
// - opener policy: navigable's active document's opener policy
// - current context is navigation source: true if navigable's active document's origin is same origin with
// entry's document state's initiator origin otherwise false
OpenerPolicyEnforcementResult coop_enforcement_result = {
.url = active_document.url(),
.origin = active_document.origin(),
.opener_policy = active_document.opener_policy(),
.current_context_is_navigation_source = entry->document_state()->initiator_origin().has_value() && active_document.origin().is_same_origin(*entry->document_state()->initiator_origin())
};
// 15. Let finalSandboxFlags be an empty sandboxing flag set.
SandboxingFlagSet final_sandbox_flags = {};
// 16. Let responsePolicyContainer be null.
GC::Ptr<PolicyContainer> response_policy_container = {};
// 17. Let responseCOOP be a new opener policy.
OpenerPolicy response_coop = {};
// 18. Let locationURL be null.
ErrorOr<Optional<URL::URL>> location_url { OptionalNone {} };
// 19. Let currentURL be request's current URL.
URL::URL current_url = request->current_url();
// 20. Let commitEarlyHints be null.
Function<void(DOM::Document&)> commit_early_hints = nullptr;
// 21. While true:
while (true) {
// 1. If request's reserved client is not null and currentURL's origin is not the same as request's reserved client's creation URL's origin, then:
if (request->reserved_client() && !current_url.origin().is_same_origin(request->reserved_client()->creation_url.origin())) {
// 1. Run the environment discarding steps for request's reserved client.
request->reserved_client()->discard_environment();
// 2. Set request's reserved client to null.
request->set_reserved_client(nullptr);
// 3. Set commitEarlyHints to null.
commit_early_hints = nullptr;
}
// 2. If request's reserved client is null, then:
if (!request->reserved_client()) {
// 1. Let topLevelCreationURL be currentURL.
Optional<URL::URL> top_level_creation_url = current_url;
// 2. Let topLevelOrigin be null.
Optional<URL::Origin> top_level_origin;
// 3. If navigable is not a top-level traversable, then:
if (!navigable->is_top_level_traversable()) {
// 1. Let parentEnvironment be navigable's parent's active document's relevant settings object.
auto& parent_environment = navigable->parent()->active_document()->relevant_settings_object();
// 2. Set topLevelCreationURL to parentEnvironment's top-level creation URL.
top_level_creation_url = parent_environment.top_level_creation_url;
// 3. Set topLevelOrigin to parentEnvironment's top-level origin.
top_level_origin = parent_environment.top_level_origin;
}
// 4. Set request's reserved client to a new environment whose id is a unique opaque string,
// target browsing context is navigable's active browsing context,
// creation URL is currentURL,
// top-level creation URL is topLevelCreationURL,
// and top-level origin is topLevelOrigin.
// FIXME: Make this a proper unique opaque string.
static int next_id = 1;
auto id_string = MUST(String::formatted("create-by-fetching-{}", next_id++));
request->set_reserved_client(realm.create<Environment>(id_string, current_url, top_level_creation_url, top_level_origin, navigable->active_browsing_context()));
}
// 3. If the result of should navigation request of type be blocked by Content Security Policy? given request and cspNavigationType is "Blocked", then set response to a network error and break. [CSP]
if (ContentSecurityPolicy::should_navigation_request_of_type_be_blocked_by_content_security_policy(request, csp_navigation_type) == ContentSecurityPolicy::Directives::Directive::Result::Blocked) {
response_holder->set_response(Fetch::Infrastructure::Response::network_error(vm, "Blocked by Content Security Policy"_string));
break;
}
// 4. Set response to null.
response_holder->set_response(nullptr);
// 5. If fetchController is null, then set fetchController to the result of fetching request,
// with processEarlyHintsResponse set to processEarlyHintsResponseas defined below, processResponse
// set to processResponse as defined below, and useParallelQueue set to true.
if (!fetch_controller) {
// FIXME: Let processEarlyHintsResponse be the following algorithm given a response earlyResponse:
// Let processResponse be the following algorithm given a response fetchedResponse:
auto process_response = [response_holder](GC::Ref<Fetch::Infrastructure::Response> fetch_response) {
// 1. Set response to fetchedResponse.
response_holder->set_response(fetch_response);
};
fetch_controller = TRY(Fetch::Fetching::fetch(
realm,
request,
Fetch::Infrastructure::FetchAlgorithms::create(vm,
{
.process_request_body_chunk_length = {},
.process_request_end_of_body = {},
.process_early_hints_response = {},
.process_response = move(process_response),
.process_response_end_of_body = {},
.process_response_consume_body = {},
}),
Fetch::Fetching::UseParallelQueue::Yes));
}
// 6. Otherwise, process the next manual redirect for fetchController.
else {
fetch_controller->process_next_manual_redirect();
}
// 7. Wait until either response is non-null, or navigable's ongoing navigation changes to no longer equal navigationId.
HTML::main_thread_event_loop().spin_until(GC::create_function(vm.heap(), [navigation_id, navigable, response_holder]() {
if (response_holder->response() != nullptr)
return true;
if (navigation_id.has_value() && (!navigable->ongoing_navigation().has<String>() || navigable->ongoing_navigation().get<String>() != *navigation_id))
return true;
return false;
}));
// If the latter condition occurs, then abort fetchController, and return. Otherwise, proceed onward.
if (navigation_id.has_value() && (!navigable->ongoing_navigation().has<String>() || navigable->ongoing_navigation().get<String>() != *navigation_id)) {
fetch_controller->abort(realm, {});
return Navigable::NullOrError {};
}
// 8. If request's body is null, then set entry's document state's resource to null.
if (!request->body().has<Empty>()) {
entry->document_state()->set_resource(Empty {});
}
// 9. Set responsePolicyContainer to the result of creating a policy container from a fetch response given response and request's reserved client.
response_policy_container = create_a_policy_container_from_a_fetch_response(realm.heap(), *response_holder->response(), request->reserved_client());
// 10. Set finalSandboxFlags to the union of targetSnapshotParams's sandboxing flags and responsePolicyContainer's CSP list's CSP-derived sandboxing flags.
final_sandbox_flags = target_snapshot_params.sandboxing_flags | response_policy_container->csp_list->csp_derived_sandboxing_flags();
// 11. Set responseOrigin to the result of determining the origin given response's URL, finalSandboxFlags, and entry's document state's initiator origin.
response_origin = determine_the_origin(response_holder->response()->url(), final_sandbox_flags, entry->document_state()->initiator_origin());
// 12. If navigable is a top-level traversable, then:
if (navigable->is_top_level_traversable()) {
// 1. Set responseCOOP to the result of obtaining an opener policy given response and request's reserved client.
response_coop = obtain_an_opener_policy(*response_holder->response(), request->reserved_client());
// FIXME: 2. Set coopEnforcementResult to the result of enforcing the response's opener policy given navigable's active browsing context,
// response's URL, responseOrigin, responseCOOP, coopEnforcementResult and request's referrer.
// FIXME: 3. If finalSandboxFlags is not empty and responseCOOP's value is not "unsafe-none", then set response to an appropriate network error and break.
// NOTE: This results in a network error as one cannot simultaneously provide a clean slate to a response
// using opener policy and sandbox the result of navigating to that response.
}
// 13. FIXME If response is not a network error, navigable is a child navigable, and the result of performing a cross-origin resource policy check
// with navigable's container document's origin, navigable's container document's relevant settings object, request's destination, response,
// and true is blocked, then set response to a network error and break.
// NOTE: Here we're running the cross-origin resource policy check against the parent navigable rather than navigable itself
// This is because we care about the same-originness of the embedded content against the parent context, not the navigation source.
// 14. Set locationURL to response's location URL given currentURL's fragment.
location_url = response_holder->response()->location_url(current_url.fragment());
// 15. If locationURL is failure or null, then break.
if (location_url.is_error() || !location_url.value().has_value()) {
break;
}
// 16. Assert: locationURL is a URL.
// 17. Set entry's classic history API state to StructuredSerializeForStorage(null).
entry->set_classic_history_api_state(MUST(structured_serialize_for_storage(vm, JS::js_null())));
// 18. Let oldDocState be entry's document state.
auto old_doc_state = entry->document_state();
// 19. Set entry's document state to a new document state, with
// history policy container: a clone of the oldDocState's history policy container if it is non-null; null otherwise
// request referrer: oldDocState's request referrer
// request referrer policy: oldDocState's request referrer policy
// origin: oldDocState's origin
// resource: oldDocState's resource
// ever populated: oldDocState's ever populated
// navigable target name: oldDocState's navigable target name
auto new_document_state = navigable->heap().allocate<DocumentState>();
new_document_state->set_history_policy_container(old_doc_state->history_policy_container());
new_document_state->set_request_referrer(old_doc_state->request_referrer());
new_document_state->set_request_referrer_policy(old_doc_state->request_referrer_policy());
new_document_state->set_origin(old_doc_state->origin());
new_document_state->set_resource(old_doc_state->resource());
new_document_state->set_ever_populated(old_doc_state->ever_populated());
new_document_state->set_navigable_target_name(old_doc_state->navigable_target_name());
entry->set_document_state(new_document_state);
// 20. If locationURL's scheme is not an HTTP(S) scheme, then:
if (!Fetch::Infrastructure::is_http_or_https_scheme(location_url.value()->scheme())) {
// 1. Set entry's document state's resource to null.
entry->document_state()->set_resource(Empty {});
// 2. Break.
break;
}
// 21. Set currentURL to locationURL.
current_url = location_url.value().value();
// 22. Set entry's URL to currentURL.
entry->set_url(current_url);
}
// 22. If locationURL is a URL whose scheme is not a fetch scheme, then return a new non-fetch scheme navigation params, with
if (!location_url.is_error() && location_url.value().has_value() && !Fetch::Infrastructure::is_fetch_scheme(location_url.value().value().scheme())) {
// - id: navigationId
// - navigable: navigable
// - URL: locationURL
// - target snapshot sandboxing flags: targetSnapshotParams's sandboxing flags
// - source snapshot has transient activation: sourceSnapshotParams's has transient activation
// - initiator origin: responseOrigin
// FIXME: - navigation timing type: navTimingType
// - user involvement: userInvolvement
return vm.heap().allocate<NonFetchSchemeNavigationParams>(
navigation_id,
navigable,
location_url.release_value().value(),
target_snapshot_params.sandboxing_flags,
source_snapshot_params.has_transient_activation,
move(*response_origin),
user_involvement);
}
// 23. If any of the following are true:
// - response is a network error;
// - locationURL is failure; or
// - locationURL is a URL whose scheme is a fetch scheme
// then return null.
if (response_holder->response()->is_network_error()) {
// AD-HOC: We pass the error message if we have one in NullWithError
if (response_holder->response()->network_error_message().has_value())
return response_holder->response()->network_error_message().value();
else
return Navigable::NullOrError {};
} else if (location_url.is_error() || (location_url.value().has_value() && Fetch::Infrastructure::is_fetch_scheme(location_url.value().value().scheme())))
return Navigable::NullOrError {};
// 24. Assert: locationURL is null and response is not a network error.
VERIFY(!location_url.value().has_value());
VERIFY(!response_holder->response()->is_network_error());
// 25. Let resultPolicyContainer be the result of determining navigation params policy container given response's URL,
// entry's document state's history policy container, sourceSnapshotParams's source policy container, null, and responsePolicyContainer.
GC::Ptr<PolicyContainer> history_policy_container = entry->document_state()->history_policy_container().visit(
[](GC::Ref<PolicyContainer> const& c) -> GC::Ptr<PolicyContainer> { return c; },
[](DocumentState::Client) -> GC::Ptr<PolicyContainer> { return {}; });
auto result_policy_container = determine_navigation_params_policy_container(*response_holder->response()->url(), realm.heap(), history_policy_container, source_snapshot_params.source_policy_container, {}, response_policy_container);
// 26. If navigable's container is an iframe, and response's timing allow passed flag is set,
// then set navigable's container's pending resource-timing start time to null.
if (navigable->container() && response_holder->response()->timing_allow_passed()) {
if (auto* iframe_element = as_if<HTML::HTMLIFrameElement>(*navigable->container()))
iframe_element->set_pending_resource_start_time({});
}
// 27. Return a new navigation params, with
// id: navigationId
// navigable: navigable
// request: request
// response: response
// fetch controller: fetchController
// commit early hints: commitEarlyHints
// opener policy: responseCOOP
// reserved environment: request's reserved client
// origin: responseOrigin
// policy container: resultPolicyContainer
// final sandboxing flag set: finalSandboxFlags
// COOP enforcement result: coopEnforcementResult
// FIXME: navigation timing type: navTimingType
// about base URL: entry's document state's about base URL
// user involvement: userInvolvement
return vm.heap().allocate<NavigationParams>(
navigation_id,
navigable,
request,
*response_holder->response(),
fetch_controller,
move(commit_early_hints),
coop_enforcement_result,
request->reserved_client(),
*response_origin,
result_policy_container,
final_sandbox_flags,
response_coop,
entry->document_state()->about_base_url(),
user_involvement);
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#populating-a-session-history-entry
WebIDL::ExceptionOr<void> Navigable::populate_session_history_entry_document(
GC::Ptr<SessionHistoryEntry> entry,
SourceSnapshotParams const& source_snapshot_params,
TargetSnapshotParams const& target_snapshot_params,
UserNavigationInvolvement user_involvement,
Optional<String> navigation_id,
Navigable::NavigationParamsVariant navigation_params,
ContentSecurityPolicy::Directives::Directive::NavigationType csp_navigation_type,
bool allow_POST,
GC::Ptr<GC::Function<void()>> completion_steps)
{
// AD-HOC: Not in the spec but subsequent steps will fail if the navigable doesn't have an active window.
if (!active_window())
return {};
// FIXME: 1. Assert: this is running in parallel.
// 2. Assert: if navigationParams is non-null, then navigationParams's response is non-null.
if (!navigation_params.has<NullOrError>())
VERIFY(navigation_params.has<GC::Ref<NavigationParams>>() && navigation_params.get<GC::Ref<NavigationParams>>()->response);
// 3. Let documentResource be entry's document state's resource.
auto document_resource = entry->document_state()->resource();
// 4. If navigationParams is null, then:
if (navigation_params.has<NullOrError>()) {
// 1. If documentResource is a string, then set navigationParams to the result of creating navigation params
// from a srcdoc resource given entry, navigable, targetSnapshotParams, userInvolvement, navigationId, and
// navTimingType.
if (document_resource.has<String>()) {
navigation_params = create_navigation_params_from_a_srcdoc_resource(entry, this, target_snapshot_params, user_involvement, navigation_id);
}
// 2. Otherwise, if all of the following are true:
// - entry's URL's scheme is a fetch scheme; and
// - documentResource is null, or allowPOST is true and documentResource's request body is not failure,
// (FIXME: check if request body is not failure)
// then set navigationParams to the result of creating navigation params by fetching given entry, navigable,
// sourceSnapshotParams, targetSnapshotParams, cspNavigationType, userInvolvement, navigationId, and
// navTimingType.
else if (Fetch::Infrastructure::is_fetch_scheme(entry->url().scheme()) && (document_resource.has<Empty>() || allow_POST)) {
navigation_params = TRY(create_navigation_params_by_fetching(entry, this, source_snapshot_params, target_snapshot_params, csp_navigation_type, user_involvement, navigation_id));
}
// 3. Otherwise, if entry's URL's scheme is not a fetch scheme, then set navigationParams to a new non-fetch
// scheme navigation params, with:
else if (!Fetch::Infrastructure::is_fetch_scheme(entry->url().scheme())) {
// - id: navigationId
// - navigable: navigable
// - URL: entry's URL
// - target snapshot sandboxing flags: targetSnapshotParams's sandboxing flags
// - source snapshot has transient activation: sourceSnapshotParams's has transient activation
// - initiator origin: entry's document state's initiator origin
// FIXME: - navigation timing type: navTimingType
// - user involvement: userInvolvement
navigation_params = vm().heap().allocate<NonFetchSchemeNavigationParams>(
navigation_id,
this,
entry->url(),
target_snapshot_params.sandboxing_flags,
source_snapshot_params.has_transient_activation,
*entry->document_state()->initiator_origin(),
user_involvement);
}
}
// AD-HOC: Not in the spec but subsequent steps will fail if the navigable doesn't have an active window.
if (!active_window())
return {};
// 5. Queue a global task on the navigation and traversal task source, given navigable's active window, to run these steps:
queue_global_task(Task::Source::NavigationAndTraversal, *active_window(), GC::create_function(heap(), [this, entry, navigation_params = move(navigation_params), navigation_id, user_involvement, completion_steps, csp_navigation_type]() mutable {
// NOTE: This check is not in the spec but we should not continue navigation if navigable has been destroyed.
if (has_been_destroyed())
return;
// 1. If navigable's ongoing navigation no longer equals navigationId, then run completionSteps and abort these steps.
if (navigation_id.has_value() && (!ongoing_navigation().has<String>() || ongoing_navigation().get<String>() != *navigation_id)) {
if (completion_steps)
completion_steps->function()();
return;
}
// 2. Let saveExtraDocumentState be true.
auto saveExtraDocumentState = true;
// 3. If navigationParams is a non-fetch scheme navigation params, then:
if (navigation_params.has<GC::Ref<NonFetchSchemeNavigationParams>>()) {
// 1. Set entry's document state's document to the result of running attempt to create a non-fetch scheme
// document given navigationParams.
// NOTE: This can result in setting entry's document state's document to null, e.g., when handing-off to
// external software.
entry->document_state()->set_document(attempt_to_create_a_non_fetch_scheme_document(navigation_params.get<GC::Ref<NonFetchSchemeNavigationParams>>()));
if (entry->document()) {
entry->document_state()->set_ever_populated(true);
}
// 2. Set saveExtraDocumentState to false.
saveExtraDocumentState = false;
}
// 4. Otherwise, if any of the following are true:
// - navigationParams is null;
// - the result of should navigation response to navigation request of type in target be blocked by Content Security Policy? given navigationParams's request, navigationParams's response, navigationParams's policy container's CSP list, cspNavigationType, and navigable is "Blocked";
// - FIXME: navigationParams's reserved environment is non-null and the result of checking a navigation response's adherence to its embedder policy given navigationParams's response, navigable, and navigationParams's policy container's embedder policy is false; or
// - the result of checking a navigation response's adherence to `X-Frame-Options` given navigationParams's response, navigable, navigationParams's policy container's CSP list, and navigationParams's origin is false,
// then:
else if (navigation_params.visit(
[](NullOrError) { return true; },
[this, csp_navigation_type](GC::Ref<NavigationParams> navigation_params) {
auto csp_result = ContentSecurityPolicy::should_navigation_response_to_navigation_request_of_type_in_target_be_blocked_by_content_security_policy(navigation_params->request, *navigation_params->response, navigation_params->policy_container->csp_list, csp_navigation_type, *this);
if (csp_result == ContentSecurityPolicy::Directives::Directive::Result::Blocked)
return true;
// FIXME: Pass in navigationParams's policy container's CSP list
return !check_a_navigation_responses_adherence_to_x_frame_options(navigation_params->response, this, navigation_params->policy_container->csp_list, navigation_params->origin);
},
[](GC::Ref<NonFetchSchemeNavigationParams>) { return false; })) {
// 1. Set entry's document state's document to the result of creating a document for inline content that doesn't have a DOM, given navigable, null, navTimingType, and userInvolvement. The inline content should indicate to the user the sort of error that occurred.
auto error_message = navigation_params.has<NullOrError>() ? navigation_params.get<NullOrError>().value_or("Unknown error"_string) : "The request was denied."_string;
auto error_html = load_error_page(entry->url(), error_message).release_value_but_fixme_should_propagate_errors();
entry->document_state()->set_document(create_document_for_inline_content(this, navigation_id, user_involvement, [this, error_html](auto& document) {
auto parser = HTML::HTMLParser::create(document, error_html, "utf-8"sv);
document.set_url(URL::about_error());
parser->run();
// NOTE: Once the page has been set up, the user agent must act as if it had stopped parsing.
// FIXME: Directly calling parser->the_end results in a deadlock, because it waits for the warning image to load.
// However the response is never processed when parser->the_end is called.
// Queuing a global task is a workaround for now.
queue_a_task(Task::Source::Unspecified, HTML::main_thread_event_loop(), document, GC::create_function(heap(), [&document]() {
HTMLParser::the_end(document);
}));
}));
// 2. Make document unsalvageable given entry's document state's document and "navigation-failure".
entry->document()->make_unsalvageable("navigation-failure"_string);
// 3. Set saveExtraDocumentState to false.
saveExtraDocumentState = false;
// 4. If navigationParams is not null, then:
if (!navigation_params.has<NullOrError>()) {
// 1. Run the environment discarding steps for navigationParams's reserved environment.
navigation_params.visit(
[](GC::Ref<NavigationParams> const& it) {
it->reserved_environment->discard_environment();
},
[](auto const&) {});
// FIXME: 2. Invoke WebDriver BiDi navigation failed with navigable and a new WebDriver BiDi navigation status whose id is navigationId, status is "canceled", and url is navigationParams's response's URL.
}
}
// FIXME: 5. Otherwise, if navigationParams's response has a `Content-Disposition` header specifying the attachment
// disposition type, then:
else if (false) {
}
// 6. Otherwise, if navigationParams's response's status is not 204 and is not 205, then set entry's document state's document to the result of
// loading a document given navigationParams, sourceSnapshotParams, and entry's document state's initiator origin.
else if (auto const& response = navigation_params.get<GC::Ref<NavigationParams>>()->response; response->status() != 204 && response->status() != 205) {
auto document = load_document(navigation_params.get<GC::Ref<NavigationParams>>());
entry->document_state()->set_document(document);
}
// 7. If entry's document state's document is not null, then:
if (entry->document()) {
// 1. Set entry's document state's ever populated to true.
entry->document_state()->set_ever_populated(true);
// 2. If saveExtraDocumentState is true:
if (saveExtraDocumentState) {
// 1. Let document be entry's document state's document.
auto document = entry->document();
// 2. Set entry's document state's origin to document's origin.
entry->document_state()->set_origin(document->origin());
// 3. If document's URL requires storing the policy container in history, then:
if (url_requires_storing_the_policy_container_in_history(document->url())) {
// 1. Assert: navigationParams is a navigation params (i.e., neither null nor a non-fetch scheme navigation params).
VERIFY(navigation_params.has<GC::Ref<NavigationParams>>());
// 2. Set entry's document state's history policy container to navigationParams's policy container.
entry->document_state()->set_history_policy_container(GC::Ref { *navigation_params.get<GC::Ref<NavigationParams>>()->policy_container });
}
}
// 3. If entry's document state's request referrer is "client", and navigationParams is a navigation params (i.e., neither null nor a non-fetch scheme navigation params), then:
if (entry->document_state()->request_referrer() == Fetch::Infrastructure::Request::Referrer::Client
&& (!navigation_params.has<NullOrError>() && navigation_params.has<GC::Ref<NonFetchSchemeNavigationParams>>())) {
// 1. Assert: navigationParams's request is not null.
VERIFY(navigation_params.has<GC::Ref<NavigationParams>>() && navigation_params.get<GC::Ref<NavigationParams>>()->request);
// 2. Set entry's document state's request referrer to navigationParams's request's referrer.
entry->document_state()->set_request_referrer(navigation_params.get<GC::Ref<NavigationParams>>()->request->referrer());
}
}
// 8. Run completionSteps.
if (completion_steps)
completion_steps->function()();
}));
return {};
}
WebIDL::ExceptionOr<void> Navigable::navigate(NavigateParams params)
{
// AD-HOC: Not in the spec but subsequent steps will fail if the navigable doesn't have an active window.
if (!active_window())
return {};
auto source_document = params.source_document;
auto exceptions_enabled = params.exceptions_enabled;
auto& active_document = *this->active_document();
auto& realm = active_document.realm();
auto& page_client = active_document.page().client();
// AD-HOC: If we are not able to continue in this process, request a new process from the UI.
if (is_top_level_traversable() && !page_client.is_url_suitable_for_same_process_navigation(active_document.url(), params.url)) {
page_client.request_new_process_for_navigation(params.url);
return {};
}
// 2. Let sourceSnapshotParams be the result of snapshotting source snapshot params given sourceDocument.
auto source_snapshot_params = source_document->snapshot_source_snapshot_params();
// 3. Let initiatorOriginSnapshot be sourceDocument's origin.
auto initiator_origin_snapshot = source_document->origin();
// 4. Let initiatorBaseURLSnapshot be sourceDocument's document base URL.
auto initiator_base_url_snapshot = source_document->base_url();
// 5. If sourceDocument's node navigable is not allowed by sandboxing to navigate navigable given sourceSnapshotParams, then:
if (!source_document->navigable()->allowed_by_sandboxing_to_navigate(*this, source_snapshot_params)) {
// 1. If exceptionsEnabled is true, then throw a "SecurityError" DOMException.
if (exceptions_enabled) {
return WebIDL::SecurityError::create(realm, "Source document's node navigable is not allowed to navigate"_utf16);
}
// 2 Return.
return {};
}
if (m_pending_navigations.is_empty() && params.url.equals(URL::about_blank())) {
begin_navigation(move(params));
return {};
}
if (!m_has_session_history_entry_and_ready_for_navigation) {
m_pending_navigations.append(move(params));
return {};
}
begin_navigation(move(params));
return {};
}
// To navigate a navigable navigable to a URL url using an optional Document-or-null sourceDocument (default null),
// with an optional POST resource, string, or null documentResource (default null),
// an optional response-or-null response (default null), an optional boolean exceptionsEnabled (default false),
// an optional NavigationHistoryBehavior historyHandling (default "auto"),
// an optional serialized state-or-null navigationAPIState (default null),
// an optional entry list or null formDataEntryList (default null),
// an optional referrer policy referrerPolicy (default the empty string),
// an optional user navigation involvement userInvolvement (default "none"),
// an optional Element sourceElement (default null),
// and an optional boolean initialInsertion (default false):
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#navigate
void Navigable::begin_navigation(NavigateParams params)
{
// AD-HOC: Not in the spec but subsequent steps will fail if the navigable doesn't have an active window.
if (!active_window())
return;
auto const& url = params.url;
auto source_document = params.source_document;
auto const& document_resource = params.document_resource;
auto response = params.response;
auto history_handling = params.history_handling;
auto const& navigation_api_state = params.navigation_api_state;
auto const& form_data_entry_list = params.form_data_entry_list;
auto referrer_policy = params.referrer_policy;
auto user_involvement = params.user_involvement;
auto source_element = params.source_element;
auto initial_insertion = params.initial_insertion;
auto& active_document = *this->active_document();
auto& vm = this->vm();
// 1. Let cspNavigationType be "form-submission" if formDataEntryList is non-null; otherwise "other".
auto csp_navigation_type = form_data_entry_list.has_value() ? ContentSecurityPolicy::Directives::Directive::NavigationType::FormSubmission : ContentSecurityPolicy::Directives::Directive::NavigationType::Other;
// 2. Let sourceSnapshotParams be the result of snapshotting source snapshot params given sourceDocument.
auto source_snapshot_params = source_document->snapshot_source_snapshot_params();
// 3. Let initiatorOriginSnapshot be a new opaque origin.
auto initiator_origin_snapshot = URL::Origin::create_opaque();
// 4. Let initiatorBaseURLSnapshot be about:blank.
auto initiator_base_url_snapshot = URL::about_blank();
// FIXME: 5. If sourceDocument is null:
if (false) {
// 1. Assert: userInvolvement is "browser UI".
VERIFY(user_involvement == UserNavigationInvolvement::BrowserUI);
// 2. If url's scheme is "javascript", then set initiatorOriginSnapshot to navigable's active document's origin.
if (url.scheme() == "javascript"sv)
initiator_origin_snapshot = active_document.origin();
}
// 6. Otherwise:
else {
// 1. Assert: userInvolvement is not "browser UI".
// FIXME: We currently crash if we do this! Uncomment once other places are fixed to handle browser UI navigation.
// VERIFY(user_involvement != UserNavigationInvolvement::BrowserUI);
// 2. If sourceDocument's node navigable is not allowed by sandboxing to navigate navigable given sourceSnapshotParams:
// NB: This step is handled in Navigable::navigate()
// 3. Set initiatorOriginSnapshot to sourceDocument's origin.
initiator_origin_snapshot = source_document->origin();
// 4. Set initiatorBaseURLSnapshot to sourceDocument's document base URL.
initiator_base_url_snapshot = source_document->base_url();
}
// 5. If sourceDocument's node navigable is not allowed by sandboxing to navigate navigable given sourceSnapshotParams, then:
// NOTE: This step is handled in Navigable::navigate()
// 7. Let navigationId be the result of generating a random UUID.
String navigation_id = MUST(Crypto::generate_random_uuid());
// FIXME: 8. If the surrounding agent is equal to navigable's active document's relevant agent, then continue these steps.
// Otherwise, queue a global task on the navigation and traversal task source given navigable's active window to continue these steps.
// 9. If navigable's active document's unload counter is greater than 0,
// then invoke WebDriver BiDi navigation failed with navigable and a WebDriver BiDi navigation status whose id
// is navigationId, status is "canceled", and url is url, and return.
if (active_document.unload_counter() > 0) {
// FIXME: invoke WebDriver BiDi navigation failed with navigable and a WebDriver BiDi navigation status whose id
// is navigationId, status is "canceled", and url is url
return;
}
// 10. Let container be navigable's container.
auto& container = m_container;
// 11. If container is an iframe element and will lazy load element steps given container returns true,
// then stop intersection-observing a lazy loading element container and set container's lazy load resumption steps to null.
if (container && container->is_html_iframe_element()) {
auto& iframe_element = static_cast<HTMLIFrameElement&>(*container);
if (iframe_element.will_lazy_load_element()) {
iframe_element.document().stop_intersection_observing_a_lazy_loading_element(iframe_element);
iframe_element.set_lazy_load_resumption_steps(nullptr);
}
}
// 12. If historyHandling is "auto", then:
if (history_handling == Bindings::NavigationHistoryBehavior::Auto) {
// FIXME: Fix spec typo targetNavigable --> navigable
// 1. If url equals navigable's active document's URL,
// and initiatorOriginSnapshot is same origin with targetNavigable's active document's origin,
// then set historyHandling to "replace".
if (url == active_document.url() && initiator_origin_snapshot.is_same_origin(active_document.origin()))
history_handling = Bindings::NavigationHistoryBehavior::Replace;
// 2. Otherwise, set historyHandling to "push".
else
history_handling = Bindings::NavigationHistoryBehavior::Push;
}
// 13. If the navigation must be a replace given url and navigable's active document, then set historyHandling to "replace".
if (navigation_must_be_a_replace(url, active_document))
history_handling = Bindings::NavigationHistoryBehavior::Replace;
// 14. If all of the following are true:
// - documentResource is null;
// - response is null;
// - url equals navigable's active session history entry's URL with exclude fragments set to true; and
// - url's fragment is non-null,
// then:
if (document_resource.has<Empty>()
&& !response
&& url.equals(active_session_history_entry()->url(), URL::ExcludeFragment::Yes)
&& url.fragment().has_value()) {
// 1. Navigate to a fragment given navigable, url, historyHandling, userInvolvement, sourceElement, navigationAPIState, and navigationId.
navigate_to_a_fragment(url, to_history_handling_behavior(history_handling), user_involvement, source_element, navigation_api_state, navigation_id);
// 2. Return.
return;
}
// 15. If navigable's parent is non-null, then set navigable's is delaying load events to true.
if (parent() != nullptr)
set_delaying_load_events(true);
// 16. Let targetSnapshotParams be the result of snapshotting target snapshot params given navigable.
[[maybe_unused]] auto target_snapshot_params = snapshot_target_snapshot_params();
// FIXME: 17. Invoke WebDriver BiDi navigation started with navigable and a new WebDriver BiDi navigation status whose id is navigationId, status is "pending", and url is url.
// 18. If navigable's ongoing navigation is "traversal", then:
if (ongoing_navigation().has<Traversal>()) {
// FIXME: 1. Invoke WebDriver BiDi navigation failed with navigable and a new WebDriver BiDi navigation status whose id is navigationId, status is "canceled", and url is url.
// 2. Return.
return;
}
// 19. Set the ongoing navigation for navigable to navigationId.
set_ongoing_navigation(navigation_id);
// 20. If url's scheme is "javascript", then:
if (url.scheme() == "javascript"sv) {
// 1. Queue a global task on the navigation and traversal task source given navigable's active window to navigate to a javascript: URL given navigable, url, historyHandling, sourceSnapshotParams, initiatorOriginSnapshot, userInvolvement, cspNavigationType, and initialInsertion.
VERIFY(active_window());
queue_global_task(Task::Source::NavigationAndTraversal, *active_window(), GC::create_function(heap(), [this, url, history_handling, source_snapshot_params, initiator_origin_snapshot, user_involvement, csp_navigation_type, initial_insertion, navigation_id] {
navigate_to_a_javascript_url(url, to_history_handling_behavior(history_handling), source_snapshot_params, initiator_origin_snapshot, user_involvement, csp_navigation_type, initial_insertion, navigation_id);
}));
// 2. Return.
return;
}
// 21. If all of the following are true:
// - userInvolvement is not "browser UI";
// - navigable's active document's origin is same origin-domain with sourceDocument's origin;
// - navigable's active document's is initial about:blank is false; and
// - url's scheme is a fetch scheme
// then:
if (user_involvement != UserNavigationInvolvement::BrowserUI && active_document.origin().is_same_origin_domain(source_document->origin()) && !active_document.is_initial_about_blank() && Fetch::Infrastructure::is_fetch_scheme(url.scheme())) {
// 1. Let navigation be navigable's active window's navigation API.
VERIFY(active_window());
auto navigation = active_window()->navigation();
// 2. Let entryListForFiring be formDataEntryList if documentResource is a POST resource; otherwise, null.
auto entry_list_for_firing = [&]() -> Optional<Vector<XHR::FormDataEntry>> {
if (document_resource.has<POSTResource>())
return form_data_entry_list;
return {};
}();
// 3. Let navigationAPIStateForFiring be navigationAPIState if navigationAPIState is not null;
// otherwise, StructuredSerializeForStorage(undefined).
auto navigation_api_state_for_firing = navigation_api_state.value_or(MUST(structured_serialize_for_storage(vm, JS::js_undefined())));
// 4. Let continue be the result of firing a push/replace/reload navigate event at navigation
// with navigationType set to historyHandling, isSameDocument set to false, userInvolvement set to userInvolvement,
// sourceElement set to sourceElement, formDataEntryList set to entryListForFiring, destinationURL set to url,
// and navigationAPIState set to navigationAPIStateForFiring.
auto navigation_type = [](Bindings::NavigationHistoryBehavior history_handling) {
switch (history_handling) {
case Bindings::NavigationHistoryBehavior::Push:
return Bindings::NavigationType::Push;
case Bindings::NavigationHistoryBehavior::Replace:
return Bindings::NavigationType::Replace;
case Bindings::NavigationHistoryBehavior::Auto:
default:
VERIFY_NOT_REACHED();
}
}(history_handling);
auto continue_ = navigation->fire_a_push_replace_reload_navigate_event(navigation_type, url, false, user_involvement, source_element, entry_list_for_firing, navigation_api_state_for_firing);
// 5. If continue is false, then return.
if (!continue_)
return;
}
// AD-HOC: Tell the UI that we started loading.
if (is_top_level_traversable()) {
active_browsing_context()->page().client().page_did_start_loading(url, false);
}
// FIXME: 22. If sourceDocument is navigable's container document, then reserve deferred fetch quota for navigable's container given url's origin.
// 23. In parallel, run these steps:
Platform::EventLoopPlugin::the().deferred_invoke(GC::create_function(heap(), [this, source_snapshot_params, target_snapshot_params, csp_navigation_type, document_resource, url, navigation_id, referrer_policy, initiator_origin_snapshot, response, history_handling, initiator_base_url_snapshot, user_involvement] {
// AD-HOC: Not in the spec but subsequent steps will fail if the navigable doesn't have an active window.
if (!active_window()) {
set_delaying_load_events(false);
return;
}
// 1. Let unloadPromptCanceled be the result of checking if unloading is user-canceled for navigable's active document's inclusive descendant navigables.
auto unload_prompt_canceled = traversable_navigable()->check_if_unloading_is_canceled(this->active_document()->inclusive_descendant_navigables());
// 2. If unloadPromptCanceled is not "continue", or navigable's ongoing navigation is no longer navigationId:
if (unload_prompt_canceled != TraversableNavigable::CheckIfUnloadingIsCanceledResult::Continue || !ongoing_navigation().has<String>() || ongoing_navigation().get<String>() != navigation_id) {
// FIXME: 1. Invoke WebDriver BiDi navigation failed with navigable and a new WebDriver BiDi navigation status whose id is navigationId, status is "canceled", and url is url.
// 2. Abort these steps.
set_delaying_load_events(false);
return;
}
// AD-HOC: Not in the spec but subsequent steps will fail if the navigable doesn't have an active window.
if (!active_window()) {
set_delaying_load_events(false);
return;
}
// 3. Queue a global task on the navigation and traversal task source given navigable's active window to abort a document and its descendants given navigable's active document.
queue_global_task(Task::Source::NavigationAndTraversal, *active_window(), GC::create_function(heap(), [this] {
VERIFY(this->active_document());
this->active_document()->abort_a_document_and_its_descendants();
}));
// 4. Let documentState be a new document state with
// request referrer policy: referrerPolicy
// initiator origin: initiatorOriginSnapshot
// resource: documentResource
// navigable target name: navigable's target name
GC::Ref<DocumentState> document_state = *heap().allocate<DocumentState>();
document_state->set_request_referrer_policy(referrer_policy);
document_state->set_initiator_origin(initiator_origin_snapshot);
document_state->set_resource(document_resource);
document_state->set_navigable_target_name(target_name());
// 5. If url matches about:blank or is about:srcdoc, then:
// FIXME: Is calling url_matches_about_srcdoc() correct? https://github.com/whatwg/html/issues/10900
if (url_matches_about_blank(url) || url_matches_about_srcdoc(url)) {
// AD-HOC: document_resource cannot have an Empty if the url is about:srcdoc since we rely on document_resource
// having a String to call create_navigation_params_from_a_srcdoc_resource
if (url_matches_about_srcdoc(url) && document_resource.has<Empty>()) {
document_state->set_resource({ String {} });
}
// 1. Set documentState's origin to initiatorOriginSnapshot.
document_state->set_origin(document_state->initiator_origin());
// 2. Set documentState's about base URL to initiatorBaseURLSnapshot.
document_state->set_about_base_url(initiator_base_url_snapshot);
}
// 6. Let historyEntry be a new session history entry, with its URL set to url and its document state set to documentState.
GC::Ref<SessionHistoryEntry> history_entry = *heap().allocate<SessionHistoryEntry>();
history_entry->set_url(url);
history_entry->set_document_state(document_state);
// 7. Let navigationParams be null.
NavigationParamsVariant navigation_params = Navigable::NullOrError {};
// FIXME: 8. If response is non-null:
if (response) {
}
// 9. Attempt to populate the history entry's document for historyEntry, given navigable, "navigate",
// sourceSnapshotParams, targetSnapshotParams, userInvolvement, navigationId, navigationParams,
// cspNavigationType, with allowPOST set to true and completionSteps set to the following step:
populate_session_history_entry_document(history_entry, source_snapshot_params, target_snapshot_params, user_involvement, navigation_id, navigation_params, csp_navigation_type, true, GC::create_function(heap(), [this, history_entry, history_handling, navigation_id, user_involvement] {
// 1. Append session history traversal steps to navigable's traversable to finalize a cross-document navigation given navigable, historyHandling, userInvolvement, and historyEntry.
traversable_navigable()->append_session_history_traversal_steps(GC::create_function(heap(), [this, history_entry, history_handling, navigation_id, user_involvement] {
if (this->has_been_destroyed()) {
// NOTE: This check is not in the spec but we should not continue navigation if navigable has been destroyed.
set_delaying_load_events(false);
return;
}
if (this->ongoing_navigation() != navigation_id) {
// NOTE: This check is not in the spec but we should not continue navigation if ongoing navigation id has changed.
set_delaying_load_events(false);
return;
}
finalize_a_cross_document_navigation(*this, to_history_handling_behavior(history_handling), user_involvement, history_entry);
}));
})).release_value_but_fixme_should_propagate_errors();
}));
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#navigate-fragid
void Navigable::navigate_to_a_fragment(URL::URL const& url, HistoryHandlingBehavior history_handling, UserNavigationInvolvement user_involvement, GC::Ptr<DOM::Element> source_element, Optional<SerializationRecord> navigation_api_state, String navigation_id)
{
// 1. Let navigation be navigable's active window's navigation API.
VERIFY(active_window());
auto navigation = active_window()->navigation();
// 2. Let destinationNavigationAPIState be navigable's active session history entry's navigation API state.
// 3. If navigationAPIState is not null, then set destinationNavigationAPIState to navigationAPIState.
auto destination_navigation_api_state = navigation_api_state.has_value() ? *navigation_api_state : active_session_history_entry()->navigation_api_state();
// 4. Let continue be the result of firing a push/replace/reload navigate event at navigation with navigationType
// set to historyHandling, isSameDocument set to true, userInvolvement set to userInvolvement, sourceElement set
// to sourceElement, destinationURL set to url, and navigationAPIState set to destinationNavigationAPIState.
auto navigation_type = history_handling == HistoryHandlingBehavior::Push ? Bindings::NavigationType::Push : Bindings::NavigationType::Replace;
bool const continue_ = navigation->fire_a_push_replace_reload_navigate_event(navigation_type, url, true, user_involvement, source_element, {}, destination_navigation_api_state);
// 5. If continue is false, then return.
if (!continue_)
return;
// 6. Let historyEntry be a new session history entry, with
// URL: url
// document state: navigable's active session history entry's document state
// navigation API state: destinationNavigationAPIState
// scroll restoration mode: navigable's active session history entry's scroll restoration mode
GC::Ref<SessionHistoryEntry> history_entry = heap().allocate<SessionHistoryEntry>();
history_entry->set_url(url);
history_entry->set_document_state(active_session_history_entry()->document_state());
history_entry->set_navigation_api_state(destination_navigation_api_state);
history_entry->set_scroll_restoration_mode(active_session_history_entry()->scroll_restoration_mode());
// 7. Let entryToReplace be navigable's active session history entry if historyHandling is "replace", otherwise null.
auto entry_to_replace = history_handling == HistoryHandlingBehavior::Replace ? active_session_history_entry() : nullptr;
// 8. Let history be navigable's active document's history object.
auto history = active_document()->history();
// 9. Let scriptHistoryIndex be history's index.
auto script_history_index = history->m_index;
// 10. Let scriptHistoryLength be history's length.
auto script_history_length = history->m_length;
// 11. If historyHandling is "push", then:
if (history_handling == HistoryHandlingBehavior::Push) {
// 1. Set history's state to null.
history->set_state(JS::js_null());
// 2. Increment scriptHistoryIndex.
script_history_index++;
// 3. Set scriptHistoryLength to scriptHistoryIndex + 1.
script_history_length = script_history_index + 1;
}
// 12. Set navigable's active session history entry to historyEntry.
m_active_session_history_entry = history_entry;
// 13. Update document for history step application given navigable's active document, historyEntry, true, scriptHistoryIndex, and scriptHistoryLength.
// AD HOC: Skip updating the navigation api entries twice here
active_document()->update_for_history_step_application(*history_entry, true, script_history_length, script_history_index, navigation_type, {}, {}, false);
// 14. Update the navigation API entries for a same-document navigation given navigation, historyEntry, and historyHandling.
navigation->update_the_navigation_api_entries_for_a_same_document_navigation(history_entry, navigation_type);
// 15. Scroll to the fragment given navigable's active document.
// FIXME: Specification doesn't say when document url needs to update during fragment navigation
active_document()->set_url(url);
active_document()->scroll_to_the_fragment();
// 16. Let traversable be navigable's traversable navigable.
auto traversable = traversable_navigable();
// 17. Append the following session history synchronous navigation steps involving navigable to traversable:
traversable->append_session_history_synchronous_navigation_steps(*this, GC::create_function(heap(), [this, traversable, history_entry, entry_to_replace, navigation_id, history_handling, user_involvement] {
// 1. Finalize a same-document navigation given traversable, navigable, historyEntry, entryToReplace, historyHandling, and userInvolvement.
finalize_a_same_document_navigation(*traversable, *this, history_entry, entry_to_replace, history_handling, user_involvement);
// FIXME: 2. Invoke WebDriver BiDi fragment navigated with navigable and a new WebDriver BiDi
// navigation status whose id is navigationId, url is url, and status is "complete".
(void)navigation_id;
}));
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#evaluate-a-javascript:-url
// https://whatpr.org/html/9893/browsing-the-web.html#evaluate-a-javascript:-url
GC::Ptr<DOM::Document> Navigable::evaluate_javascript_url(URL::URL const& url, URL::Origin const& new_document_origin, UserNavigationInvolvement user_involvement, String navigation_id)
{
auto& vm = this->vm();
VERIFY(active_window());
auto& realm = active_window()->realm();
// 1. Let urlString be the result of running the URL serializer on url.
auto url_string = url.serialize();
// 2. Let encodedScriptSource be the result of removing the leading "javascript:" from urlString.
auto encoded_script_source = url_string.bytes_as_string_view().substring_view(11);
// 3. Let scriptSource be the UTF-8 decoding of the percent-decoding of encodedScriptSource.
auto script_source = URL::percent_decode(encoded_script_source);
// 4. Let settings be targetNavigable's active document's relevant settings object.
auto& settings = active_document()->relevant_settings_object();
// 5. Let baseURL be settings's API base URL.
auto base_url = settings.api_base_url();
// 6. Let script be the result of creating a classic script given scriptSource, settings's realm, baseURL, and the default classic script fetch options.
auto script = HTML::ClassicScript::create("(javascript url)", script_source, settings.realm(), base_url);
// 7. Let evaluationStatus be the result of running the classic script script.
auto evaluation_status = script->run();
// 8. Let result be null.
String result;
// 9. If evaluationStatus is a normal completion, and evaluationStatus.[[Value]] is a String, then set result to evaluationStatus.[[Value]].
if (evaluation_status.type() == JS::Completion::Type::Normal && evaluation_status.value().is_string()) {
result = evaluation_status.value().as_string().utf8_string();
} else {
// 10. Otherwise, return null.
return nullptr;
}
// 11. Let response be a new response with
// URL: targetNavigable's active document's URL
// header list: «(`Content-Type`, `text/html;charset=utf-8`)»
// body: the UTF-8 encoding of result, as a body
auto response = Fetch::Infrastructure::Response::create(vm);
response->url_list().append(active_document()->url());
auto header = Fetch::Infrastructure::Header::from_string_pair("Content-Type"sv, "text/html"sv);
response->header_list()->append(move(header));
response->set_body(Fetch::Infrastructure::byte_sequence_as_body(realm, result.bytes()));
// 12. Let policyContainer be targetNavigable's active document's policy container.
auto const& policy_container = active_document()->policy_container();
// 13. Let finalSandboxFlags be policyContainer's CSP list's CSP-derived sandboxing flags.
auto final_sandbox_flags = policy_container->csp_list->csp_derived_sandboxing_flags();
// 14. Let coop be targetNavigable's active document's opener policy.
auto const& coop = active_document()->opener_policy();
// 15. Let coopEnforcementResult be a new opener policy enforcement result with
// url: url
// origin: newDocumentOrigin
// opener policy: coop
OpenerPolicyEnforcementResult coop_enforcement_result {
.url = url,
.origin = new_document_origin,
.opener_policy = coop,
};
// 16. Let navigationParams be a new navigation params, with
// id: navigationId
// navigable: targetNavigable
// request: null
// response: response
// fetch controller: null
// commit early hints: null
// COOP enforcement result: coopEnforcementResult
// reserved environment: null
// origin: newDocumentOrigin
// policy container: policyContainer
// final sandboxing flag set: finalSandboxFlags
// opener policy: coop
// FIXME: navigation timing type: "navigate"
// about base URL: targetNavigable's active document's about base URL
// user involvement: userInvolvement
auto navigation_params = vm.heap().allocate<NavigationParams>(
navigation_id,
this,
nullptr,
response,
nullptr,
nullptr,
move(coop_enforcement_result),
nullptr,
new_document_origin,
policy_container,
final_sandbox_flags,
coop,
active_document()->about_base_url(),
user_involvement);
// 17. Return the result of loading an HTML document given navigationParams.
return load_document(navigation_params);
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#navigate-to-a-javascript:-url
void Navigable::navigate_to_a_javascript_url(URL::URL const& url, HistoryHandlingBehavior history_handling, GC::Ref<SourceSnapshotParams> source_snapshot_params, URL::Origin const& initiator_origin, UserNavigationInvolvement user_involvement, ContentSecurityPolicy::Directives::Directive::NavigationType csp_navigation_type, InitialInsertion initial_insertion, String navigation_id)
{
auto& vm = this->vm();
// 1. Assert: historyHandling is "replace".
VERIFY(history_handling == HistoryHandlingBehavior::Replace);
// 2. Set the ongoing navigation for targetNavigable to null.
set_ongoing_navigation({});
// 3. If initiatorOrigin is not same origin-domain with targetNavigable's active document's origin, then return.
if (!initiator_origin.is_same_origin_domain(active_document()->origin()))
return;
// 4. Let request be a new request whose URL is url and whose policy container is sourceSnapshotParams's source policy container.
auto request = Fetch::Infrastructure::Request::create(vm);
request->set_url(url);
request->set_policy_container(source_snapshot_params->source_policy_container);
// AD-HOC: See https://github.com/whatwg/html/issues/4651, requires some investigation to figure out what we should be setting here.
request->set_client(source_snapshot_params->fetch_client);
// 5. If the result of should navigation request of type be blocked by Content Security Policy? given request and cspNavigationType is "Blocked", then return.
if (ContentSecurityPolicy::should_navigation_request_of_type_be_blocked_by_content_security_policy(request, csp_navigation_type) == ContentSecurityPolicy::Directives::Directive::Result::Blocked)
return;
// 6. Let newDocument be the result of evaluating a javascript: URL given targetNavigable, url, initiatorOrigin, and userInvolvement.
auto new_document = evaluate_javascript_url(url, initiator_origin, user_involvement, navigation_id);
// 7. If newDocument is null:
if (!new_document) {
// 1. If initialInsertion is true and targetNavigable's active document's is initial about:blank is true,
// then run the iframe load event steps given targetNavigable's container.
if (initial_insertion == InitialInsertion::Yes && active_document()->is_initial_about_blank()) {
run_iframe_load_event_steps(as<HTMLIFrameElement>(*container()));
}
// 2. Return.
// NOTE: In this case, some JavaScript code was executed, but no new Document was created, so we will not perform a navigation.
return;
}
// 8. Assert: initiatorOrigin is newDocument's origin.
VERIFY(initiator_origin == new_document->origin());
// 9. Let entryToReplace be targetNavigable's active session history entry.
auto entry_to_replace = active_session_history_entry();
// 10. Let oldDocState be entryToReplace's document state.
auto old_doc_state = entry_to_replace->document_state();
// 11. Let documentState be a new document state with
// document: newDocument
// history policy container: a clone of the oldDocState's history policy container if it is non-null; null otherwise
// request referrer: oldDocState's request referrer
// request referrer policy: oldDocState's request referrer policy
// initiator origin: initiatorOrigin
// origin: initiatorOrigin
// about base URL: oldDocState's about base URL
// resource: null
// ever populated: true
// navigable target name: oldDocState's navigable target name
GC::Ref<DocumentState> document_state = *heap().allocate<DocumentState>();
document_state->set_document(new_document);
document_state->set_history_policy_container(old_doc_state->history_policy_container());
document_state->set_request_referrer(old_doc_state->request_referrer());
document_state->set_request_referrer_policy(old_doc_state->request_referrer_policy());
document_state->set_initiator_origin(initiator_origin);
document_state->set_origin(initiator_origin);
document_state->set_about_base_url(old_doc_state->about_base_url());
document_state->set_ever_populated(true);
document_state->set_navigable_target_name(old_doc_state->navigable_target_name());
// 12. Let historyEntry be a new session history entry, with
// URL: entryToReplace's URL
// document state: documentState
GC::Ref<SessionHistoryEntry> history_entry = *heap().allocate<SessionHistoryEntry>();
history_entry->set_url(entry_to_replace->url());
history_entry->set_document_state(document_state);
// 13. Append session history traversal steps to targetNavigable's traversable to finalize a cross-document navigation with targetNavigable, historyHandling, userInvolvement, and historyEntry.
traversable_navigable()->append_session_history_traversal_steps(GC::create_function(heap(), [this, history_entry, history_handling, user_involvement] {
finalize_a_cross_document_navigation(*this, history_handling, user_involvement, history_entry);
}));
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#reload
void Navigable::reload(UserNavigationInvolvement user_involvement)
{
// 1. Set navigable's active session history entry's document state's reload pending to true.
active_session_history_entry()->document_state()->set_reload_pending(true);
// 2. Let traversable be navigable's traversable navigable.
auto traversable = traversable_navigable();
// 3. Append the following session history traversal steps to traversable:
traversable->append_session_history_traversal_steps(GC::create_function(heap(), [traversable, user_involvement] {
// 1. Apply the reload history step to traversable given userInvolvement.
traversable->apply_the_reload_history_step(user_involvement);
}));
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#the-navigation-must-be-a-replace
bool navigation_must_be_a_replace(URL::URL const& url, DOM::Document const& document)
{
return url.scheme() == "javascript"sv || document.is_initial_about_blank();
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#allowed-to-navigate
bool Navigable::allowed_by_sandboxing_to_navigate(Navigable const& target, SourceSnapshotParams const& source_snapshot_params)
{
auto& source = *this;
auto is_ancestor_of = [](Navigable const& a, Navigable const& b) {
for (auto parent = b.parent(); parent; parent = parent->parent()) {
if (parent.ptr() == &a)
return true;
}
return false;
};
// A navigable source is allowed by sandboxing to navigate a second navigable target,
// given a source snapshot params sourceSnapshotParams, if the following steps return true:
// 1. If source is target, then return true.
if (&source == &target)
return true;
// 2. If source is an ancestor of target, then return true.
if (is_ancestor_of(source, target))
return true;
// 3. If target is an ancestor of source, then:
if (is_ancestor_of(target, source)) {
// 1. If target is not a top-level traversable, then return true.
if (!target.is_top_level_traversable())
return true;
// 2. If sourceSnapshotParams's has transient activation is true, and sourceSnapshotParams's sandboxing flags's
// sandboxed top-level navigation with user activation browsing context flag is set, then return false.
if (source_snapshot_params.has_transient_activation && has_flag(source_snapshot_params.sandboxing_flags, SandboxingFlagSet::SandboxedTopLevelNavigationWithUserActivation))
return false;
// 3. If sourceSnapshotParams's has transient activation is false, and sourceSnapshotParams's sandboxing flags's
// sandboxed top-level navigation without user activation browsing context flag is set, then return false.
if (!source_snapshot_params.has_transient_activation && has_flag(source_snapshot_params.sandboxing_flags, SandboxingFlagSet::SandboxedTopLevelNavigationWithoutUserActivation))
return false;
// 4. Return true.
return true;
}
// 4. If target is a top-level traversable:
if (target.is_top_level_traversable()) {
// FIXME: 1. If source is the one permitted sandboxed navigator of target, then return true.
// 2. If sourceSnapshotParams's sandboxing flags's sandboxed navigation browsing context flag is set, then return false.
if (has_flag(source_snapshot_params.sandboxing_flags, SandboxingFlagSet::SandboxedNavigation))
return false;
// 3. Return true.
return true;
}
// 5. If sourceSnapshotParams's sandboxing flags's sandboxed navigation browsing context flag is set, then return false.
// 6. Return true.
return !has_flag(source_snapshot_params.sandboxing_flags, SandboxingFlagSet::SandboxedNavigation);
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#snapshotting-target-snapshot-params
TargetSnapshotParams Navigable::snapshot_target_snapshot_params()
{
// To snapshot target snapshot params given a navigable targetNavigable, return a new target snapshot params
// with sandboxing flags set to the result of determining the creation sandboxing flags given targetNavigable's
// active browsing context and targetNavigable's container.
return { determine_the_creation_sandboxing_flags(*active_browsing_context(), container()) };
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#finalize-a-cross-document-navigation
void finalize_a_cross_document_navigation(GC::Ref<Navigable> navigable, HistoryHandlingBehavior history_handling, UserNavigationInvolvement user_involvement, GC::Ref<SessionHistoryEntry> history_entry)
{
// NOTE: This is not in the spec but we should not navigate destroyed navigable.
if (navigable->has_been_destroyed())
return;
// 1. FIXME: Assert: this is running on navigable's traversable navigable's session history traversal queue.
// 2. Set navigable's is delaying load events to false.
navigable->set_delaying_load_events(false);
// 3. If historyEntry's document is null, then return.
if (!history_entry->document())
return;
// 4. If all of the following are true:
// - navigable's parent is null;
// - historyEntry's document's browsing context is not an auxiliary browsing context whose opener browsing context is non-null; and
// - historyEntry's document's origin is not navigable's active document's origin
// then set historyEntry's document state's navigable target name to the empty string.
if (navigable->parent() == nullptr
&& !(history_entry->document()->browsing_context()->is_auxiliary() && history_entry->document()->browsing_context()->opener_browsing_context() != nullptr)
&& history_entry->document()->origin() != navigable->active_document()->origin()) {
history_entry->document_state()->set_navigable_target_name(String {});
}
// 5. Let entryToReplace be navigable's active session history entry if historyHandling is "replace", otherwise null.
auto entry_to_replace = history_handling == HistoryHandlingBehavior::Replace ? navigable->active_session_history_entry() : nullptr;
// 6. Let traversable be navigable's traversable navigable.
auto traversable = navigable->traversable_navigable();
// 7. Let targetStep be null.
int target_step;
// 8. Let targetEntries be the result of getting session history entries for navigable.
auto& target_entries = navigable->get_session_history_entries();
// 9. If entryToReplace is null, then:
if (entry_to_replace == nullptr) {
// 1. Clear the forward session history of traversable.
traversable->clear_the_forward_session_history();
// 2. Set targetStep to traversable's current session history step + 1.
target_step = traversable->current_session_history_step() + 1;
// 3. Set historyEntry's step to targetStep.
history_entry->set_step(target_step);
// 4. Append historyEntry to targetEntries.
target_entries.append(history_entry);
} else {
// 1. Replace entryToReplace with historyEntry in targetEntries.
*(target_entries.find(*entry_to_replace)) = history_entry;
// 2. Set historyEntry's step to entryToReplace's step.
history_entry->set_step(entry_to_replace->step());
// 3. If historyEntry's document state's origin is same origin with entryToReplace's document state's origin,
// then set historyEntry's navigation API key to entryToReplace's navigation API key.
if (history_entry->document_state()->origin().has_value() && entry_to_replace->document_state()->origin().has_value() && history_entry->document_state()->origin()->is_same_origin(*entry_to_replace->document_state()->origin())) {
history_entry->set_navigation_api_key(entry_to_replace->navigation_api_key());
}
// 4. Set targetStep to traversable's current session history step.
target_step = traversable->current_session_history_step();
}
// 10. Apply the push/replace history step targetStep to traversable given historyHandling and userInvolvement.
traversable->apply_the_push_or_replace_history_step(target_step, history_handling, user_involvement, TraversableNavigable::SynchronousNavigation::No);
// AD-HOC: If we're inside a navigable container, let's trigger a relayout in the container document.
// This allows size negotiation between the containing document and SVG documents to happen.
if (auto container = navigable->container()) {
if (auto layout_node = container->layout_node())
layout_node->set_needs_layout_update(DOM::SetNeedsLayoutReason::FinalizeACrossDocumentNavigation);
}
}
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#url-and-history-update-steps
void perform_url_and_history_update_steps(DOM::Document& document, URL::URL new_url, Optional<SerializationRecord> serialized_data, HistoryHandlingBehavior history_handling)
{
// 1. Let navigable be document's node navigable.
auto navigable = document.navigable();
// 2. Let activeEntry be navigable's active session history entry.
auto active_entry = navigable->active_session_history_entry();
// 3. Let newEntry be a new session history entry, with
// URL: newURL
// serialized state: if serializedData is not null, serializedData; otherwise activeEntry's classic history API state
// document state: activeEntry's document state
// scroll restoration mode: activeEntry's scroll restoration mode
// FIXME: persisted user state: activeEntry's persisted user state
GC::Ref<SessionHistoryEntry> new_entry = document.heap().allocate<SessionHistoryEntry>();
new_entry->set_url(new_url);
new_entry->set_classic_history_api_state(serialized_data.value_or(active_entry->classic_history_api_state()));
new_entry->set_document_state(active_entry->document_state());
new_entry->set_scroll_restoration_mode(active_entry->scroll_restoration_mode());
// 4. If document's is initial about:blank is true, then set historyHandling to "replace".
if (document.is_initial_about_blank()) {
history_handling = HistoryHandlingBehavior::Replace;
}
// 5. Let entryToReplace be activeEntry if historyHandling is "replace", otherwise null.
auto entry_to_replace = history_handling == HistoryHandlingBehavior::Replace ? active_entry : nullptr;
// 6. If historyHandling is "push", then:
if (history_handling == HistoryHandlingBehavior::Push) {
// 1. Increment document's history object's index.
document.history()->m_index++;
// 2. Set document's history object's length to its index + 1.
document.history()->m_length = document.history()->m_index + 1;
}
// If serializedData is not null, then restore the history object state given document and newEntry.
if (serialized_data.has_value())
document.restore_the_history_object_state(new_entry);
// 8. Set the URL given document to newURL.
document.set_url(new_url);
// 9. Set document's latest entry to newEntry.
document.set_latest_entry(new_entry);
// 10. Set navigable's active session history entry to newEntry.
navigable->set_active_session_history_entry(new_entry);
// 11. Update the navigation API entries for a same-document navigation given document's relevant global object's navigation API, newEntry, and historyHandling.
auto& relevant_global_object = as<Window>(HTML::relevant_global_object(document));
auto navigation_type = history_handling == HistoryHandlingBehavior::Push ? Bindings::NavigationType::Push : Bindings::NavigationType::Replace;
relevant_global_object.navigation()->update_the_navigation_api_entries_for_a_same_document_navigation(new_entry, navigation_type);
// 12. Let traversable be navigable's traversable navigable.
auto traversable = navigable->traversable_navigable();
// 13. Append the following session history synchronous navigation steps involving navigable to traversable:
traversable->append_session_history_synchronous_navigation_steps(*navigable, GC::create_function(document.realm().heap(), [traversable, navigable, new_entry, entry_to_replace, history_handling] {
// 1. Finalize a same-document navigation given traversable, navigable, newEntry, entryToReplace, historyHandling, and "none".
finalize_a_same_document_navigation(*traversable, *navigable, new_entry, entry_to_replace, history_handling, UserNavigationInvolvement::None);
// 2. FIXME: Invoke WebDriver BiDi history updated with navigable.
}));
}
void Navigable::scroll_offset_did_change()
{
// https://w3c.github.io/csswg-drafts/cssom-view-1/#scrolling-events
// Whenever a viewport gets scrolled (whether in response to user interaction or by an API), the user agent must run these steps:
// 1. Let doc be the viewports associated Document.
auto doc = active_document();
VERIFY(doc);
// 2. If doc is already in docs pending scroll event targets, abort these steps.
for (auto& target : doc->pending_scroll_event_targets()) {
if (target.ptr() == doc)
return;
}
// 3. Append doc to docs pending scroll event targets.
doc->pending_scroll_event_targets().append(*doc);
}
CSSPixelRect Navigable::to_top_level_rect(CSSPixelRect const& a_rect)
{
auto rect = a_rect;
rect.set_location(to_top_level_position(a_rect.location()));
return rect;
}
CSSPixelPoint Navigable::to_top_level_position(CSSPixelPoint a_position)
{
auto position = a_position;
for (auto ancestor = this; ancestor; ancestor = ancestor->parent()) {
if (is<TraversableNavigable>(*ancestor))
break;
if (!ancestor->container())
return {};
if (!ancestor->container()->paintable())
return {};
// FIXME: Handle CSS transforms that might affect the ancestor.
position.translate_by(ancestor->container()->paintable()->box_type_agnostic_position());
}
return position;
}
void Navigable::set_viewport_size(CSSPixelSize size)
{
if (m_viewport_size == size)
return;
m_viewport_size = size;
if (!m_is_svg_page) {
m_backing_store_manager->restart_resize_timer();
m_backing_store_manager->resize_backing_stores_if_needed(Web::Painting::BackingStoreManager::WindowResizingInProgress::Yes);
m_pending_set_browser_zoom_request = false;
}
if (auto document = active_document()) {
// NOTE: Resizing the viewport changes the reference value for viewport-relative CSS lengths.
document->invalidate_style(DOM::StyleInvalidationReason::NavigableSetViewportSize);
document->set_needs_media_query_evaluation();
if (auto layout_node = document->layout_node())
layout_node->set_needs_layout_update(DOM::SetNeedsLayoutReason::NavigableSetViewportSize);
}
if (auto document = active_document()) {
document->set_needs_display(InvalidateDisplayList::No);
document->inform_all_viewport_clients_about_the_current_viewport_rect();
// Schedule the HTML event loop to ensure that a `resize` event gets fired.
HTML::main_thread_event_loop().schedule();
}
}
void Navigable::perform_scroll_of_viewport(CSSPixelPoint new_position)
{
if (m_viewport_scroll_offset != new_position) {
m_viewport_scroll_offset = new_position;
scroll_offset_did_change();
if (auto document = active_document()) {
document->set_needs_display(InvalidateDisplayList::No);
document->set_needs_to_refresh_scroll_state(true);
document->inform_all_viewport_clients_about_the_current_viewport_rect();
}
}
// Schedule the HTML event loop to ensure that a `resize` event gets fired.
HTML::main_thread_event_loop().schedule();
}
// https://html.spec.whatwg.org/multipage/webappapis.html#rendering-opportunity
bool Navigable::has_a_rendering_opportunity() const
{
// A navigable has a rendering opportunity if the user agent is currently able to present
// the contents of the navigable to the user,
// accounting for hardware refresh rate constraints and user agent throttling for performance reasons,
// but considering content presentable even if it's outside the viewport.
// A navigable's rendering opportunities are determined based on hardware constraints
// such as display refresh rates and other factors such as page performance
// or whether its active document's visibility state is "visible".
// Rendering opportunities typically occur at regular intervals.
// FIXME: Return `false` here if we're an inactive browser tab.
return is_ready_to_paint();
}
// https://html.spec.whatwg.org/multipage/nav-history-apis.html#inform-the-navigation-api-about-child-navigable-destruction
void Navigable::inform_the_navigation_api_about_child_navigable_destruction()
{
// 1. Inform the navigation API about aborting navigation in navigable.
inform_the_navigation_api_about_aborting_navigation();
// FIXME: 2. Let navigation be navigable's active window's navigation API.
// FIXME: 3. Let traversalAPIMethodTrackers be a clone of navigation's upcoming traverse API method trackers.
// FIXME: 4. For each apiMethodTracker of traversalAPIMethodTrackers: reject the finished promise for apiMethodTracker with a new "AbortError" DOMException created in navigation's relevant realm.
}
// https://html.spec.whatwg.org/multipage/nav-history-apis.html#inform-the-navigation-api-about-aborting-navigation
void Navigable::inform_the_navigation_api_about_aborting_navigation()
{
// FIXME: 1. If this algorithm is running on navigable's active window's relevant agent's event loop, then continue on to the following steps.
// Otherwise, queue a global task on the navigation and traversal task source given navigable's active window to run the following steps.
// AD-HOC: Not in the spec but subsequent steps will fail if the navigable doesn't have an active window.
if (!active_window())
return;
queue_global_task(Task::Source::NavigationAndTraversal, *active_window(), GC::create_function(heap(), [this] {
// 2. Let navigation be navigable's active window's navigation API.
VERIFY(active_window());
auto navigation = active_window()->navigation();
// 3. If navigation's ongoing navigate event is null, then return.
if (navigation->ongoing_navigate_event() == nullptr)
return;
// 4. Abort the ongoing navigation given navigation.
navigation->abort_the_ongoing_navigation();
}));
}
bool Navigable::is_focused() const
{
return &m_page->focused_navigable() == this;
}
static String visible_text_in_range(DOM::Range const& range)
{
// NOTE: This is an adaption of Range stringification, but we skip over DOM nodes that don't have a corresponding layout node.
StringBuilder builder;
if (range.start_container() == range.end_container() && is<DOM::Text>(*range.start_container())) {
if (!range.start_container()->layout_node())
return String {};
return static_cast<DOM::Text const&>(*range.start_container()).data().substring_view(range.start_offset(), range.end_offset() - range.start_offset()).to_utf8_but_should_be_ported_to_utf16();
}
if (is<DOM::Text>(*range.start_container()) && range.start_container()->layout_node())
builder.append(static_cast<DOM::Text const&>(*range.start_container()).data().substring_view(range.start_offset()));
range.for_each_contained([&](GC::Ref<DOM::Node> node) {
if (is<DOM::Text>(*node) && node->layout_node())
builder.append(static_cast<DOM::Text const&>(*node).data());
return IterationDecision::Continue;
});
if (is<DOM::Text>(*range.end_container()) && range.end_container()->layout_node())
builder.append(static_cast<DOM::Text const&>(*range.end_container()).data().substring_view(0, range.end_offset()));
return MUST(builder.to_string());
}
String Navigable::selected_text() const
{
auto document = active_document();
if (!document)
return String {};
auto selection = document->get_selection();
auto range = selection->range();
if (!range)
return String {};
return visible_text_in_range(*range);
}
void Navigable::select_all()
{
auto document = active_document();
if (!document)
return;
auto selection = document->get_selection();
if (!selection)
return;
if (auto target = document->active_input_events_target()) {
target->select_all();
} else if (auto* body = document->body()) {
(void)selection->select_all_children(*body);
}
}
void Navigable::paste(Utf16String const& text)
{
auto document = active_document();
if (!document)
return;
m_event_handler.handle_paste(text);
}
// https://drafts.csswg.org/css-view-transitions-1/#snapshot-containing-block
CSSPixelRect Navigable::snapshot_containing_block()
{
// The snapshot containing block is a rectangle that covers all areas of the window that could potentially display
// page content (and is therefore consistent regardless of root scrollbars or interactive widgets).
// Within a child navigable, the snapshot containing block is the union of the navigables viewport with any scrollbar gutters.
// FIXME: Actually get the correct rectangle here.
return viewport_rect();
}
// https://drafts.csswg.org/css-view-transitions-1/#snapshot-containing-block-size
CSSPixelSize Navigable::snapshot_containing_block_size()
{
return this->snapshot_containing_block().size();
}
void Navigable::register_navigation_observer(Badge<NavigationObserver>, NavigationObserver& navigation_observer)
{
auto result = m_navigation_observers.set(navigation_observer);
VERIFY(result == AK::HashSetResult::InsertedNewEntry);
}
void Navigable::unregister_navigation_observer(Badge<NavigationObserver>, NavigationObserver& navigation_observer)
{
bool was_removed = m_navigation_observers.remove(navigation_observer);
VERIFY(was_removed);
}
// https://html.spec.whatwg.org/multipage/document-lifecycle.html#nav-stop
void Navigable::stop_loading()
{
// 1. Let document be navigable's active document.
auto document = active_document();
// 2. If document's unload counter is 0, and navigable's ongoing navigation is a navigation ID, then set the ongoing navigation for navigable to null.
if (document->unload_counter() == 0 && ongoing_navigation().has<String>())
set_ongoing_navigation(Empty {});
// 3. Abort a document and its descendants given document.
document->abort_a_document_and_its_descendants();
}
void Navigable::set_has_session_history_entry_and_ready_for_navigation()
{
m_has_session_history_entry_and_ready_for_navigation = true;
while (!m_pending_navigations.is_empty()) {
auto navigation_params = m_pending_navigations.take_first();
begin_navigation(navigation_params);
}
}
bool Navigable::is_ready_to_paint() const
{
return m_number_of_queued_rasterization_tasks <= 1;
}
void Navigable::ready_to_paint()
{
m_number_of_queued_rasterization_tasks--;
VERIFY(m_number_of_queued_rasterization_tasks >= 0 && m_number_of_queued_rasterization_tasks < 2);
}
void Navigable::paint_next_frame()
{
if (!is_top_level_traversable())
return;
auto [backing_store_id, painting_surface] = m_backing_store_manager->acquire_store_for_next_frame();
if (!painting_surface)
return;
VERIFY(m_number_of_queued_rasterization_tasks <= 1);
m_number_of_queued_rasterization_tasks++;
auto viewport_rect = page().css_to_device_rect(this->viewport_rect()).to_type<int>();
PaintConfig paint_config { .paint_overlay = true, .should_show_line_box_borders = m_should_show_line_box_borders, .canvas_fill_rect = Gfx::IntRect { {}, viewport_rect.size() } };
auto page_client = &page().top_level_traversable()->page().client();
start_display_list_rendering(*painting_surface, paint_config, [page_client, viewport_rect, backing_store_id] {
if (!page_client)
return;
page_client->page_did_paint(viewport_rect, backing_store_id);
});
}
void Navigable::start_display_list_rendering(Gfx::PaintingSurface& painting_surface, PaintConfig paint_config, Function<void()>&& callback)
{
m_needs_repaint = false;
auto document = active_document();
if (!document) {
callback();
return;
}
auto display_list = document->record_display_list(paint_config);
if (!display_list) {
callback();
return;
}
auto& document_paintable = *document->paintable();
Painting::ScrollStateSnapshotByDisplayList scroll_state_snapshot_by_display_list;
document_paintable.refresh_scroll_state();
auto scroll_state_snapshot = document_paintable.scroll_state().snapshot();
scroll_state_snapshot_by_display_list.set(*display_list, move(scroll_state_snapshot));
// Collect scroll state snapshots for each nested navigable
document_paintable.for_each_in_inclusive_subtree_of_type<Painting::NavigableContainerViewportPaintable>([&scroll_state_snapshot_by_display_list](auto& navigable_container_paintable) {
auto const* hosted_document = navigable_container_paintable.navigable_container().content_document_without_origin_check();
if (!hosted_document || !hosted_document->paintable())
return TraversalDecision::Continue;
// We are only interested in collecting scroll state snapshots for visible nested navigables, which is
// detectable by checking if they have a cached display list that should've been populated by
// record_display_list() on top-level document.
auto navigable_display_list = hosted_document->cached_display_list();
if (!navigable_display_list)
return TraversalDecision::Continue;
const_cast<DOM::Document&>(*hosted_document).paintable()->refresh_scroll_state();
auto navigable_scroll_state_snapshot = hosted_document->paintable()->scroll_state().snapshot();
scroll_state_snapshot_by_display_list.set(*navigable_display_list, move(navigable_scroll_state_snapshot));
return TraversalDecision::Continue;
});
m_rendering_thread.enqueue_rendering_task(*display_list, move(scroll_state_snapshot_by_display_list), painting_surface, move(callback));
}
RefPtr<Gfx::SkiaBackendContext> Navigable::skia_backend_context() const
{
return m_skia_backend_context;
}
// https://drafts.csswg.org/cssom-view/#viewport-perform-a-scroll
void Navigable::scroll_viewport_by_delta(CSSPixelPoint delta)
{
// 1. Let doc be the viewports associated Document.
auto doc = active_document();
// 2. Let vv be the VisualViewport whose associated document is doc.
auto vv = doc->visual_viewport();
CSSPixelRect viewport_rect = { m_viewport_scroll_offset, m_viewport_size };
// 3. Let maxX be the difference between viewports scrolling boxs width and the value of vvs width attribute.
auto max_x = viewport_rect.width().to_double() - vv->width();
// 4. Let maxY be the difference between viewports scrolling boxs height and the value of vvs height attribute.
auto max_y = viewport_rect.height().to_double() - vv->height();
// 5. Let dx be the horizontal component of position - the value vvs pageLeft attribute
// NOTE: Function accepts delta so we use that directly.
auto dx = delta.x().to_double();
// 6. Let dy be the vertical component of position - the value of vvs pageTop attribute
// NOTE: Function accepts delta so we use that directly.
auto dy = delta.y().to_double();
// 7. Let visual x be the value of vvs offsetLeft attribute.
auto visual_x = vv->offset_left();
// 8. Let visual y be the value of vvs offsetTop attribute.
auto visual_y = vv->offset_top();
// 9. Let visual dx be min(maxX, max(0, visual x + dx)) - visual x.
auto visual_dx = min(max_x, max(0.0, visual_x + dx)) - visual_x;
// 10. Let visual dy be min(maxY, max(0, visual y + dy)) - visual y.
auto visual_dy = min(max_y, max(0.0, visual_y + dy)) - visual_y;
// 11. Let layout dx be dx - visual dx
auto layout_dx = dx - visual_dx;
// 12. Let layout dy be dy - visual dy
auto layout_dy = dy - visual_dy;
// 13. Let element be docs root element if there is one, null otherwise.
// 14. Perform a scroll of the viewports scrolling box to its current scroll position + (layout dx, layout dy) with element as the associated element, and behavior as the scroll behavior.
auto scrolling_area = doc->paintable_box()->scrollable_overflow_rect()->to_type<float>();
auto new_viewport_scroll_offset = m_viewport_scroll_offset.to_type<double>() + Gfx::Point(layout_dx, layout_dy);
// NOTE: Clamp to the scrolling area.
new_viewport_scroll_offset.set_x(max(0.0, min(new_viewport_scroll_offset.x(), scrolling_area.width() - viewport_size().width().to_double())));
new_viewport_scroll_offset.set_y(max(0.0, min(new_viewport_scroll_offset.y(), scrolling_area.height() - viewport_size().height().to_double())));
perform_scroll_of_viewport(new_viewport_scroll_offset.to_type<CSSPixels>());
// 15. Perform a scroll of vvs scrolling box to its current scroll position + (visual dx, visual dy) with element as the associated element, and behavior as the scroll behavior.
vv->scroll_by({ visual_dx, visual_dy });
doc->set_needs_display(InvalidateDisplayList::No);
}
void Navigable::reset_zoom()
{
auto document = active_document();
if (!document)
return;
document->visual_viewport()->reset();
}
}