#include "content/browser/preloading/prerender/prerender_commit_deferring_condition.h"
#include "base/memory/ptr_util.h"
#include "base/task/sequenced_task_runner.h"
#include "content/browser/preloading/prerender/prerender_host.h"
#include "content/browser/preloading/prerender/prerender_metrics.h"
#include "content/browser/renderer_host/frame_tree.h"
#include "content/browser/renderer_host/frame_tree_node.h"
#include "content/browser/renderer_host/navigation_request.h"
#include "content/public/browser/render_frame_host.h"
namespace content {
namespace {
FrameTreeNode* GetRootPrerenderFrameTreeNode(
FrameTreeNodeId prerender_frame_tree_node_id) {
FrameTreeNode* root =
FrameTreeNode::GloballyFindByID(prerender_frame_tree_node_id);
if (root) {
CHECK(root->IsOutermostMainFrame());
}
return root;
}
}
std::unique_ptr<CommitDeferringCondition>
PrerenderCommitDeferringCondition::MaybeCreate(
NavigationRequest& navigation_request,
NavigationType navigation_type,
std::optional<FrameTreeNodeId> candidate_prerender_frame_tree_node_id) {
if (navigation_type != NavigationType::kPrerenderedPageActivation)
return nullptr;
return base::WrapUnique(new PrerenderCommitDeferringCondition(
navigation_request, candidate_prerender_frame_tree_node_id.value()));
}
PrerenderCommitDeferringCondition::~PrerenderCommitDeferringCondition() =
default;
PrerenderCommitDeferringCondition::PrerenderCommitDeferringCondition(
NavigationRequest& navigation_request,
FrameTreeNodeId candidate_prerender_frame_tree_node_id)
: CommitDeferringCondition(navigation_request),
WebContentsObserver(navigation_request.GetWebContents()),
candidate_prerender_frame_tree_node_id_(
candidate_prerender_frame_tree_node_id) {
CHECK(candidate_prerender_frame_tree_node_id_);
}
CommitDeferringCondition::Result
PrerenderCommitDeferringCondition::WillCommitNavigation(
base::OnceClosure resume) {
FrameTreeNode* prerender_frame_tree_node =
GetRootPrerenderFrameTreeNode(candidate_prerender_frame_tree_node_id_);
if (!prerender_frame_tree_node) {
return Result::kProceed;
}
PrerenderHost& prerender_host =
PrerenderHost::GetFromFrameTreeNode(*prerender_frame_tree_node);
if (!prerender_frame_tree_node->HasNavigation()) {
RecordPrerenderActivationCommitDeferTime(
base::TimeDelta(), prerender_host.trigger_type(),
prerender_host.embedder_histogram_suffix());
return Result::kProceed;
}
if (!prerender_host.IsUrlMatch(GetNavigationHandle().GetURL())) {
return Result::kProceed;
}
done_closure_ = std::move(resume);
defer_start_time_ = base::TimeTicks::Now();
return Result::kDefer;
}
const char* PrerenderCommitDeferringCondition::TraceEventName() const {
return "PrerenderCommitDeferringCondition";
}
void PrerenderCommitDeferringCondition::DidFinishNavigation(
NavigationHandle* handle) {
auto* finished_navigation_request = NavigationRequest::From(handle);
FrameTreeNode* prerender_frame_tree_node =
GetRootPrerenderFrameTreeNode(candidate_prerender_frame_tree_node_id_);
if (!prerender_frame_tree_node)
return;
if (finished_navigation_request->frame_tree_node() !=
prerender_frame_tree_node) {
return;
}
if (done_closure_ && !prerender_frame_tree_node->HasNavigation()) {
base::SequencedTaskRunner::GetCurrentDefault()->PostTask(
FROM_HERE, std::move(done_closure_));
base::TimeDelta delta = base::TimeTicks::Now() - defer_start_time_;
PrerenderHost& prerender_host =
PrerenderHost::GetFromFrameTreeNode(*prerender_frame_tree_node);
RecordPrerenderActivationCommitDeferTime(
delta, prerender_host.trigger_type(),
prerender_host.embedder_histogram_suffix());
}
}
}