// Copyright (c) 2011 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 "build/build_config.h" #if defined(TOOLKIT_GTK) #include <gtk/gtk.h> #endif #include "chrome/browser/extensions/browser_action_test_util.h" #include "chrome/browser/extensions/extension_apitest.h" #include "chrome/browser/extensions/extension_browser_event_router.h" #include "chrome/browser/extensions/extension_service.h" #include "chrome/browser/extensions/extension_tabs_module.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/browser_window.h" #include "chrome/common/extensions/extension_action.h" #include "chrome/common/url_constants.h" #include "chrome/test/ui_test_utils.h" #include "content/browser/tab_contents/tab_contents.h" #include "ui/gfx/rect.h" #include "ui/gfx/size.h" class BrowserActionApiTest : public ExtensionApiTest { public: BrowserActionApiTest() {} virtual ~BrowserActionApiTest() {} protected: BrowserActionTestUtil GetBrowserActionsBar() { return BrowserActionTestUtil(browser()); } bool OpenPopup(int index) { ResultCatcher catcher; GetBrowserActionsBar().Press(index); ui_test_utils::WaitForNotification( NotificationType::EXTENSION_POPUP_VIEW_READY); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); return GetBrowserActionsBar().HasPopup(); } }; IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, Basic) { ASSERT_TRUE(test_server()->Start()); ASSERT_TRUE(RunExtensionTest("browser_action/basics")) << message_; const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; // Test that there is a browser action in the toolbar. ASSERT_EQ(1, GetBrowserActionsBar().NumberOfBrowserActions()); // Tell the extension to update the browser action state. ResultCatcher catcher; ui_test_utils::NavigateToURL(browser(), GURL(extension->GetResourceURL("update.html"))); ASSERT_TRUE(catcher.GetNextResult()); // Test that we received the changes. ExtensionAction* action = extension->browser_action(); ASSERT_EQ("Modified", action->GetTitle(ExtensionAction::kDefaultTabId)); ASSERT_EQ("badge", action->GetBadgeText(ExtensionAction::kDefaultTabId)); ASSERT_EQ(SkColorSetARGB(255, 255, 255, 255), action->GetBadgeBackgroundColor(ExtensionAction::kDefaultTabId)); // Simulate the browser action being clicked. ui_test_utils::NavigateToURL(browser(), test_server()->GetURL("files/extensions/test_file.txt")); ExtensionService* service = browser()->profile()->GetExtensionService(); service->browser_event_router()->BrowserActionExecuted( browser()->profile(), action->extension_id(), browser()); // Verify the command worked. TabContents* tab = browser()->GetSelectedTabContents(); bool result = false; ASSERT_TRUE(ui_test_utils::ExecuteJavaScriptAndExtractBool( tab->render_view_host(), L"", L"setInterval(function(){" L" if(document.body.bgColor == 'red'){" L" window.domAutomationController.send(true)}}, 100)", &result)); ASSERT_TRUE(result); } IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, DynamicBrowserAction) { ASSERT_TRUE(RunExtensionTest("browser_action/no_icon")) << message_; const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; // Test that there is a browser action in the toolbar and that it has no icon. ASSERT_EQ(1, GetBrowserActionsBar().NumberOfBrowserActions()); EXPECT_FALSE(GetBrowserActionsBar().HasIcon(0)); // Tell the extension to update the icon using setIcon({imageData:...}). ResultCatcher catcher; ui_test_utils::NavigateToURL(browser(), GURL(extension->GetResourceURL("update.html"))); ASSERT_TRUE(catcher.GetNextResult()); // Test that we received the changes. EXPECT_TRUE(GetBrowserActionsBar().HasIcon(0)); // Tell the extension to update using setIcon({path:...}); ui_test_utils::NavigateToURL(browser(), GURL(extension->GetResourceURL("update2.html"))); ASSERT_TRUE(catcher.GetNextResult()); // Test that we received the changes. EXPECT_TRUE(GetBrowserActionsBar().HasIcon(0)); // TODO(aa): Would be nice here to actually compare that the pixels change. } // This test is flaky as per http://crbug.com/74557. IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, FLAKY_TabSpecificBrowserActionState) { ASSERT_TRUE(RunExtensionTest("browser_action/tab_specific_state")) << message_; const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; // Test that there is a browser action in the toolbar and that it has an icon. ASSERT_EQ(1, GetBrowserActionsBar().NumberOfBrowserActions()); EXPECT_TRUE(GetBrowserActionsBar().HasIcon(0)); // Execute the action, its title should change. ResultCatcher catcher; GetBrowserActionsBar().Press(0); ASSERT_TRUE(catcher.GetNextResult()); EXPECT_EQ("Showing icon 2", GetBrowserActionsBar().GetTooltip(0)); // Open a new tab, the title should go back. browser()->NewTab(); EXPECT_EQ("hi!", GetBrowserActionsBar().GetTooltip(0)); // Go back to first tab, changed title should reappear. browser()->ActivateTabAt(0, true); EXPECT_EQ("Showing icon 2", GetBrowserActionsBar().GetTooltip(0)); // Reload that tab, default title should come back. ui_test_utils::NavigateToURL(browser(), GURL("about:blank")); EXPECT_EQ("hi!", GetBrowserActionsBar().GetTooltip(0)); } // http://code.google.com/p/chromium/issues/detail?id=70829 // Only mac is okay. #if !defined(OS_MACOSX) #define MAYBE_BrowserActionPopup DISABLED_BrowserActionPopup #else #define MAYBE_BrowserActionPopup BrowserActionPopup #endif IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, MAYBE_BrowserActionPopup) { ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII( "browser_action/popup"))); BrowserActionTestUtil actions_bar = GetBrowserActionsBar(); const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; // The extension's popup's size grows by |growFactor| each click. const int growFactor = 500; gfx::Size minSize = BrowserActionTestUtil::GetMinPopupSize(); gfx::Size middleSize = gfx::Size(growFactor, growFactor); gfx::Size maxSize = BrowserActionTestUtil::GetMaxPopupSize(); // Ensure that two clicks will exceed the maximum allowed size. ASSERT_GT(minSize.height() + growFactor * 2, maxSize.height()); ASSERT_GT(minSize.width() + growFactor * 2, maxSize.width()); // Simulate a click on the browser action and verify the size of the resulting // popup. The first one tries to be 0x0, so it should be the min values. ASSERT_TRUE(OpenPopup(0)); EXPECT_EQ(minSize, actions_bar.GetPopupBounds().size()); EXPECT_TRUE(actions_bar.HidePopup()); ASSERT_TRUE(OpenPopup(0)); EXPECT_EQ(middleSize, actions_bar.GetPopupBounds().size()); EXPECT_TRUE(actions_bar.HidePopup()); // One more time, but this time it should be constrained by the max values. ASSERT_TRUE(OpenPopup(0)); EXPECT_EQ(maxSize, actions_bar.GetPopupBounds().size()); EXPECT_TRUE(actions_bar.HidePopup()); } // Test that calling chrome.browserAction.setPopup() can enable and change // a popup. IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, BrowserActionAddPopup) { ASSERT_TRUE(RunExtensionTest("browser_action/add_popup")) << message_; const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; int tab_id = ExtensionTabUtil::GetTabId(browser()->GetSelectedTabContents()); ExtensionAction* browser_action = extension->browser_action(); ASSERT_TRUE(browser_action) << "Browser action test extension should have a browser action."; ASSERT_FALSE(browser_action->HasPopup(tab_id)); ASSERT_FALSE(browser_action->HasPopup(ExtensionAction::kDefaultTabId)); // Simulate a click on the browser action icon. The onClicked handler // will add a popup. { ResultCatcher catcher; GetBrowserActionsBar().Press(0); ASSERT_TRUE(catcher.GetNextResult()); } // The call to setPopup in background.html set a tab id, so the // current tab's setting should have changed, but the default setting // should not have changed. ASSERT_TRUE(browser_action->HasPopup(tab_id)) << "Clicking on the browser action should have caused a popup to " << "be added."; ASSERT_FALSE(browser_action->HasPopup(ExtensionAction::kDefaultTabId)) << "Clicking on the browser action should not have set a default " << "popup."; ASSERT_STREQ("/a_popup.html", browser_action->GetPopupUrl(tab_id).path().c_str()); // Now change the popup from a_popup.html to another_popup.html by loading // a page which removes the popup using chrome.browserAction.setPopup(). { ResultCatcher catcher; ui_test_utils::NavigateToURL( browser(), GURL(extension->GetResourceURL("change_popup.html"))); ASSERT_TRUE(catcher.GetNextResult()); } // The call to setPopup in change_popup.html did not use a tab id, // so the default setting should have changed as well as the current tab. ASSERT_TRUE(browser_action->HasPopup(tab_id)); ASSERT_TRUE(browser_action->HasPopup(ExtensionAction::kDefaultTabId)); ASSERT_STREQ("/another_popup.html", browser_action->GetPopupUrl(tab_id).path().c_str()); } // Test that calling chrome.browserAction.setPopup() can remove a popup. IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, BrowserActionRemovePopup) { // Load the extension, which has a browser action with a default popup. ASSERT_TRUE(RunExtensionTest("browser_action/remove_popup")) << message_; const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; int tab_id = ExtensionTabUtil::GetTabId(browser()->GetSelectedTabContents()); ExtensionAction* browser_action = extension->browser_action(); ASSERT_TRUE(browser_action) << "Browser action test extension should have a browser action."; ASSERT_TRUE(browser_action->HasPopup(tab_id)) << "Expect a browser action popup before the test removes it."; ASSERT_TRUE(browser_action->HasPopup(ExtensionAction::kDefaultTabId)) << "Expect a browser action popup is the default for all tabs."; // Load a page which removes the popup using chrome.browserAction.setPopup(). { ResultCatcher catcher; ui_test_utils::NavigateToURL( browser(), GURL(extension->GetResourceURL("remove_popup.html"))); ASSERT_TRUE(catcher.GetNextResult()); } ASSERT_FALSE(browser_action->HasPopup(tab_id)) << "Browser action popup should have been removed."; ASSERT_TRUE(browser_action->HasPopup(ExtensionAction::kDefaultTabId)) << "Browser action popup default should not be changed by setting " << "a specific tab id."; } IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, IncognitoBasic) { ASSERT_TRUE(test_server()->Start()); ASSERT_TRUE(RunExtensionTest("browser_action/basics")) << message_; const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; // Test that there is a browser action in the toolbar. ASSERT_EQ(1, GetBrowserActionsBar().NumberOfBrowserActions()); // Open an incognito window and test that the browser action isn't there by // default. Profile* incognito_profile = browser()->profile()->GetOffTheRecordProfile(); Browser* incognito_browser = Browser::Create(incognito_profile); ASSERT_EQ(0, BrowserActionTestUtil(incognito_browser).NumberOfBrowserActions()); // Now enable the extension in incognito mode, and test that the browser // action shows up. Note that we don't update the existing window at the // moment, so we just create a new one. browser()->profile()->GetExtensionService()->extension_prefs()-> SetIsIncognitoEnabled(extension->id(), true); incognito_browser->CloseWindow(); incognito_browser = Browser::Create(incognito_profile); ASSERT_EQ(1, BrowserActionTestUtil(incognito_browser).NumberOfBrowserActions()); // TODO(mpcomplete): simulate a click and have it do the right thing in // incognito. } IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, IncognitoDragging) { ExtensionService* service = browser()->profile()->GetExtensionService(); // The tooltips for each respective browser action. const char kTooltipA[] = "Make this page red"; const char kTooltipB[] = "grow"; const char kTooltipC[] = "Test setPopup()"; const size_t size_before = service->extensions()->size(); ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII( "browser_action/basics"))); ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII( "browser_action/popup"))); ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII( "browser_action/add_popup"))); // Test that there are 3 browser actions in the toolbar. ASSERT_EQ(size_before + 3, service->extensions()->size()); ASSERT_EQ(3, GetBrowserActionsBar().NumberOfBrowserActions()); // Now enable 2 of the extensions in incognito mode, and test that the browser // actions show up. service->extension_prefs()->SetIsIncognitoEnabled( service->extensions()->at(size_before)->id(), true); service->extension_prefs()->SetIsIncognitoEnabled( service->extensions()->at(size_before + 2)->id(), true); Profile* incognito_profile = browser()->profile()->GetOffTheRecordProfile(); Browser* incognito_browser = Browser::Create(incognito_profile); BrowserActionTestUtil incognito_bar(incognito_browser); // Navigate just to have a tab in this window, otherwise wonky things happen. ui_test_utils::OpenURLOffTheRecord(browser()->profile(), GURL(chrome::kChromeUIExtensionsURL)); ASSERT_EQ(2, incognito_bar.NumberOfBrowserActions()); // Ensure that the browser actions are in the right order (ABC). EXPECT_EQ(kTooltipA, GetBrowserActionsBar().GetTooltip(0)); EXPECT_EQ(kTooltipB, GetBrowserActionsBar().GetTooltip(1)); EXPECT_EQ(kTooltipC, GetBrowserActionsBar().GetTooltip(2)); EXPECT_EQ(kTooltipA, incognito_bar.GetTooltip(0)); EXPECT_EQ(kTooltipC, incognito_bar.GetTooltip(1)); // Now rearrange them and ensure that they are rearranged correctly in both // regular and incognito mode. // ABC -> CAB service->toolbar_model()->MoveBrowserAction( service->extensions()->at(size_before + 2), 0); EXPECT_EQ(kTooltipC, GetBrowserActionsBar().GetTooltip(0)); EXPECT_EQ(kTooltipA, GetBrowserActionsBar().GetTooltip(1)); EXPECT_EQ(kTooltipB, GetBrowserActionsBar().GetTooltip(2)); EXPECT_EQ(kTooltipC, incognito_bar.GetTooltip(0)); EXPECT_EQ(kTooltipA, incognito_bar.GetTooltip(1)); // CAB -> CBA service->toolbar_model()->MoveBrowserAction( service->extensions()->at(size_before + 1), 1); EXPECT_EQ(kTooltipC, GetBrowserActionsBar().GetTooltip(0)); EXPECT_EQ(kTooltipB, GetBrowserActionsBar().GetTooltip(1)); EXPECT_EQ(kTooltipA, GetBrowserActionsBar().GetTooltip(2)); EXPECT_EQ(kTooltipC, incognito_bar.GetTooltip(0)); EXPECT_EQ(kTooltipA, incognito_bar.GetTooltip(1)); }