| // Copyright 2013 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 <sstream> |
| |
| #include "base/macros.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "chrome/browser/chrome_notification_types.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/browser/search/search.h" |
| #include "chrome/browser/ui/browser.h" |
| #include "chrome/browser/ui/browser_window.h" |
| #include "chrome/browser/ui/location_bar/location_bar.h" |
| #include "chrome/browser/ui/search/instant_test_base.h" |
| #include "chrome/browser/ui/search/instant_test_utils.h" |
| #include "chrome/browser/ui/search/search_tab_helper.h" |
| #include "chrome/browser/ui/tabs/tab_strip_model.h" |
| #include "chrome/common/url_constants.h" |
| #include "chrome/test/base/in_process_browser_test.h" |
| #include "chrome/test/base/ui_test_utils.h" |
| #include "components/omnibox/browser/omnibox_edit_model.h" |
| #include "components/omnibox/browser/omnibox_view.h" |
| #include "content/public/browser/navigation_controller.h" |
| #include "content/public/browser/notification_service.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/test/browser_test_utils.h" |
| #include "content/public/test/test_utils.h" |
| #include "net/test/embedded_test_server/embedded_test_server.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| class InstantExtendedTest : public InProcessBrowserTest, |
| public InstantTestBase { |
| public: |
| InstantExtendedTest() |
| : on_most_visited_change_calls_(0), |
| most_visited_items_count_(0), |
| first_most_visited_item_id_(0), |
| on_focus_changed_calls_(0), |
| is_focused_(false) {} |
| |
| protected: |
| void SetUpInProcessBrowserTestFixture() override { |
| ASSERT_TRUE(https_test_server().Start()); |
| GURL base_url = https_test_server().GetURL("/instant_extended.html?"); |
| GURL ntp_url = https_test_server().GetURL("/instant_extended_ntp.html?"); |
| InstantTestBase::Init(base_url, ntp_url, false); |
| } |
| |
| bool UpdateSearchState(content::WebContents* contents) WARN_UNUSED_RESULT { |
| return instant_test_utils::GetIntFromJS(contents, |
| "onMostVisitedChangedCalls", |
| &on_most_visited_change_calls_) && |
| instant_test_utils::GetIntFromJS(contents, "mostVisitedItemsCount", |
| &most_visited_items_count_) && |
| instant_test_utils::GetIntFromJS(contents, "firstMostVisitedItemId", |
| &first_most_visited_item_id_) && |
| instant_test_utils::GetIntFromJS(contents, "onFocusChangedCalls", |
| &on_focus_changed_calls_) && |
| instant_test_utils::GetBoolFromJS(contents, "isFocused", |
| &is_focused_); |
| } |
| |
| OmniboxView* omnibox() { |
| return instant_browser()->window()->GetLocationBar()->GetOmniboxView(); |
| } |
| |
| void FocusOmnibox() { |
| // If the omnibox already has focus, just notify SearchTabHelper. |
| if (omnibox()->model()->has_focus()) { |
| content::WebContents* active_tab = |
| instant_browser()->tab_strip_model()->GetActiveWebContents(); |
| SearchTabHelper::FromWebContents(active_tab) |
| ->OmniboxFocusChanged(OMNIBOX_FOCUS_VISIBLE, |
| OMNIBOX_FOCUS_CHANGE_EXPLICIT); |
| } else { |
| instant_browser()->window()->GetLocationBar()->FocusLocation(false); |
| } |
| } |
| |
| void SetOmniboxText(const std::string& text) { |
| FocusOmnibox(); |
| omnibox()->SetUserText(base::UTF8ToUTF16(text)); |
| } |
| |
| void PressEnterAndWaitForNavigation() { |
| content::WindowedNotificationObserver nav_observer( |
| content::NOTIFICATION_NAV_ENTRY_COMMITTED, |
| content::NotificationService::AllSources()); |
| instant_browser()->window()->GetLocationBar()->AcceptInput(); |
| nav_observer.Wait(); |
| } |
| |
| void PressEnterAndWaitForFrameLoad() { |
| content::WindowedNotificationObserver nav_observer( |
| content::NOTIFICATION_LOAD_COMPLETED_MAIN_FRAME, |
| content::NotificationService::AllSources()); |
| instant_browser()->window()->GetLocationBar()->AcceptInput(); |
| nav_observer.Wait(); |
| } |
| |
| std::string GetOmniboxText() { |
| return base::UTF16ToUTF8(omnibox()->GetText()); |
| } |
| |
| int on_most_visited_change_calls_; |
| int most_visited_items_count_; |
| int first_most_visited_item_id_; |
| int on_focus_changed_calls_; |
| bool is_focused_; |
| }; |
| |
| // Test to verify that switching tabs should not dispatch onmostvisitedchanged |
| // events. |
| IN_PROC_BROWSER_TEST_F(InstantExtendedTest, NoMostVisitedChangedOnTabSwitch) { |
| // Initialize Instant. |
| ASSERT_NO_FATAL_FAILURE(SetupInstant(browser())); |
| |
| // Open new tab. |
| ui_test_utils::NavigateToURLWithDisposition( |
| browser(), GURL(chrome::kChromeUINewTabURL), |
| WindowOpenDisposition::NEW_FOREGROUND_TAB, |
| ui_test_utils::BROWSER_TEST_WAIT_FOR_TAB | |
| ui_test_utils::BROWSER_TEST_WAIT_FOR_NAVIGATION); |
| EXPECT_EQ(2, browser()->tab_strip_model()->count()); |
| |
| // Make sure new tab received the onmostvisitedchanged event once. |
| content::WebContents* active_tab = |
| browser()->tab_strip_model()->GetActiveWebContents(); |
| EXPECT_TRUE(UpdateSearchState(active_tab)); |
| EXPECT_EQ(1, on_most_visited_change_calls_); |
| |
| // Activate the previous tab. |
| browser()->tab_strip_model()->ActivateTabAt(0); |
| |
| // Switch back to new tab. |
| browser()->tab_strip_model()->ActivateTabAt(1); |
| |
| // Confirm that new tab got no onmostvisitedchanged event. |
| active_tab = browser()->tab_strip_model()->GetActiveWebContents(); |
| EXPECT_TRUE(UpdateSearchState(active_tab)); |
| EXPECT_EQ(1, on_most_visited_change_calls_); |
| } |
| |
| IN_PROC_BROWSER_TEST_F(InstantExtendedTest, NavigateBackToNTP) { |
| ASSERT_NO_FATAL_FAILURE(SetupInstant(browser())); |
| FocusOmnibox(); |
| |
| // Open a new tab page. |
| ui_test_utils::NavigateToURLWithDisposition( |
| browser(), GURL(chrome::kChromeUINewTabURL), |
| WindowOpenDisposition::NEW_FOREGROUND_TAB, |
| ui_test_utils::BROWSER_TEST_WAIT_FOR_TAB | |
| ui_test_utils::BROWSER_TEST_WAIT_FOR_NAVIGATION); |
| EXPECT_EQ(2, browser()->tab_strip_model()->count()); |
| |
| SetOmniboxText("flowers"); |
| PressEnterAndWaitForNavigation(); |
| |
| // Navigate back to NTP. |
| content::WebContents* active_tab = |
| browser()->tab_strip_model()->GetActiveWebContents(); |
| EXPECT_TRUE(active_tab->GetController().CanGoBack()); |
| content::WindowedNotificationObserver load_stop_observer( |
| content::NOTIFICATION_LOAD_STOP, |
| content::Source<content::NavigationController>( |
| &active_tab->GetController())); |
| active_tab->GetController().GoBack(); |
| load_stop_observer.Wait(); |
| |
| active_tab = browser()->tab_strip_model()->GetActiveWebContents(); |
| EXPECT_TRUE(search::IsInstantNTP(active_tab)); |
| } |
| |
| IN_PROC_BROWSER_TEST_F(InstantExtendedTest, |
| DispatchMVChangeEventWhileNavigatingBackToNTP) { |
| // Setup Instant. |
| ASSERT_NO_FATAL_FAILURE(SetupInstant(browser())); |
| FocusOmnibox(); |
| |
| // Open new tab. |
| ui_test_utils::NavigateToURLWithDisposition( |
| browser(), GURL(chrome::kChromeUINewTabURL), |
| WindowOpenDisposition::NEW_FOREGROUND_TAB, |
| ui_test_utils::BROWSER_TEST_WAIT_FOR_TAB | |
| ui_test_utils::BROWSER_TEST_WAIT_FOR_NAVIGATION); |
| |
| content::WebContents* active_tab = |
| browser()->tab_strip_model()->GetActiveWebContents(); |
| EXPECT_TRUE(UpdateSearchState(active_tab)); |
| EXPECT_EQ(1, on_most_visited_change_calls_); |
| |
| content::WindowedNotificationObserver observer( |
| content::NOTIFICATION_LOAD_STOP, |
| content::NotificationService::AllSources()); |
| // Set the text and press enter to navigate from NTP. |
| SetOmniboxText("Pen"); |
| PressEnterAndWaitForNavigation(); |
| observer.Wait(); |
| |
| // Navigate back to NTP. |
| content::WindowedNotificationObserver back_observer( |
| content::NOTIFICATION_LOAD_STOP, |
| content::NotificationService::AllSources()); |
| active_tab = browser()->tab_strip_model()->GetActiveWebContents(); |
| EXPECT_TRUE(active_tab->GetController().CanGoBack()); |
| active_tab->GetController().GoBack(); |
| back_observer.Wait(); |
| |
| // Verify that onmostvisitedchange event is dispatched when we navigate from |
| // SRP to NTP. |
| active_tab = browser()->tab_strip_model()->GetActiveWebContents(); |
| EXPECT_TRUE(UpdateSearchState(active_tab)); |
| EXPECT_EQ(1, on_most_visited_change_calls_); |
| } |
| |
| // Check that clicking on a result sends the correct referrer. |
| IN_PROC_BROWSER_TEST_F(InstantExtendedTest, Referrer) { |
| ASSERT_TRUE(embedded_test_server()->Start()); |
| GURL result_url = embedded_test_server()->GetURL( |
| "/referrer_policy/referrer-policy-log.html"); |
| ASSERT_NO_FATAL_FAILURE(SetupInstant(browser())); |
| FocusOmnibox(); |
| |
| // Type a query and press enter to get results. |
| SetOmniboxText("query"); |
| PressEnterAndWaitForFrameLoad(); |
| |
| // Simulate going to a result. |
| content::WebContents* contents = |
| browser()->tab_strip_model()->GetActiveWebContents(); |
| std::ostringstream stream; |
| stream << "var link = document.createElement('a');"; |
| stream << "link.href = \"" << result_url.spec() << "\";"; |
| stream << "document.body.appendChild(link);"; |
| stream << "link.click();"; |
| EXPECT_TRUE(content::ExecuteScript(contents, stream.str())); |
| |
| content::WaitForLoadStop(contents); |
| std::string expected_title = |
| "Referrer is " + base_url().GetWithEmptyPath().spec(); |
| EXPECT_EQ(base::ASCIIToUTF16(expected_title), contents->GetTitle()); |
| } |