LibWeb: Add SessionHistoryTraversalQueue
This commit is contained in:
parent
6d866dc5ba
commit
08788072c1
Notes:
sideshowbarker
2024-07-16 22:22:13 +09:00
Author: https://github.com/kalenikaliaksandr Commit: https://github.com/SerenityOS/serenity/commit/08788072c1 Pull-request: https://github.com/SerenityOS/serenity/pull/20575
5 changed files with 134 additions and 80 deletions
Userland/Libraries/LibWeb/HTML
|
@ -939,61 +939,64 @@ WebIDL::ExceptionOr<void> Navigable::navigate(
|
|||
// for historyEntry, given navigable, "navigate", sourceSnapshotParams,
|
||||
// targetSnapshotParams, navigationId, navigationParams, cspNavigationType, with allowPOST
|
||||
// set to true and completionSteps set to the following step:
|
||||
populate_session_history_entry_document(history_entry, navigation_params, navigation_id, source_snapshot_params, [this, history_entry, history_handling] {
|
||||
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#finalize-a-cross-document-navigation
|
||||
populate_session_history_entry_document(history_entry, navigation_params, navigation_id, source_snapshot_params, [this, history_entry, history_handling, navigation_id] {
|
||||
traversable_navigable()->append_session_history_traversal_steps([this, history_entry, history_handling, navigation_id] {
|
||||
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#finalize-a-cross-document-navigation
|
||||
|
||||
// 1. FIXME: Assert: this is running on navigable's traversable navigable's session history traversal queue.
|
||||
// 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.
|
||||
set_delaying_load_events(false);
|
||||
// 2. Set navigable's is delaying load events to false.
|
||||
set_delaying_load_events(false);
|
||||
|
||||
// 3. If historyEntry's document is null, then return.
|
||||
if (!history_entry->document_state->document())
|
||||
return;
|
||||
// 3. If historyEntry's document is null, then return.
|
||||
if (!history_entry->document_state->document())
|
||||
return;
|
||||
|
||||
// 4. FIXME: 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.
|
||||
// 4. FIXME: 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.
|
||||
|
||||
// 5. 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;
|
||||
// 5. 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;
|
||||
|
||||
// 6. Let traversable be navigable's traversable navigable.
|
||||
auto traversable = traversable_navigable();
|
||||
// 6. Let traversable be navigable's traversable navigable.
|
||||
auto traversable = traversable_navigable();
|
||||
|
||||
// 7. Let targetStep be null.
|
||||
int target_step;
|
||||
// 7. Let targetStep be null.
|
||||
int target_step;
|
||||
|
||||
// 8. Let targetEntries be the result of getting session history entries for navigable.
|
||||
auto& target_entries = get_session_history_entries();
|
||||
// 8. Let targetEntries be the result of getting session history entries for navigable.
|
||||
auto& target_entries = get_session_history_entries();
|
||||
|
||||
// 9. If entryToReplace is null, then:
|
||||
if (entry_to_replace == nullptr) {
|
||||
// FIXME: 1. Clear the forward session history of traversable.
|
||||
// 9. If entryToReplace is null, then:
|
||||
if (entry_to_replace == nullptr) {
|
||||
// FIXME: 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;
|
||||
// 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->step = target_step;
|
||||
// 3. Set historyEntry's step to targetStep.
|
||||
history_entry->step = target_step;
|
||||
|
||||
// 4. Append historyEntry to targetEntries.
|
||||
target_entries.append(move(history_entry));
|
||||
} else {
|
||||
// 1. Replace entryToReplace with historyEntry in targetEntries.
|
||||
*(target_entries.find(*entry_to_replace)) = history_entry;
|
||||
// 4. Append historyEntry to targetEntries.
|
||||
target_entries.append(move(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->step = entry_to_replace->step;
|
||||
// 2. Set historyEntry's step to entryToReplace's step.
|
||||
history_entry->step = entry_to_replace->step;
|
||||
|
||||
// 3. Set targetStep to traversable's current session history step.
|
||||
target_step = traversable->current_session_history_step();
|
||||
}
|
||||
// 3. Set targetStep to traversable's current session history step.
|
||||
target_step = traversable->current_session_history_step();
|
||||
}
|
||||
|
||||
// 10. Apply the history step targetStep to traversable.
|
||||
traversable->apply_the_history_step(target_step);
|
||||
// 10. Apply the history step targetStep to traversable.
|
||||
traversable->apply_the_history_step(target_step);
|
||||
});
|
||||
}).release_value_but_fixme_should_propagate_errors();
|
||||
});
|
||||
|
||||
|
@ -1022,10 +1025,11 @@ void Navigable::reload()
|
|||
// 2. Let traversable be navigable's traversable navigable.
|
||||
auto traversable = traversable_navigable();
|
||||
|
||||
// FIXME: 3. Append the following session history traversal steps to traversable:
|
||||
|
||||
// 1. Apply pending history changes to traversable with true.
|
||||
traversable->apply_pending_history_changes();
|
||||
// 3. Append the following session history traversal steps to traversable:
|
||||
traversable->append_session_history_traversal_steps([traversable] {
|
||||
// 1. Apply pending history changes to traversable with true.
|
||||
traversable->apply_pending_history_changes();
|
||||
});
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -102,30 +102,31 @@ WebIDL::ExceptionOr<void> NavigableContainer::create_new_child_navigable()
|
|||
// 11. Let traversable be parentNavigable's traversable navigable.
|
||||
auto traversable = parent_navigable->traversable_navigable();
|
||||
|
||||
// FIXME: 12. Append the following session history traversal steps to traversable:
|
||||
// 12. Append the following session history traversal steps to traversable:
|
||||
traversable->append_session_history_traversal_steps([traversable, navigable, parent_navigable, history_entry] {
|
||||
// 1. Let parentDocState be parentNavigable's active session history entry's document state.
|
||||
|
||||
// 1. Let parentDocState be parentNavigable's active session history entry's document state.
|
||||
auto parent_doc_state = parent_navigable->active_session_history_entry()->document_state;
|
||||
|
||||
auto parent_doc_state = parent_navigable->active_session_history_entry()->document_state;
|
||||
// 2. Let targetStepSHE be the first session history entry in traversable's session history entries whose document state equals parentDocState.
|
||||
auto target_step_she = *(traversable->session_history_entries().find_if([parent_doc_state](auto& entry) {
|
||||
return entry->document_state == parent_doc_state;
|
||||
}));
|
||||
|
||||
// 2. Let targetStepSHE be the first session history entry in traversable's session history entries whose document state equals parentDocState.
|
||||
auto target_step_she = *(traversable->session_history_entries().find_if([parent_doc_state](auto& entry) {
|
||||
return entry->document_state == parent_doc_state;
|
||||
}));
|
||||
// 3. Set historyEntry's step to targetStepSHE's step.
|
||||
history_entry->step = target_step_she->step;
|
||||
|
||||
// 3. Set historyEntry's step to targetStepSHE's step.
|
||||
history_entry->step = target_step_she->step;
|
||||
// 4. Let nestedHistory be a new nested history whose id is navigable's id and entries list is « historyEntry ».
|
||||
DocumentState::NestedHistory nested_history {
|
||||
.id = navigable->id(),
|
||||
.entries { *history_entry },
|
||||
};
|
||||
|
||||
// 4. Let nestedHistory be a new nested history whose id is navigable's id and entries list is « historyEntry ».
|
||||
DocumentState::NestedHistory nested_history {
|
||||
.id = navigable->id(),
|
||||
.entries { *history_entry },
|
||||
};
|
||||
// 5. Append nestedHistory to parentDocState's nested histories.
|
||||
parent_doc_state->nested_histories().append(move(nested_history));
|
||||
|
||||
// 5. Append nestedHistory to parentDocState's nested histories.
|
||||
parent_doc_state->nested_histories().append(move(nested_history));
|
||||
|
||||
// FIXME: 6. Update for navigable creation/destruction given traversable
|
||||
// FIXME: 6. Update for navigable creation/destruction given traversable
|
||||
});
|
||||
|
||||
return {};
|
||||
}
|
||||
|
@ -342,10 +343,11 @@ void NavigableContainer::destroy_the_child_navigable()
|
|||
// 7. Let traversable be container's node navigable's traversable navigable.
|
||||
auto traversable = this->navigable()->traversable_navigable();
|
||||
|
||||
// FIXME: 8. Append the following session history traversal steps to traversable:
|
||||
|
||||
// 1. Apply pending history changes to traversable.
|
||||
traversable->apply_pending_history_changes();
|
||||
// 8. Append the following session history traversal steps to traversable:
|
||||
traversable->append_session_history_traversal_steps([traversable] {
|
||||
// 1. Apply pending history changes to traversable.
|
||||
traversable->apply_pending_history_changes();
|
||||
});
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -0,0 +1,39 @@
|
|||
/*
|
||||
* Copyright (c) 2023, Aliaksandr Kalenik <kalenik.aliaksandr@gmail.com>
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <LibCore/Timer.h>
|
||||
|
||||
namespace Web::HTML {
|
||||
|
||||
// https://html.spec.whatwg.org/multipage/document-sequences.html#tn-session-history-traversal-queue
|
||||
class SessionHistoryTraversalQueue {
|
||||
public:
|
||||
SessionHistoryTraversalQueue()
|
||||
{
|
||||
m_timer = Core::Timer::create_single_shot(0, [this] {
|
||||
while (m_queue.size() > 0) {
|
||||
auto steps = m_queue.take_first();
|
||||
steps();
|
||||
}
|
||||
}).release_value_but_fixme_should_propagate_errors();
|
||||
}
|
||||
|
||||
void append(JS::SafeFunction<void()> steps)
|
||||
{
|
||||
m_queue.append(move(steps));
|
||||
if (!m_timer->is_active()) {
|
||||
m_timer->start();
|
||||
}
|
||||
}
|
||||
|
||||
private:
|
||||
Vector<JS::SafeFunction<void()>> m_queue;
|
||||
RefPtr<Core::Timer> m_timer;
|
||||
};
|
||||
|
||||
}
|
|
@ -479,24 +479,25 @@ void TraversableNavigable::traverse_the_history_by_delta(int delta)
|
|||
// FIXME: 2. If sourceDocument is given, then:
|
||||
|
||||
// 3. Append the following session history traversal steps to traversable:
|
||||
append_session_history_traversal_steps([this, delta] {
|
||||
// 1. Let allSteps be the result of getting all used history steps for traversable.
|
||||
auto all_steps = get_all_used_history_steps();
|
||||
|
||||
// 1. Let allSteps be the result of getting all used history steps for traversable.
|
||||
auto all_steps = get_all_used_history_steps();
|
||||
// 2. Let currentStepIndex be the index of traversable's current session history step within allSteps.
|
||||
auto current_step_index = *all_steps.find_first_index(current_session_history_step());
|
||||
|
||||
// 2. Let currentStepIndex be the index of traversable's current session history step within allSteps.
|
||||
auto current_step_index = *all_steps.find_first_index(current_session_history_step());
|
||||
// 3. Let targetStepIndex be currentStepIndex plus delta
|
||||
auto target_step_index = current_step_index + delta;
|
||||
|
||||
// 3. Let targetStepIndex be currentStepIndex plus delta
|
||||
auto target_step_index = current_step_index + delta;
|
||||
// 4. If allSteps[targetStepIndex] does not exist, then abort these steps.
|
||||
if (target_step_index >= all_steps.size()) {
|
||||
return;
|
||||
}
|
||||
|
||||
// 4. If allSteps[targetStepIndex] does not exist, then abort these steps.
|
||||
if (target_step_index >= all_steps.size()) {
|
||||
return;
|
||||
}
|
||||
|
||||
// 5. Apply the history step allSteps[targetStepIndex] to traversable, with checkForUserCancelation set to true,
|
||||
// sourceSnapshotParams set to sourceSnapshotParams, and initiatorToCheck set to initiatorToCheck.
|
||||
apply_the_history_step(all_steps[target_step_index]);
|
||||
// 5. Apply the history step allSteps[targetStepIndex] to traversable, with checkForUserCancelation set to true,
|
||||
// sourceSnapshotParams set to sourceSnapshotParams, and initiatorToCheck set to initiatorToCheck.
|
||||
apply_the_history_step(all_steps[target_step_index]);
|
||||
});
|
||||
}
|
||||
|
||||
// https://html.spec.whatwg.org/multipage/browsing-the-web.html#apply-pending-history-changes
|
||||
|
|
|
@ -8,6 +8,7 @@
|
|||
|
||||
#include <AK/Vector.h>
|
||||
#include <LibWeb/HTML/Navigable.h>
|
||||
#include <LibWeb/HTML/SessionHistoryTraversalQueue.h>
|
||||
#include <LibWeb/HTML/VisibilityState.h>
|
||||
|
||||
namespace Web::HTML {
|
||||
|
@ -47,6 +48,11 @@ public:
|
|||
|
||||
void destroy_top_level_traversable();
|
||||
|
||||
void append_session_history_traversal_steps(JS::SafeFunction<void()> steps)
|
||||
{
|
||||
m_session_history_traversal_queue.append(move(steps));
|
||||
}
|
||||
|
||||
private:
|
||||
TraversableNavigable();
|
||||
|
||||
|
@ -65,6 +71,8 @@ private:
|
|||
|
||||
// https://html.spec.whatwg.org/multipage/document-sequences.html#system-visibility-state
|
||||
VisibilityState m_system_visibility_state { VisibilityState::Visible };
|
||||
|
||||
SessionHistoryTraversalQueue m_session_history_traversal_queue;
|
||||
};
|
||||
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue