blob: 9fc474936d8d663e2b39f9b299850336bc851415 [file] [log] [blame]
// Copyright 2020 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/ui/focus_tab_after_navigation_helper.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/search/search.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/browser_finder.h"
#include "chrome/browser/ui/browser_window.h"
#include "chrome/browser/ui/tabs/tab_strip_model.h"
#include "chrome/common/webui_url_constants.h"
#include "content/public/browser/browser_url_handler.h"
#include "content/public/browser/navigation_handle.h"
#include "content/public/browser/web_contents.h"
FocusTabAfterNavigationHelper::FocusTabAfterNavigationHelper(
content::WebContents* contents)
: content::WebContentsObserver(contents) {}
FocusTabAfterNavigationHelper::~FocusTabAfterNavigationHelper() = default;
void FocusTabAfterNavigationHelper::ReadyToCommitNavigation(
content::NavigationHandle* navigation) {
// Focus the tab contents if needed. This is done at the ReadyToCommit time
// to:
// 1) ignore same-document navigations (ReadyToCommitNavigation method is not
// invoked for same-document navigations)
// 2) postpone moving the focus until we are ready to commit the page
// 3) move the focus before the page starts rendering
// (only 1 is a hard-requirement; 2 and 3 seem desirable but there are no
// known scenarios where violating these requirements would lead to bugs).
if (ShouldFocusTabContents(navigation))
web_contents()->SetInitialFocus();
}
bool FocusTabAfterNavigationHelper::ShouldFocusTabContents(
content::NavigationHandle* navigation) {
// Don't focus content in an inactive window or tab.
Browser* browser = chrome::FindBrowserWithWebContents(web_contents());
if (!browser)
return false;
if (!browser->window()->IsActive())
return false;
if (browser->tab_strip_model()->GetActiveWebContents() != web_contents())
return false;
// Don't focus content after subframe navigations.
if (!navigation->IsInMainFrame())
return false;
// Browser-initiated navigations (e.g. typing in an omnibox) are taken care of
// in Browser::UpdateUIForNavigationInTab. See also https://crbug.com/1048591
// for possible regression risks related to returning |true| here.
if (!navigation->IsRendererInitiated())
return false;
// Renderer-initiated navigations shouldn't focus the tab contents, unless the
// navigation is leaving the NTP. See also https://crbug.com/1027719.
bool started_at_ntp = IsNtpURL(web_contents()->GetLastCommittedURL());
if (!started_at_ntp)
return false;
// Navigations initiated via chrome.tabs.update and similar APIs should not
// steal focus from the omnibox. See also https://crbug.com/1085779.
if (navigation->GetPageTransition() & ui::PAGE_TRANSITION_FROM_API)
return false;
// Rewrite chrome://newtab to compare with the navigation URL.
GURL rewritten_ntp_url = web_contents()->GetLastCommittedURL();
Profile* profile =
Profile::FromBrowserContext(web_contents()->GetBrowserContext());
content::BrowserURLHandler::GetInstance()->RewriteURLIfNecessary(
&rewritten_ntp_url, profile);
// Focus if the destination is not the NTP.
return !IsNtpURL(navigation->GetURL()) &&
(navigation->GetURL() != rewritten_ntp_url);
}
bool FocusTabAfterNavigationHelper::IsNtpURL(const GURL& url) {
// TODO(lukasza): https://crbug.com/1034999: Try to avoid special-casing
// kChromeUINewTabURL below and covering it via IsNTPOrRelatedURL instead.
if (url == GURL(chrome::kChromeUINewTabURL))
return true;
Profile* profile =
Profile::FromBrowserContext(web_contents()->GetBrowserContext());
return search::IsNTPOrRelatedURL(url, profile);
}
WEB_CONTENTS_USER_DATA_KEY_IMPL(FocusTabAfterNavigationHelper)