// 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 #include "base/macros.h" #include "build/build_config.h" #include "chrome/browser/extensions/api/extension_action/extension_action_api.h" #include "chrome/browser/extensions/browser_action_test_util.h" #include "chrome/browser/extensions/extension_action.h" #include "chrome/browser/extensions/extension_action_icon_factory.h" #include "chrome/browser/extensions/extension_action_manager.h" #include "chrome/browser/extensions/extension_action_runner.h" #include "chrome/browser/extensions/extension_apitest.h" #include "chrome/browser/extensions/extension_tab_util.h" #include "chrome/browser/extensions/extension_util.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/browser_commands.h" #include "chrome/browser/ui/browser_finder.h" #include "chrome/browser/ui/browser_navigator_params.h" #include "chrome/browser/ui/browser_window.h" #include "chrome/browser/ui/tabs/tab_strip_model.h" #include "chrome/common/url_constants.h" #include "chrome/test/base/ui_test_utils.h" #include "content/public/browser/notification_service.h" #include "content/public/browser/render_frame_host.h" #include "content/public/browser/web_contents.h" #include "content/public/test/browser_test_utils.h" #include "extensions/browser/extension_registry.h" #include "extensions/browser/extension_system.h" #include "extensions/browser/notification_types.h" #include "extensions/browser/process_manager.h" #include "extensions/browser/test_extension_registry_observer.h" #include "extensions/common/feature_switch.h" #include "extensions/test/extension_test_message_listener.h" #include "extensions/test/result_catcher.h" #include "grit/theme_resources.h" #include "net/dns/mock_host_resolver.h" #include "net/test/embedded_test_server/embedded_test_server.h" #include "ui/base/resource/resource_bundle.h" #include "ui/gfx/geometry/rect.h" #include "ui/gfx/geometry/size.h" #include "ui/gfx/image/canvas_image_source.h" #include "ui/gfx/image/image_skia.h" #include "ui/gfx/image/image_skia_operations.h" #include "ui/gfx/image/image_unittest_util.h" #include "ui/gfx/skia_util.h" using content::WebContents; namespace extensions { namespace { void ExecuteExtensionAction(Browser* browser, const Extension* extension) { ExtensionActionRunner::GetForWebContents( browser->tab_strip_model()->GetActiveWebContents()) ->RunAction(extension, true); } // An ImageSkia source that will do nothing (i.e., have a blank skia). We need // this because we need a blank canvas at a certain size, and that can't be done // by just using a null ImageSkia. class BlankImageSource : public gfx::CanvasImageSource { public: explicit BlankImageSource(const gfx::Size& size) : gfx::CanvasImageSource(size, false) {} ~BlankImageSource() override {} void Draw(gfx::Canvas* canvas) override {} private: DISALLOW_COPY_AND_ASSIGN(BlankImageSource); }; const char kEmptyImageDataError[] = "The imageData property must contain an ImageData object or dictionary " "of ImageData objects."; const char kEmptyPathError[] = "The path property must not be empty."; // Makes sure |bar_rendering| has |model_icon| in the middle (there's additional // padding that correlates to the rest of the button, and this is ignored). void VerifyIconsMatch(const gfx::Image& bar_rendering, const gfx::Image& model_icon) { gfx::Rect icon_portion(gfx::Point(), bar_rendering.Size()); icon_portion.ClampToCenteredSize(model_icon.Size()); EXPECT_TRUE(gfx::test::AreBitmapsEqual( model_icon.AsImageSkia().GetRepresentation(1.0f).sk_bitmap(), gfx::ImageSkiaOperations::ExtractSubset(bar_rendering.AsImageSkia(), icon_portion) .GetRepresentation(1.0f) .sk_bitmap())); } class BrowserActionApiTest : public ExtensionApiTest { public: BrowserActionApiTest() {} ~BrowserActionApiTest() override {} protected: BrowserActionTestUtil* GetBrowserActionsBar() { if (!browser_action_test_util_) browser_action_test_util_.reset(new BrowserActionTestUtil(browser())); return browser_action_test_util_.get(); } bool OpenPopup(int index) { ResultCatcher catcher; content::WindowedNotificationObserver popup_observer( content::NOTIFICATION_LOAD_COMPLETED_MAIN_FRAME, content::NotificationService::AllSources()); GetBrowserActionsBar()->Press(index); popup_observer.Wait(); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); return GetBrowserActionsBar()->HasPopup(); } ExtensionAction* GetBrowserAction(const Extension& extension) { return ExtensionActionManager::Get(browser()->profile())-> GetBrowserAction(extension); } private: scoped_ptr browser_action_test_util_; DISALLOW_COPY_AND_ASSIGN(BrowserActionApiTest); }; IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, Basic) { ASSERT_TRUE(embedded_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()) << catcher.message(); // Test that we received the changes. ExtensionAction* action = GetBrowserAction(*extension); 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(), embedded_test_server()->GetURL("/extensions/test_file.txt")); ExecuteExtensionAction(browser(), extension); ASSERT_TRUE(catcher.GetNextResult()) << catcher.message(); } IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, DynamicBrowserAction) { ASSERT_TRUE(RunExtensionTest("browser_action/no_icon")) << message_; const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; #if defined (OS_MACOSX) // We need this on mac so we don't loose 2x representations from browser icon // in transformations gfx::ImageSkia -> NSImage -> gfx::ImageSkia. std::vector supported_scale_factors; supported_scale_factors.push_back(ui::SCALE_FACTOR_100P); supported_scale_factors.push_back(ui::SCALE_FACTOR_200P); ui::SetSupportedScaleFactors(supported_scale_factors); #endif // We should not be creating icons asynchronously, so we don't need an // observer. ExtensionActionIconFactory icon_factory( profile(), extension, GetBrowserAction(*extension), NULL); // Test that there is a browser action in the toolbar. ASSERT_EQ(1, GetBrowserActionsBar()->NumberOfBrowserActions()); EXPECT_TRUE(GetBrowserActionsBar()->HasIcon(0)); gfx::Image action_icon = icon_factory.GetIcon(0); uint32_t action_icon_last_id = action_icon.ToSkBitmap()->getGenerationID(); // Let's check that |GetIcon| doesn't always return bitmap with new id. ASSERT_EQ(action_icon_last_id, icon_factory.GetIcon(0).ToSkBitmap()->getGenerationID()); gfx::Image last_bar_icon = GetBrowserActionsBar()->GetIcon(0); EXPECT_TRUE(gfx::test::AreImagesEqual(last_bar_icon, GetBrowserActionsBar()->GetIcon(0))); // The reason we don't test more standard scales (like 1x, 2x, etc.) is that // these may be generated from the provided scales. float kSmallIconScale = 21.f / ExtensionAction::ActionIconSize(); float kLargeIconScale = 42.f / ExtensionAction::ActionIconSize(); ASSERT_FALSE(ui::IsSupportedScale(kSmallIconScale)); ASSERT_FALSE(ui::IsSupportedScale(kLargeIconScale)); // Tell the extension to update the icon using ImageData object. ResultCatcher catcher; GetBrowserActionsBar()->Press(0); ASSERT_TRUE(catcher.GetNextResult()); EXPECT_FALSE(gfx::test::AreImagesEqual(last_bar_icon, GetBrowserActionsBar()->GetIcon(0))); last_bar_icon = GetBrowserActionsBar()->GetIcon(0); action_icon = icon_factory.GetIcon(0); uint32_t action_icon_current_id = action_icon.ToSkBitmap()->getGenerationID(); EXPECT_GT(action_icon_current_id, action_icon_last_id); action_icon_last_id = action_icon_current_id; VerifyIconsMatch(last_bar_icon, action_icon); // Check that only the smaller size was set (only a 21px icon was provided). EXPECT_TRUE(action_icon.ToImageSkia()->HasRepresentation(kSmallIconScale)); EXPECT_FALSE(action_icon.ToImageSkia()->HasRepresentation(kLargeIconScale)); // Tell the extension to update the icon using path. GetBrowserActionsBar()->Press(0); ASSERT_TRUE(catcher.GetNextResult()); // Make sure the browser action bar updated. EXPECT_FALSE(gfx::test::AreImagesEqual(last_bar_icon, GetBrowserActionsBar()->GetIcon(0))); last_bar_icon = GetBrowserActionsBar()->GetIcon(0); action_icon = icon_factory.GetIcon(0); action_icon_current_id = action_icon.ToSkBitmap()->getGenerationID(); EXPECT_GT(action_icon_current_id, action_icon_last_id); action_icon_last_id = action_icon_current_id; VerifyIconsMatch(last_bar_icon, action_icon); // Check that only the smaller size was set (only a 21px icon was provided). EXPECT_TRUE(action_icon.ToImageSkia()->HasRepresentation(kSmallIconScale)); EXPECT_FALSE(action_icon.ToImageSkia()->HasRepresentation(kLargeIconScale)); // Tell the extension to update the icon using dictionary of ImageData // objects. GetBrowserActionsBar()->Press(0); ASSERT_TRUE(catcher.GetNextResult()); EXPECT_FALSE(gfx::test::AreImagesEqual(last_bar_icon, GetBrowserActionsBar()->GetIcon(0))); last_bar_icon = GetBrowserActionsBar()->GetIcon(0); action_icon = icon_factory.GetIcon(0); action_icon_current_id = action_icon.ToSkBitmap()->getGenerationID(); EXPECT_GT(action_icon_current_id, action_icon_last_id); action_icon_last_id = action_icon_current_id; VerifyIconsMatch(last_bar_icon, action_icon); // Check both sizes were set (as two icon sizes were provided). EXPECT_TRUE(action_icon.ToImageSkia()->HasRepresentation(kSmallIconScale)); EXPECT_TRUE(action_icon.AsImageSkia().HasRepresentation(kLargeIconScale)); // Tell the extension to update the icon using dictionary of paths. GetBrowserActionsBar()->Press(0); ASSERT_TRUE(catcher.GetNextResult()); EXPECT_FALSE(gfx::test::AreImagesEqual(last_bar_icon, GetBrowserActionsBar()->GetIcon(0))); last_bar_icon = GetBrowserActionsBar()->GetIcon(0); action_icon = icon_factory.GetIcon(0); action_icon_current_id = action_icon.ToSkBitmap()->getGenerationID(); EXPECT_GT(action_icon_current_id, action_icon_last_id); action_icon_last_id = action_icon_current_id; VerifyIconsMatch(last_bar_icon, action_icon); // Check both sizes were set (as two icon sizes were provided). EXPECT_TRUE(action_icon.ToImageSkia()->HasRepresentation(kSmallIconScale)); EXPECT_TRUE(action_icon.AsImageSkia().HasRepresentation(kLargeIconScale)); // Tell the extension to update the icon using dictionary of ImageData // objects, but setting only one size. GetBrowserActionsBar()->Press(0); ASSERT_TRUE(catcher.GetNextResult()); EXPECT_FALSE(gfx::test::AreImagesEqual(last_bar_icon, GetBrowserActionsBar()->GetIcon(0))); last_bar_icon = GetBrowserActionsBar()->GetIcon(0); action_icon = icon_factory.GetIcon(0); action_icon_current_id = action_icon.ToSkBitmap()->getGenerationID(); EXPECT_GT(action_icon_current_id, action_icon_last_id); action_icon_last_id = action_icon_current_id; VerifyIconsMatch(last_bar_icon, action_icon); // Check that only the smaller size was set (only a 21px icon was provided). EXPECT_TRUE(action_icon.ToImageSkia()->HasRepresentation(kSmallIconScale)); EXPECT_FALSE(action_icon.ToImageSkia()->HasRepresentation(kLargeIconScale)); // Tell the extension to update the icon using dictionary of paths, but // setting only one size. GetBrowserActionsBar()->Press(0); ASSERT_TRUE(catcher.GetNextResult()); EXPECT_FALSE(gfx::test::AreImagesEqual(last_bar_icon, GetBrowserActionsBar()->GetIcon(0))); last_bar_icon = GetBrowserActionsBar()->GetIcon(0); action_icon = icon_factory.GetIcon(0); action_icon_current_id = action_icon.ToSkBitmap()->getGenerationID(); EXPECT_GT(action_icon_current_id, action_icon_last_id); action_icon_last_id = action_icon_current_id; VerifyIconsMatch(last_bar_icon, action_icon); // Check that only the smaller size was set (only a 21px icon was provided). EXPECT_TRUE(action_icon.ToImageSkia()->HasRepresentation(kSmallIconScale)); EXPECT_FALSE(action_icon.ToImageSkia()->HasRepresentation(kLargeIconScale)); // Tell the extension to update the icon using dictionary of ImageData // objects, but setting only size 42. GetBrowserActionsBar()->Press(0); ASSERT_TRUE(catcher.GetNextResult()); EXPECT_FALSE(gfx::test::AreImagesEqual(last_bar_icon, GetBrowserActionsBar()->GetIcon(0))); last_bar_icon = GetBrowserActionsBar()->GetIcon(0); action_icon = icon_factory.GetIcon(0); action_icon_current_id = action_icon.ToSkBitmap()->getGenerationID(); EXPECT_GT(action_icon_current_id, action_icon_last_id); action_icon_last_id = action_icon_current_id; // Check that only the larger size was set (only a 42px icon was provided). EXPECT_FALSE(action_icon.ToImageSkia()->HasRepresentation(kSmallIconScale)); EXPECT_TRUE(action_icon.AsImageSkia().HasRepresentation(kLargeIconScale)); // Try setting icon with empty dictionary of ImageData objects. GetBrowserActionsBar()->Press(0); ASSERT_FALSE(catcher.GetNextResult()); EXPECT_EQ(kEmptyImageDataError, catcher.message()); // Try setting icon with empty dictionary of path objects. GetBrowserActionsBar()->Press(0); ASSERT_FALSE(catcher.GetNextResult()); EXPECT_EQ(kEmptyPathError, catcher.message()); } IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, 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. chrome::NewTab(browser()); EXPECT_EQ("hi!", GetBrowserActionsBar()->GetTooltip(0)); // Go back to first tab, changed title should reappear. browser()->tab_strip_model()->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 // Mac used to be ok, but then mac 10.5 started failing too. =( IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, DISABLED_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->GetPopupSize()); EXPECT_TRUE(actions_bar->HidePopup()); ASSERT_TRUE(OpenPopup(0)); EXPECT_EQ(middleSize, actions_bar->GetPopupSize()); 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->GetPopupSize()); 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()->tab_strip_model()->GetActiveWebContents()); ExtensionAction* browser_action = GetBrowserAction(*extension); 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()->tab_strip_model()->GetActiveWebContents()); ExtensionAction* browser_action = GetBrowserAction(*extension); 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(embedded_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(); base::RunLoop().RunUntilIdle(); // Wait for profile initialization. Browser* incognito_browser = new Browser(Browser::CreateParams(incognito_profile)); ASSERT_EQ(0, BrowserActionTestUtil(incognito_browser).NumberOfBrowserActions()); // Now enable the extension in incognito mode, and test that the browser // action shows up. // SetIsIncognitoEnabled() requires a reload of the extension, so we have to // wait for it. TestExtensionRegistryObserver registry_observer( ExtensionRegistry::Get(profile()), extension->id()); extensions::util::SetIsIncognitoEnabled( extension->id(), browser()->profile(), true); registry_observer.WaitForExtensionLoaded(); ASSERT_EQ(1, BrowserActionTestUtil(incognito_browser).NumberOfBrowserActions()); // TODO(mpcomplete): simulate a click and have it do the right thing in // incognito. } // Tests that events are dispatched to the correct profile for split mode // extensions. IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, IncognitoSplit) { ResultCatcher catcher; const Extension* extension = LoadExtensionWithFlags( test_data_dir_.AppendASCII("browser_action/split_mode"), kFlagEnableIncognito); ASSERT_TRUE(extension) << message_; // Open an incognito window. Profile* incognito_profile = browser()->profile()->GetOffTheRecordProfile(); Browser* incognito_browser = new Browser(Browser::CreateParams(incognito_profile)); base::RunLoop().RunUntilIdle(); // Wait for profile initialization. // Navigate just to have a tab in this window, otherwise wonky things happen. OpenURLOffTheRecord(browser()->profile(), GURL("about:blank")); ASSERT_EQ(1, BrowserActionTestUtil(incognito_browser).NumberOfBrowserActions()); // A click in the regular profile should open a tab in the regular profile. ExecuteExtensionAction(browser(), extension); ASSERT_TRUE(catcher.GetNextResult()) << catcher.message(); // A click in the incognito profile should open a tab in the // incognito profile. ExecuteExtensionAction(incognito_browser, extension); ASSERT_TRUE(catcher.GetNextResult()) << catcher.message(); } // Disabled because of failures (crashes) on ASAN bot. // See http://crbug.com/98861. IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, DISABLED_CloseBackgroundPage) { ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("browser_action/close_background"))); const Extension* extension = GetSingleLoadedExtension(); // There is a background page and a browser action with no badge text. extensions::ProcessManager* manager = extensions::ProcessManager::Get(browser()->profile()); ASSERT_TRUE(manager->GetBackgroundHostForExtension(extension->id())); ExtensionAction* action = GetBrowserAction(*extension); ASSERT_EQ("", action->GetBadgeText(ExtensionAction::kDefaultTabId)); content::WindowedNotificationObserver host_destroyed_observer( extensions::NOTIFICATION_EXTENSION_HOST_DESTROYED, content::NotificationService::AllSources()); // Click the browser action. ExecuteExtensionAction(browser(), extension); // It can take a moment for the background page to actually get destroyed // so we wait for the notification before checking that it's really gone // and the badge text has been set. host_destroyed_observer.Wait(); ASSERT_FALSE(manager->GetBackgroundHostForExtension(extension->id())); ASSERT_EQ("X", action->GetBadgeText(ExtensionAction::kDefaultTabId)); } IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, BadgeBackgroundColor) { ASSERT_TRUE(embedded_test_server()->Start()); ASSERT_TRUE(RunExtensionTest("browser_action/color")) << message_; const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; // Test that there is a browser action in the toolbar. ASSERT_EQ(1, GetBrowserActionsBar()->NumberOfBrowserActions()); // Test that CSS values (#FF0000) set color correctly. ExtensionAction* action = GetBrowserAction(*extension); ASSERT_EQ(SkColorSetARGB(255, 255, 0, 0), action->GetBadgeBackgroundColor(ExtensionAction::kDefaultTabId)); // 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 CSS values (#0F0) set color correctly. ASSERT_EQ(SkColorSetARGB(255, 0, 255, 0), action->GetBadgeBackgroundColor(ExtensionAction::kDefaultTabId)); ui_test_utils::NavigateToURL(browser(), GURL(extension->GetResourceURL("update2.html"))); ASSERT_TRUE(catcher.GetNextResult()); // Test that array values set color correctly. ASSERT_EQ(SkColorSetARGB(255, 255, 255, 255), action->GetBadgeBackgroundColor(ExtensionAction::kDefaultTabId)); ui_test_utils::NavigateToURL(browser(), GURL(extension->GetResourceURL("update3.html"))); ASSERT_TRUE(catcher.GetNextResult()); // Test that hsl() values 'hsl(120, 100%, 50%)' set color correctly. ASSERT_EQ(SkColorSetARGB(255, 0, 255, 0), action->GetBadgeBackgroundColor(ExtensionAction::kDefaultTabId)); // Test basic color keyword set correctly. ui_test_utils::NavigateToURL(browser(), GURL(extension->GetResourceURL("update4.html"))); ASSERT_TRUE(catcher.GetNextResult()); ASSERT_EQ(SkColorSetARGB(255, 0, 0, 255), action->GetBadgeBackgroundColor(ExtensionAction::kDefaultTabId)); } IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, Getters) { ASSERT_TRUE(RunExtensionTest("browser_action/getters")) << message_; const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; // Test that there is a browser action in the toolbar. ASSERT_EQ(1, GetBrowserActionsBar()->NumberOfBrowserActions()); // Test the getters for defaults. ResultCatcher catcher; ui_test_utils::NavigateToURL(browser(), GURL(extension->GetResourceURL("update.html"))); ASSERT_TRUE(catcher.GetNextResult()); // Test the getters for a specific tab. ui_test_utils::NavigateToURL(browser(), GURL(extension->GetResourceURL("update2.html"))); ASSERT_TRUE(catcher.GetNextResult()); } // Verify triggering browser action. IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, TestTriggerBrowserAction) { ASSERT_TRUE(embedded_test_server()->Start()); ASSERT_TRUE(RunExtensionTest("trigger_actions/browser_action")) << message_; const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; // Test that there is a browser action in the toolbar. ASSERT_EQ(1, GetBrowserActionsBar()->NumberOfBrowserActions()); ui_test_utils::NavigateToURL(browser(), embedded_test_server()->GetURL("/simple.html")); ExtensionAction* browser_action = GetBrowserAction(*extension); EXPECT_TRUE(browser_action != NULL); // Simulate a click on the browser action icon. { ResultCatcher catcher; GetBrowserActionsBar()->Press(0); EXPECT_TRUE(catcher.GetNextResult()); } WebContents* tab = browser()->tab_strip_model()->GetActiveWebContents(); EXPECT_TRUE(tab != NULL); // Verify that the browser action turned the background color red. const std::string script = "window.domAutomationController.send(document.body.style." "backgroundColor);"; std::string result; EXPECT_TRUE(content::ExecuteScriptAndExtractString(tab, script, &result)); EXPECT_EQ(result, "red"); } // Test that a browser action popup with a web iframe works correctly. This // primarily targets --isolate-extensions and --site-per-process modes, where // the iframe runs in a separate process. See https://crbug.com/546267. IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, BrowserActionPopupWithIframe) { host_resolver()->AddRule("*", "127.0.0.1"); ASSERT_TRUE(embedded_test_server()->Start()); ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("browser_action/popup_with_iframe"))); BrowserActionTestUtil* actions_bar = GetBrowserActionsBar(); const Extension* extension = GetSingleLoadedExtension(); ASSERT_TRUE(extension) << message_; // Simulate a click on the browser action to open the popup. ASSERT_TRUE(OpenPopup(0)); // Find the RenderFrameHost associated with the iframe in the popup. content::RenderFrameHost* frame_host = nullptr; extensions::ProcessManager* manager = extensions::ProcessManager::Get(browser()->profile()); std::set frame_hosts = manager->GetRenderFrameHostsForExtension(extension->id()); for (auto host : frame_hosts) { if (host->GetFrameName() == "child_frame") { frame_host = host; break; } } ASSERT_TRUE(frame_host); EXPECT_EQ(extension->GetResourceURL("frame.html"), frame_host->GetLastCommittedURL()); EXPECT_TRUE(frame_host->GetParent()); // Navigate the popup's iframe to a (cross-site) web page, and wait for that // page to send a message, which will ensure that the page has loaded. GURL foo_url(embedded_test_server()->GetURL("foo.com", "/popup_iframe.html")); std::string script = "location.href = '" + foo_url.spec() + "'"; std::string result; EXPECT_TRUE( content::ExecuteScriptAndExtractString(frame_host, script, &result)); EXPECT_EQ("DONE", result); EXPECT_TRUE(actions_bar->HidePopup()); } IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, BrowserActionWithRectangularIcon) { ExtensionTestMessageListener ready_listener("ready", true); ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("browser_action").AppendASCII("rect_icon"))); EXPECT_TRUE(ready_listener.WaitUntilSatisfied()); gfx::Image first_icon = GetBrowserActionsBar()->GetIcon(0); ResultCatcher catcher; ready_listener.Reply(std::string()); EXPECT_TRUE(catcher.GetNextResult()); gfx::Image next_icon = GetBrowserActionsBar()->GetIcon(0); EXPECT_FALSE(gfx::test::AreImagesEqual(first_icon, next_icon)); } // Test that we don't try and show a browser action popup with // browserAction.openPopup if there is no toolbar (e.g., for web popup windows). // Regression test for crbug.com/584747. IN_PROC_BROWSER_TEST_F(BrowserActionApiTest, BrowserActionOpenPopupOnPopup) { // Open a new web popup window. chrome::NavigateParams params(browser(), GURL("http://www.google.com/"), ui::PAGE_TRANSITION_LINK); params.disposition = NEW_POPUP; params.window_action = chrome::NavigateParams::SHOW_WINDOW; ui_test_utils::NavigateToURL(¶ms); Browser* popup_browser = params.browser; // Verify it is a popup, and it is the active window. ASSERT_TRUE(popup_browser); // The window isn't considered "active" on MacOSX for odd reasons. The more // important test is that it *is* considered the last active browser, since // that's what we check when we try to open the popup. #if !defined(OS_MACOSX) EXPECT_TRUE(popup_browser->window()->IsActive()); #endif EXPECT_FALSE(browser()->window()->IsActive()); EXPECT_FALSE(popup_browser->SupportsWindowFeature(Browser::FEATURE_TOOLBAR)); EXPECT_EQ(popup_browser, chrome::FindLastActiveWithProfile(browser()->profile())); // Load up the extension, which will call chrome.browserAction.openPopup() // when it is loaded and verify that the popup didn't open. ExtensionTestMessageListener listener("ready", true); EXPECT_TRUE(LoadExtension( test_data_dir_.AppendASCII("browser_action/open_popup_on_reply"))); EXPECT_TRUE(listener.WaitUntilSatisfied()); ResultCatcher catcher; listener.Reply(std::string()); EXPECT_TRUE(catcher.GetNextResult()) << message_; } } // namespace } // namespace extensions