WebBackForwardListItem.cpp [plain text]
#include "config.h"
#include "WebBackForwardListItem.h"
#include "SuspendedPageProxy.h"
#include "WebProcessPool.h"
#include "WebProcessProxy.h"
#include <wtf/DebugUtilities.h>
#include <wtf/URL.h>
namespace WebKit {
using namespace WebCore;
Ref<WebBackForwardListItem> WebBackForwardListItem::create(BackForwardListItemState&& backForwardListItemState, uint64_t pageID)
{
return adoptRef(*new WebBackForwardListItem(WTFMove(backForwardListItemState), pageID));
}
WebBackForwardListItem::WebBackForwardListItem(BackForwardListItemState&& backForwardListItemState, uint64_t pageID)
: m_itemState(WTFMove(backForwardListItemState))
, m_pageID(pageID)
, m_lastProcessIdentifier(m_itemState.identifier.processIdentifier)
{
auto result = allItems().add(m_itemState.identifier, this);
ASSERT_UNUSED(result, result.isNewEntry);
}
WebBackForwardListItem::~WebBackForwardListItem()
{
ASSERT(allItems().get(m_itemState.identifier) == this);
allItems().remove(m_itemState.identifier);
removeSuspendedPageFromProcessPool();
}
HashMap<BackForwardItemIdentifier, WebBackForwardListItem*>& WebBackForwardListItem::allItems()
{
static NeverDestroyed<HashMap<BackForwardItemIdentifier, WebBackForwardListItem*>> items;
return items;
}
WebBackForwardListItem* WebBackForwardListItem::itemForID(const BackForwardItemIdentifier& identifier)
{
return allItems().get(identifier);
}
static const FrameState* childItemWithDocumentSequenceNumber(const FrameState& frameState, int64_t number)
{
for (const auto& child : frameState.children) {
if (child.documentSequenceNumber == number)
return &child;
}
return nullptr;
}
static bool documentTreesAreEqual(const FrameState& a, const FrameState& b)
{
if (a.documentSequenceNumber != b.documentSequenceNumber)
return false;
if (a.children.size() != b.children.size())
return false;
for (const auto& child : a.children) {
const FrameState* otherChild = childItemWithDocumentSequenceNumber(b, child.documentSequenceNumber);
if (!otherChild || !documentTreesAreEqual(child, *otherChild))
return false;
}
return true;
}
bool WebBackForwardListItem::itemIsInSameDocument(const WebBackForwardListItem& other) const
{
if (m_pageID != other.m_pageID)
return false;
const FrameState& mainFrameState = m_itemState.pageState.mainFrameState;
const FrameState& otherMainFrameState = other.m_itemState.pageState.mainFrameState;
if (mainFrameState.stateObjectData || otherMainFrameState.stateObjectData)
return mainFrameState.documentSequenceNumber == otherMainFrameState.documentSequenceNumber;
URL url = URL({ }, mainFrameState.urlString);
URL otherURL = URL({ }, otherMainFrameState.urlString);
if ((url.hasFragmentIdentifier() || otherURL.hasFragmentIdentifier()) && equalIgnoringFragmentIdentifier(url, otherURL))
return mainFrameState.documentSequenceNumber == otherMainFrameState.documentSequenceNumber;
return documentTreesAreEqual(mainFrameState, otherMainFrameState);
}
void WebBackForwardListItem::setSuspendedPage(SuspendedPageProxy* page)
{
if (m_suspendedPage == page)
return;
removeSuspendedPageFromProcessPool();
m_suspendedPage = makeWeakPtr(page);
}
void WebBackForwardListItem::removeSuspendedPageFromProcessPool()
{
if (!m_suspendedPage)
return;
m_suspendedPage->process().processPool().removeSuspendedPage(*m_suspendedPage);
ASSERT(!m_suspendedPage);
}
#if !LOG_DISABLED
const char* WebBackForwardListItem::loggingString()
{
return debugString("Back/forward item ID ", itemID().logString(), ", original URL ", originalURL(), ", current URL ", url(), m_suspendedPage ? "(has a suspended page)" : "");
}
#endif // !LOG_DISABLED
}