// Copyright (c) 2012 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/chromeos/tab_closeable_state_watcher.h" #include "base/file_path.h" #include "chrome/browser/first_run/first_run.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/tabs/tab_strip_model.h" #include "chrome/browser/ui/app_modal_dialogs/app_modal_dialog.h" #include "chrome/browser/ui/app_modal_dialogs/native_app_modal_dialog.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/browser_list.h" #include "chrome/browser/ui/browser_window.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 "content/public/browser/notification_service.h" #include "content/public/browser/notification_types.h" #include "content/public/browser/web_contents.h" #include "googleurl/src/gurl.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" using content::WebContents; namespace chromeos { class TabCloseableStateWatcherTest : public InProcessBrowserTest { public: TabCloseableStateWatcherTest() { // This test is testing TabCloseStateWatcher, so enable it. EnableTabCloseableStateWatcher(); blank_url_ = GURL(chrome::kAboutBlankURL); ntp_url_ = GURL(chrome::kChromeUINewTabURL); other_url_ = ui_test_utils::GetTestUrl( FilePath(FilePath::kCurrentDirectory), FilePath(FILE_PATH_LITERAL("title2.html"))); } virtual bool SetUpUserDataDirectory() OVERRIDE { // Make sure the first run sentinel file exists before running these tests, // since some of them depend on not running during the first run. // TODO(bauerb): set the first run flag instead of creating a sentinel file. first_run::CreateSentinel(); return InProcessBrowserTest::SetUpUserDataDirectory(); } protected: // Wrapper for Browser::AddTabWithURL void AddTabWithURL(Browser* browser, const GURL& url) { ui_test_utils::WindowedNotificationObserver observer( content::NOTIFICATION_LOAD_STOP, content::NotificationService::AllSources()); AddTabAtIndexToBrowser(browser, 0, url, content::PAGE_TRANSITION_TYPED); observer.Wait(); } // Wrapper for TabCloseableStateWatcher::CanCloseTab bool CanCloseTab(const Browser* browser) { return browser->tabstrip_model()->delegate()->CanCloseTab(); } // Create popup browser. Browser* CreatePopupBrowser() { // This is mostly duplicated from InPocessBrowserTest::CreateBrowser, // except that a popup browser is created here. Browser* popup_browser = Browser::CreateWithParams( Browser::CreateParams(Browser::TYPE_POPUP, browser()->profile())); AddTabWithURL(popup_browser, ntp_url_); popup_browser->window()->Show(); return popup_browser; } // Create incognito browser. Browser* CreateIncognitoBrowser() { // This is mostly duplicated from InPocessBrowserTest::CreateBrowser, // except that an incognito browser is created here. Browser* incognito_browser = Browser::Create(browser()->profile()->GetOffTheRecordProfile()); AddTabWithURL(incognito_browser, ntp_url_); incognito_browser->window()->Show(); return incognito_browser; } void NavigateToURL(const GURL& url) { ui_test_utils::NavigateToURL(browser(), url); ui_test_utils::RunAllPendingInMessageLoop(); } // Navigate to URL with BeforeUnload handler. void NavigateToBeforeUnloadURL() { const std::string kBeforeUnloadHtml = "beforeunload" "" ""; NavigateToURL(GURL("data:text/html," + kBeforeUnloadHtml)); } // Data members. GURL blank_url_; GURL ntp_url_; GURL other_url_; }; // This is used to block until a new tab in the specified browser is inserted. class NewTabObserver : public TabStripModelObserver { public: explicit NewTabObserver(Browser* browser) : browser_(browser) { browser_->tabstrip_model()->AddObserver(this); ui_test_utils::RunMessageLoop(); } virtual ~NewTabObserver() { browser_->tabstrip_model()->RemoveObserver(this); } private: virtual void TabInsertedAt(TabContentsWrapper* contents, int index, bool foreground) { MessageLoopForUI::current()->Quit(); } Browser* browser_; }; // Tests with the only tab in the only normal browser: // - if tab is about:blank, it is closeable // - if tab is NewTabPage, it is not closeable // - if tab is other url, it is closeable IN_PROC_BROWSER_TEST_F(TabCloseableStateWatcherTest, OneNormalBrowserWithOneTab) { // Check that default about::blank tab is closeable. ASSERT_EQ(1, browser()->tab_count()); EXPECT_TRUE(CanCloseTab(browser())); // Naviate tab to NewTabPage, and check that it's not closeable. NavigateToURL(ntp_url_); EXPECT_FALSE(CanCloseTab(browser())); // Navigate tab to any other URL, and check that it's closeable. NavigateToURL(other_url_); EXPECT_TRUE(CanCloseTab(browser())); } // Tests with 2 tabs in the only normal browser // - as long as there's > 1 tab, all tabs in the browser are always closeable IN_PROC_BROWSER_TEST_F(TabCloseableStateWatcherTest, OneNormalBrowserWithTwoTabs) { // 1 NewTabPage with any other tab are closeable. // First, set up the first NewTabPage. NavigateToURL(ntp_url_); EXPECT_FALSE(CanCloseTab(browser())); // Add the 2nd tab with blank page. AddTabWithURL(browser(), blank_url_); ASSERT_EQ(2, browser()->tab_count()); EXPECT_TRUE(CanCloseTab(browser())); // Navigate 2nd tab to other URL. NavigateToURL(other_url_); EXPECT_TRUE(CanCloseTab(browser())); // Navigate 2nd tab to NewTabPage. NavigateToURL(ntp_url_); EXPECT_TRUE(CanCloseTab(browser())); // Close 1st NewTabPage. browser()->tabstrip_model()->CloseTabContentsAt(0, TabStripModel::CLOSE_CREATE_HISTORICAL_TAB); EXPECT_FALSE(CanCloseTab(browser())); } // Tests with one tab in one normal browser and another non-normal browser. // - non-normal browser with any tab(s) is always closeable. // - non-normal browser does not affect closeable state of tab(s) in normal // browser(s). IN_PROC_BROWSER_TEST_F(TabCloseableStateWatcherTest, SecondNonNormalBrowser) { // Open non-normal browser. Browser* popup_browser = CreatePopupBrowser(); EXPECT_TRUE(CanCloseTab(browser())); EXPECT_TRUE(CanCloseTab(popup_browser)); // Navigate to NewTabPage for 1st browser. NavigateToURL(ntp_url_); EXPECT_FALSE(CanCloseTab(browser())); EXPECT_TRUE(CanCloseTab(popup_browser)); // Close non-normal browser. popup_browser->CloseWindow(); EXPECT_FALSE(CanCloseTab(browser())); } // Tests closing a closeable tab - tab should be closed, browser should remain // opened with a NewTabPage. IN_PROC_BROWSER_TEST_F(TabCloseableStateWatcherTest, CloseCloseableTab) { EXPECT_EQ(1, browser()->tab_count()); EXPECT_TRUE(CanCloseTab(browser())); browser()->CloseTab(); EXPECT_EQ(1, browser()->tab_count()); EXPECT_EQ(ntp_url_, browser()->GetSelectedWebContents()->GetURL()); } // Tests closing a closeable browser - all tabs in browser should be closed, // browser should remain opened with a NewTabPage. IN_PROC_BROWSER_TEST_F(TabCloseableStateWatcherTest, CloseCloseableBrowser) { EXPECT_EQ(1, browser()->tab_count()); EXPECT_TRUE(CanCloseTab(browser())); browser()->CloseWindow(); EXPECT_EQ(1u, BrowserList::size()); EXPECT_EQ(browser(), *(BrowserList::begin())); EXPECT_EQ(1, browser()->tab_count()); EXPECT_EQ(ntp_url_, browser()->GetSelectedWebContents()->GetURL()); } // Tests closing a non-closeable tab and hence non-closeable browser - tab and // browser should remain opened. IN_PROC_BROWSER_TEST_F(TabCloseableStateWatcherTest, CloseNonCloseableTabAndBrowser) { // Close non-closeable tab. EXPECT_EQ(1, browser()->tab_count()); NavigateToURL(ntp_url_); EXPECT_FALSE(CanCloseTab(browser())); WebContents* web_contents = browser()->GetSelectedWebContents(); browser()->CloseTab(); EXPECT_EQ(1, browser()->tab_count()); EXPECT_EQ(web_contents, browser()->GetSelectedWebContents()); // Close browser with non-closeable tab. browser()->CloseWindow(); EXPECT_EQ(1u, BrowserList::size()); EXPECT_EQ(browser(), *(BrowserList::begin())); EXPECT_EQ(1, browser()->tab_count()); EXPECT_EQ(web_contents, browser()->GetSelectedWebContents()); } // Tests an incognito browsr with a normal browser. // - when incognito browser is opened, all browsers (including previously // non-clsoeable normal browsers) become closeable. // - when incognito browser is closed, normal browsers return to adhering to the // original closebable rules. IN_PROC_BROWSER_TEST_F(TabCloseableStateWatcherTest, SecondIncognitoBrowser) { NavigateToURL(ntp_url_); EXPECT_FALSE(CanCloseTab(browser())); // Open an incognito browser. Browser* incognito_browser = CreateIncognitoBrowser(); EXPECT_TRUE(incognito_browser->profile()->IsOffTheRecord()); EXPECT_EQ(2u, BrowserList::size()); EXPECT_TRUE(CanCloseTab(browser())); EXPECT_TRUE(CanCloseTab(incognito_browser)); // Close incognito browser. incognito_browser->CloseWindow(); ui_test_utils::RunAllPendingInMessageLoop(); EXPECT_EQ(1u, BrowserList::size()); EXPECT_EQ(browser(), *(BrowserList::begin())); EXPECT_FALSE(CanCloseTab(browser())); } // Tests closing an incognito browser - the incognito browser should close, // and a new normal browser opened with a NewTabPage (which is not closeable). IN_PROC_BROWSER_TEST_F(TabCloseableStateWatcherTest, CloseIncognitoBrowser) { NavigateToURL(ntp_url_); // Open an incognito browser. Browser* incognito_browser = CreateIncognitoBrowser(); EXPECT_TRUE(incognito_browser->profile()->IsOffTheRecord()); EXPECT_EQ(2u, BrowserList::size()); // Close 1st normal browser. browser()->CloseWindow(); ui_test_utils::RunAllPendingInMessageLoop(); EXPECT_EQ(1u, BrowserList::size()); EXPECT_EQ(incognito_browser, *(BrowserList::begin())); EXPECT_TRUE(CanCloseTab(incognito_browser)); // Close incognito browser. incognito_browser->CloseWindow(); Browser* new_browser = ui_test_utils::WaitForNewBrowser(); EXPECT_EQ(1u, BrowserList::size()); EXPECT_FALSE(new_browser->profile()->IsOffTheRecord()); EXPECT_EQ(1, new_browser->tab_count()); EXPECT_EQ(ntp_url_, new_browser->GetSelectedWebContents()->GetURL()); } // Tests closing of browser with BeforeUnload handler where user clicks cancel // (i.e. stay on the page and cancel closing) - browser and its tabs should stay // the same. // Sporadically crashing test. See http://crbug.com/79333 IN_PROC_BROWSER_TEST_F(TabCloseableStateWatcherTest, DISABLED_CloseBrowserWithBeforeUnloadHandlerCancel) { // Navigate to URL with BeforeUnload handler. NavigateToBeforeUnloadURL(); EXPECT_TRUE(CanCloseTab(browser())); // Close browser, click Cancel in BeforeUnload confirm dialog. WebContents* web_contents = browser()->GetSelectedWebContents(); browser()->CloseWindow(); AppModalDialog* confirm = ui_test_utils::WaitForAppModalDialog(); confirm->native_dialog()->CancelAppModalDialog(); ui_test_utils::RunAllPendingInMessageLoop(); EXPECT_EQ(1u, BrowserList::size()); EXPECT_EQ(browser(), *(BrowserList::begin())); EXPECT_EQ(1, browser()->tab_count()); EXPECT_EQ(web_contents, browser()->GetSelectedWebContents()); // Close the browser. browser()->CloseWindow(); confirm = ui_test_utils::WaitForAppModalDialog(); confirm->native_dialog()->AcceptAppModalDialog(); ui_test_utils::RunAllPendingInMessageLoop(); } // Tests closing of browser with BeforeUnload handler where user clicks OK (i.e. // leave the page and proceed with closing), all tabs in browser should close, // browser remains opened with a NewTabPage. IN_PROC_BROWSER_TEST_F(TabCloseableStateWatcherTest, CloseBrowserWithBeforeUnloadHandlerOK) { // Navigate to URL with BeforeUnload handler. NavigateToBeforeUnloadURL(); EXPECT_TRUE(CanCloseTab(browser())); // Close browser, click OK in BeforeUnload confirm dialog. browser()->CloseWindow(); AppModalDialog* confirm = ui_test_utils::WaitForAppModalDialog(); confirm->native_dialog()->AcceptAppModalDialog(); NewTabObserver new_tab_observer(browser()); EXPECT_EQ(1u, BrowserList::size()); EXPECT_EQ(browser(), *(BrowserList::begin())); EXPECT_EQ(1, browser()->tab_count()); EXPECT_EQ(ntp_url_, browser()->GetSelectedWebContents()->GetURL()); } } // namespace chromeos