blob: 4d65a1e0dc6f336044305e514ac9a85e4bcc55d4 [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/ui/sync/tab_contents_synced_tab_delegate.h"
#include "base/memory/ref_counted.h"
#include "chrome/browser/complex_tasks/task_tab_helper.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/supervised_user/supervised_user_navigation_observer.h"
#include "components/sessions/content/content_serialized_navigation_builder.h"
#include "components/sync/base/features.h"
#include "components/sync_sessions/sync_sessions_client.h"
#include "components/sync_sessions/synced_window_delegate.h"
#include "components/sync_sessions/synced_window_delegates_getter.h"
#include "content/public/browser/favicon_status.h"
#include "content/public/browser/navigation_controller.h"
#include "content/public/browser/navigation_entry.h"
#include "content/public/browser/web_contents.h"
#include "extensions/buildflags/buildflags.h"
#if BUILDFLAG(ENABLE_EXTENSIONS)
#include "chrome/browser/apps/app_service/web_contents_app_id_utils.h"
#endif
using content::NavigationEntry;
namespace {
// The minimum time between two sync updates of `last_active_time` when the tab
// hasn't changed.
constexpr base::TimeDelta kSyncActiveTimeThreshold = base::Minutes(10);
// Helper to access the correct NavigationEntry, accounting for pending entries.
NavigationEntry* GetPossiblyPendingEntryAtIndex(
content::WebContents* web_contents,
int i) {
int pending_index = web_contents->GetController().GetPendingEntryIndex();
if (pending_index == i) {
return web_contents->GetController().GetPendingEntry();
}
NavigationEntry* entry = web_contents->GetController().GetEntryAtIndex(i);
// Don't use the entry for sync if it doesn't exist or is the initial
// NavigationEntry.
// TODO(crbug.com/40194151): Guarantee this won't be called when on the
// initial NavigationEntry instead of bailing out here.
if (!entry || entry->IsInitialEntry()) {
return nullptr;
}
return entry;
}
} // namespace
void TabContentsSyncedTabDelegate::ResetCachedLastActiveTime() {
cached_last_active_time_.reset();
}
base::Time TabContentsSyncedTabDelegate::GetLastActiveTime() {
const base::Time last_active_time = web_contents_->GetLastActiveTime();
const base::TimeTicks last_active_time_ticks =
web_contents_->GetLastActiveTimeTicks();
if (cached_last_active_time_.has_value() &&
last_active_time_ticks - cached_last_active_time_.value().first <
kSyncActiveTimeThreshold) {
return cached_last_active_time_.value().second;
}
cached_last_active_time_ = {last_active_time_ticks, last_active_time};
return last_active_time;
}
bool TabContentsSyncedTabDelegate::IsBeingDestroyed() const {
return web_contents_->IsBeingDestroyed();
}
std::string TabContentsSyncedTabDelegate::GetExtensionAppId() const {
#if BUILDFLAG(ENABLE_EXTENSIONS)
return apps::GetAppIdForWebContents(web_contents_);
#else
return std::string();
#endif
}
bool TabContentsSyncedTabDelegate::IsInitialBlankNavigation() const {
return web_contents_->GetController().IsInitialBlankNavigation();
}
int TabContentsSyncedTabDelegate::GetCurrentEntryIndex() const {
return web_contents_->GetController().GetCurrentEntryIndex();
}
int TabContentsSyncedTabDelegate::GetEntryCount() const {
return web_contents_->GetController().GetEntryCount();
}
GURL TabContentsSyncedTabDelegate::GetVirtualURLAtIndex(int i) const {
DCHECK(web_contents_);
NavigationEntry* entry = GetPossiblyPendingEntryAtIndex(web_contents_, i);
return entry ? entry->GetVirtualURL() : GURL();
}
void TabContentsSyncedTabDelegate::GetSerializedNavigationAtIndex(
int i,
sessions::SerializedNavigationEntry* serialized_entry) const {
DCHECK(web_contents_);
NavigationEntry* entry = GetPossiblyPendingEntryAtIndex(web_contents_, i);
if (entry) {
// Explicitly exclude page state when serializing the navigation entry.
// Sync ignores the page state anyway (e.g. form data is not synced), and
// the page state can be expensive to serialize.
*serialized_entry =
sessions::ContentSerializedNavigationBuilder::FromNavigationEntry(
i, entry,
sessions::ContentSerializedNavigationBuilder::EXCLUDE_PAGE_STATE);
}
}
bool TabContentsSyncedTabDelegate::ProfileHasChildAccount() const {
return Profile::FromBrowserContext(web_contents_->GetBrowserContext())
->IsChild();
}
const std::vector<std::unique_ptr<const sessions::SerializedNavigationEntry>>*
TabContentsSyncedTabDelegate::GetBlockedNavigations() const {
SupervisedUserNavigationObserver* navigation_observer =
SupervisedUserNavigationObserver::FromWebContents(web_contents_);
#if BUILDFLAG(IS_ANDROID)
// TabHelpers::AttachTabHelpers() will not be called for a placeholder tab's
// WebContents that is temporarily created from a serialized state in
// SyncedTabDelegateAndroid::ReadPlaceholderTabSnapshotIfItShouldSync(). When
// this occurs, early-out and return a nullptr.
if (!navigation_observer) {
return nullptr;
}
#endif // BUILDFLAG(IS_ANDROID)
DCHECK(navigation_observer);
return &navigation_observer->blocked_navigations();
}
bool TabContentsSyncedTabDelegate::ShouldSync(
sync_sessions::SyncSessionsClient* sessions_client) {
if (sessions_client->GetSyncedWindowDelegatesGetter()->FindById(
GetWindowId()) == nullptr) {
return false;
}
if (ProfileHasChildAccount()) {
#if BUILDFLAG(IS_ANDROID)
auto* blocked_navigations = GetBlockedNavigations();
if (blocked_navigations && !blocked_navigations->empty()) {
return true;
}
#else
if (!GetBlockedNavigations()->empty()) {
return true;
}
#endif // BUILDFLAG(IS_ANDROID)
}
if (IsInitialBlankNavigation()) {
return false; // This deliberately ignores a new pending entry.
}
// Don't try to sync the initial NavigationEntry, as it is not actually
// associated with any navigation.
content::NavigationEntry* last_committed_entry =
web_contents_->GetController().GetLastCommittedEntry();
if (last_committed_entry->IsInitialEntry()) {
return false;
}
int entry_count = GetEntryCount();
for (int i = 0; i < entry_count; ++i) {
if (sessions_client->ShouldSyncURL(GetVirtualURLAtIndex(i))) {
return true;
}
}
return false;
}
int64_t TabContentsSyncedTabDelegate::GetTaskIdForNavigationId(
int nav_id) const {
const tasks::TaskTabHelper* task_tab_helper = this->task_tab_helper();
if (task_tab_helper &&
task_tab_helper->get_task_id_for_navigation(nav_id) != nullptr) {
return task_tab_helper->get_task_id_for_navigation(nav_id)->id();
}
return -1;
}
int64_t TabContentsSyncedTabDelegate::GetParentTaskIdForNavigationId(
int nav_id) const {
const tasks::TaskTabHelper* task_tab_helper = this->task_tab_helper();
if (task_tab_helper &&
task_tab_helper->get_task_id_for_navigation(nav_id) != nullptr) {
return task_tab_helper->get_task_id_for_navigation(nav_id)->parent_id();
}
return -1;
}
int64_t TabContentsSyncedTabDelegate::GetRootTaskIdForNavigationId(
int nav_id) const {
const tasks::TaskTabHelper* task_tab_helper = this->task_tab_helper();
if (task_tab_helper &&
task_tab_helper->get_task_id_for_navigation(nav_id) != nullptr) {
return task_tab_helper->get_task_id_for_navigation(nav_id)->root_id();
}
return -1;
}
const content::WebContents* TabContentsSyncedTabDelegate::web_contents() const {
return web_contents_;
}
content::WebContents* TabContentsSyncedTabDelegate::web_contents() {
return web_contents_;
}
void TabContentsSyncedTabDelegate::SetWebContents(
content::WebContents* web_contents) {
web_contents_ = web_contents;
}
const tasks::TaskTabHelper* TabContentsSyncedTabDelegate::task_tab_helper()
const {
if (web_contents_ == nullptr) {
return nullptr;
}
return tasks::TaskTabHelper::FromWebContents(web_contents_);
}