summaryrefslogtreecommitdiffstats
path: root/chrome/browser/extensions/api
diff options
context:
space:
mode:
Diffstat (limited to 'chrome/browser/extensions/api')
-rw-r--r--chrome/browser/extensions/api/extension_action/browser_action_apitest.cc468
-rw-r--r--chrome/browser/extensions/api/extension_action/extension_actions_api.cc28
-rw-r--r--chrome/browser/extensions/api/extension_action/extension_actions_api.h4
-rw-r--r--chrome/browser/extensions/api/extension_action/extension_browser_actions_api.cc57
-rw-r--r--chrome/browser/extensions/api/extension_action/extension_browser_actions_api.h30
-rw-r--r--chrome/browser/extensions/api/extension_action/extension_page_actions_api.cc20
-rw-r--r--chrome/browser/extensions/api/extension_action/extension_page_actions_api.h28
-rw-r--r--chrome/browser/extensions/api/extension_action/page_action_apitest.cc207
8 files changed, 829 insertions, 13 deletions
diff --git a/chrome/browser/extensions/api/extension_action/browser_action_apitest.cc b/chrome/browser/extensions/api/extension_action/browser_action_apitest.cc
new file mode 100644
index 0000000..8236863f
--- /dev/null
+++ b/chrome/browser/extensions/api/extension_action/browser_action_apitest.cc
@@ -0,0 +1,468 @@
+// 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 "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_tab_util.h"
+#include "chrome/browser/profiles/profile.h"
+#include "chrome/browser/ui/browser.h"
+#include "chrome/browser/ui/browser_window.h"
+#include "chrome/common/chrome_notification_types.h"
+#include "chrome/common/extensions/extension_action.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/web_contents.h"
+#include "ui/gfx/rect.h"
+#include "ui/gfx/size.h"
+
+using content::WebContents;
+
+class BrowserActionApiTest : public ExtensionApiTest {
+ public:
+ BrowserActionApiTest() {}
+ virtual ~BrowserActionApiTest() {}
+
+ protected:
+ BrowserActionTestUtil GetBrowserActionsBar() {
+ return BrowserActionTestUtil(browser());
+ }
+
+ bool OpenPopup(int index) {
+ ResultCatcher catcher;
+ ui_test_utils::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();
+ }
+};
+
+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.
+ WebContents* tab = browser()->GetSelectedWebContents();
+ bool result = false;
+ ASSERT_TRUE(ui_test_utils::ExecuteJavaScriptAndExtractBool(
+ tab->GetRenderViewHost(), 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.
+ ASSERT_EQ(1, GetBrowserActionsBar().NumberOfBrowserActions());
+ EXPECT_TRUE(GetBrowserActionsBar().HasIcon(0));
+
+ // Set prev_id which holds the id of the previous image, and use it in the
+ // next test to see if the image changes.
+ uint32_t prev_id = extension->browser_action()->GetIcon(0).getGenerationID();
+
+ // 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));
+ EXPECT_NE(prev_id, extension->browser_action()->GetIcon(0).getGenerationID());
+ prev_id = extension->browser_action()->GetIcon(0).getGenerationID();
+
+ // Tell the extension to update the icon using setIcon({path:...}).
+ ui_test_utils::NavigateToURL(browser(),
+ GURL(extension->GetResourceURL("update2.html")));
+ ASSERT_TRUE(catcher.GetNextResult());
+ EXPECT_TRUE(GetBrowserActionsBar().HasIcon(0));
+ EXPECT_NE(prev_id, extension->browser_action()->GetIcon(0).getGenerationID());
+}
+
+// This test is flaky as per http://crbug.com/74557.
+IN_PROC_BROWSER_TEST_F(BrowserActionApiTest,
+ DISABLED_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
+// 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.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()->GetSelectedWebContents());
+
+ 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()->GetSelectedWebContents());
+
+ 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();
+
+ const Extension* extension_a = LoadExtension(test_data_dir_.AppendASCII(
+ "browser_action/basics"));
+ const Extension* extension_b = LoadExtension(test_data_dir_.AppendASCII(
+ "browser_action/popup"));
+ const Extension* extension_c = LoadExtension(test_data_dir_.AppendASCII(
+ "browser_action/add_popup"));
+ ASSERT_TRUE(extension_a);
+ ASSERT_TRUE(extension_b);
+ ASSERT_TRUE(extension_c);
+
+ // 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(extension_a->id(), true);
+ service->extension_prefs()->SetIsIncognitoEnabled(extension_c->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("about:blank"));
+
+ 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(extension_c, 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(extension_b, 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));
+}
+
+// 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.
+ ExtensionProcessManager* manager =
+ browser()->profile()->GetExtensionProcessManager();
+ ASSERT_TRUE(manager->GetBackgroundHostForExtension(extension->id()));
+ ExtensionAction* action = extension->browser_action();
+ ASSERT_EQ("", action->GetBadgeText(ExtensionAction::kDefaultTabId));
+
+ ui_test_utils::WindowedNotificationObserver host_destroyed_observer(
+ chrome::NOTIFICATION_EXTENSION_HOST_DESTROYED,
+ content::NotificationService::AllSources());
+
+ // Click the browser action.
+ browser()->profile()->GetExtensionService()->browser_event_router()->
+ BrowserActionExecuted(
+ browser()->profile(), action->extension_id(), browser());
+
+ // 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(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 = extension->browser_action();
+ 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.
+ action = extension->browser_action();
+ 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.
+ action = extension->browser_action();
+ ASSERT_EQ(SkColorSetARGB(255, 255, 255, 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());
+}
diff --git a/chrome/browser/extensions/api/extension_action/extension_actions_api.cc b/chrome/browser/extensions/api/extension_action/extension_actions_api.cc
index 53633a1..aa18536 100644
--- a/chrome/browser/extensions/api/extension_action/extension_actions_api.cc
+++ b/chrome/browser/extensions/api/extension_action/extension_actions_api.cc
@@ -134,3 +134,31 @@ bool ExtensionActionFunction::SetBadgeBackgroundColor() {
return true;
}
+
+bool ExtensionActionFunction::GetTitle() {
+ result_.reset(Value::CreateStringValue(extension_action_->GetTitle(tab_id_)));
+ return true;
+}
+
+bool ExtensionActionFunction::GetPopup() {
+ result_.reset(Value::CreateStringValue(
+ extension_action_->GetPopupUrl(tab_id_).spec()));
+ return true;
+}
+
+bool ExtensionActionFunction::GetBadgeText() {
+ result_.reset(Value::CreateStringValue(
+ extension_action_->GetBadgeText(tab_id_)));
+ return true;
+}
+
+bool ExtensionActionFunction::GetBadgeBackgroundColor() {
+ ListValue* list = new ListValue();
+ SkColor color = extension_action_->GetBadgeBackgroundColor(tab_id_);
+ list->Append(Value::CreateIntegerValue(SkColorGetR(color)));
+ list->Append(Value::CreateIntegerValue(SkColorGetG(color)));
+ list->Append(Value::CreateIntegerValue(SkColorGetB(color)));
+ list->Append(Value::CreateIntegerValue(SkColorGetA(color)));
+ result_.reset(list);
+ return true;
+}
diff --git a/chrome/browser/extensions/api/extension_action/extension_actions_api.h b/chrome/browser/extensions/api/extension_action/extension_actions_api.h
index 6fd3457..dc3d53b 100644
--- a/chrome/browser/extensions/api/extension_action/extension_actions_api.h
+++ b/chrome/browser/extensions/api/extension_action/extension_actions_api.h
@@ -29,6 +29,10 @@ class ExtensionActionFunction : public SyncExtensionFunction {
bool SetPopup();
bool SetBadgeBackgroundColor();
bool SetBadgeText();
+ bool GetTitle();
+ bool GetPopup();
+ bool GetBadgeBackgroundColor();
+ bool GetBadgeText();
// All the browser action APIs take a single argument called details that is
// a dictionary.
diff --git a/chrome/browser/extensions/api/extension_action/extension_browser_actions_api.cc b/chrome/browser/extensions/api/extension_action/extension_browser_actions_api.cc
index a60c150..49e8ee9 100644
--- a/chrome/browser/extensions/api/extension_action/extension_browser_actions_api.cc
+++ b/chrome/browser/extensions/api/extension_action/extension_browser_actions_api.cc
@@ -18,6 +18,13 @@ const char kNoBrowserActionError[] =
"This extension has no browser action specified.";
}
+void BrowserActionFunction::FireUpdateNotification() {
+ content::NotificationService::current()->Notify(
+ chrome::NOTIFICATION_EXTENSION_BROWSER_ACTION_UPDATED,
+ content::Source<ExtensionAction>(extension_action_),
+ content::NotificationService::NoDetails());
+}
+
bool BrowserActionFunction::RunImpl() {
ExtensionActionFunction::RunImpl();
extension_action_ = GetExtension()->browser_action();
@@ -26,32 +33,56 @@ bool BrowserActionFunction::RunImpl() {
return false;
}
- if (!RunExtensionAction())
- return false;
-
- content::NotificationService::current()->Notify(
- chrome::NOTIFICATION_EXTENSION_BROWSER_ACTION_UPDATED,
- content::Source<ExtensionAction>(extension_action_),
- content::NotificationService::NoDetails());
- return true;
+ return RunExtensionAction();
}
bool BrowserActionSetIconFunction::RunExtensionAction() {
- return SetIcon();
+ if (!SetIcon())
+ return false;
+ FireUpdateNotification();
+ return true;
}
bool BrowserActionSetTitleFunction::RunExtensionAction() {
- return SetTitle();
+ if (!SetTitle())
+ return false;
+ FireUpdateNotification();
+ return true;
}
bool BrowserActionSetPopupFunction::RunExtensionAction() {
- return SetPopup();
+ if (!SetPopup())
+ return false;
+ FireUpdateNotification();
+ return true;
}
bool BrowserActionSetBadgeTextFunction::RunExtensionAction() {
- return SetBadgeText();
+ if (!SetBadgeText())
+ return false;
+ FireUpdateNotification();
+ return true;
}
bool BrowserActionSetBadgeBackgroundColorFunction::RunExtensionAction() {
- return SetBadgeBackgroundColor();
+ if (!SetBadgeBackgroundColor())
+ return false;
+ FireUpdateNotification();
+ return true;
+}
+
+bool BrowserActionGetTitleFunction::RunExtensionAction() {
+ return GetTitle();
+}
+
+bool BrowserActionGetPopupFunction::RunExtensionAction() {
+ return GetPopup();
+}
+
+bool BrowserActionGetBadgeTextFunction::RunExtensionAction() {
+ return GetBadgeText();
+}
+
+bool BrowserActionGetBadgeBackgroundColorFunction::RunExtensionAction() {
+ return GetBadgeBackgroundColor();
}
diff --git a/chrome/browser/extensions/api/extension_action/extension_browser_actions_api.h b/chrome/browser/extensions/api/extension_action/extension_browser_actions_api.h
index 5a453fe..1ead691 100644
--- a/chrome/browser/extensions/api/extension_action/extension_browser_actions_api.h
+++ b/chrome/browser/extensions/api/extension_action/extension_browser_actions_api.h
@@ -15,6 +15,7 @@ class BrowserActionFunction : public ExtensionActionFunction {
protected:
virtual ~BrowserActionFunction() {}
virtual bool RunImpl() OVERRIDE;
+ void FireUpdateNotification();
};
// Implement chrome.browserAction.setIcon().
@@ -53,4 +54,33 @@ class BrowserActionSetBadgeBackgroundColorFunction
DECLARE_EXTENSION_FUNCTION_NAME("browserAction.setBadgeBackgroundColor")
};
+// Implement chrome.browserAction.getTitle().
+class BrowserActionGetTitleFunction : public BrowserActionFunction {
+ virtual ~BrowserActionGetTitleFunction() {}
+ virtual bool RunExtensionAction() OVERRIDE;
+ DECLARE_EXTENSION_FUNCTION_NAME("browserAction.getTitle")
+};
+
+// Implement chrome.browserAction.getPopup().
+class BrowserActionGetPopupFunction : public BrowserActionFunction {
+ virtual ~BrowserActionGetPopupFunction() {}
+ virtual bool RunExtensionAction() OVERRIDE;
+ DECLARE_EXTENSION_FUNCTION_NAME("browserAction.getPopup")
+};
+
+// Implement chrome.browserAction.getBadgeText().
+class BrowserActionGetBadgeTextFunction : public BrowserActionFunction {
+ virtual ~BrowserActionGetBadgeTextFunction() {}
+ virtual bool RunExtensionAction() OVERRIDE;
+ DECLARE_EXTENSION_FUNCTION_NAME("browserAction.getBadgeText")
+};
+
+// Implement chrome.browserAction.getBadgeBackgroundColor().
+class BrowserActionGetBadgeBackgroundColorFunction
+ : public BrowserActionFunction {
+ virtual ~BrowserActionGetBadgeBackgroundColorFunction() {}
+ virtual bool RunExtensionAction() OVERRIDE;
+ DECLARE_EXTENSION_FUNCTION_NAME("browserAction.getBadgeBackgroundColor")
+};
+
#endif // CHROME_BROWSER_EXTENSIONS_API_EXTENSION_ACTION_EXTENSION_BROWSER_ACTIONS_API_H_
diff --git a/chrome/browser/extensions/api/extension_action/extension_page_actions_api.cc b/chrome/browser/extensions/api/extension_action/extension_page_actions_api.cc
index 3b00362..81a9c2d 100644
--- a/chrome/browser/extensions/api/extension_action/extension_page_actions_api.cc
+++ b/chrome/browser/extensions/api/extension_action/extension_page_actions_api.cc
@@ -222,3 +222,23 @@ bool PageActionSetBadgeTextFunction::RunExtensionAction() {
contents_->extension_tab_helper()->PageActionStateChanged();
return true;
}
+
+bool PageActionGetTitleFunction::RunExtensionAction() {
+ return GetTitle();
+}
+
+bool PageActionGetPopupFunction::RunExtensionAction() {
+ return GetPopup();
+}
+
+// Not currently exposed to extensions. To re-enable, add mapping in
+// extension_function_dispatcher.
+bool PageActionGetBadgeBackgroundColorFunction::RunExtensionAction() {
+ return GetBadgeBackgroundColor();
+}
+
+// Not currently exposed to extensions. To re-enable, add mapping in
+// extension_function_dispatcher.
+bool PageActionGetBadgeTextFunction::RunExtensionAction() {
+ return GetBadgeText();
+}
diff --git a/chrome/browser/extensions/api/extension_action/extension_page_actions_api.h b/chrome/browser/extensions/api/extension_action/extension_page_actions_api.h
index 6a8e68b59..931d807 100644
--- a/chrome/browser/extensions/api/extension_action/extension_page_actions_api.h
+++ b/chrome/browser/extensions/api/extension_action/extension_page_actions_api.h
@@ -92,4 +92,32 @@ class PageActionSetBadgeTextFunction : public PageActionFunction {
DECLARE_EXTENSION_FUNCTION_NAME("pageAction.setBadgeText")
};
+// Implement chrome.pageAction.getTitle().
+class PageActionGetTitleFunction : public PageActionFunction {
+ virtual ~PageActionGetTitleFunction() {}
+ virtual bool RunExtensionAction() OVERRIDE;
+ DECLARE_EXTENSION_FUNCTION_NAME("pageAction.getTitle")
+};
+
+// Implement chrome.pageAction.getPopup().
+class PageActionGetPopupFunction : public PageActionFunction {
+ virtual ~PageActionGetPopupFunction() {}
+ virtual bool RunExtensionAction() OVERRIDE;
+ DECLARE_EXTENSION_FUNCTION_NAME("pageAction.getPopup")
+};
+
+// Implement chrome.pageAction.getBadgeText().
+class PageActionGetBadgeTextFunction : public PageActionFunction {
+ virtual ~PageActionGetBadgeTextFunction() {}
+ virtual bool RunExtensionAction() OVERRIDE;
+ DECLARE_EXTENSION_FUNCTION_NAME("pageAction.getBadgeText")
+};
+
+// Implement chrome.pageAction.getBadgeBackgroundColor().
+class PageActionGetBadgeBackgroundColorFunction : public PageActionFunction {
+ virtual ~PageActionGetBadgeBackgroundColorFunction() {}
+ virtual bool RunExtensionAction() OVERRIDE;
+ DECLARE_EXTENSION_FUNCTION_NAME("pageAction.getBadgeBackgroundColor")
+};
+
#endif // CHROME_BROWSER_EXTENSIONS_API_EXTENSION_ACTION_EXTENSION_PAGE_ACTIONS_API_H_
diff --git a/chrome/browser/extensions/api/extension_action/page_action_apitest.cc b/chrome/browser/extensions/api/extension_action/page_action_apitest.cc
new file mode 100644
index 0000000..7591077
--- /dev/null
+++ b/chrome/browser/extensions/api/extension_action/page_action_apitest.cc
@@ -0,0 +1,207 @@
+// 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/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_tab_util.h"
+#include "chrome/browser/profiles/profile.h"
+#include "chrome/browser/sessions/restore_tab_helper.h"
+#include "chrome/browser/ui/browser.h"
+#include "chrome/browser/ui/browser_window.h"
+#include "chrome/browser/ui/omnibox/location_bar.h"
+#include "chrome/browser/ui/tab_contents/tab_contents_wrapper.h"
+#include "chrome/common/extensions/extension.h"
+#include "chrome/common/extensions/extension_action.h"
+#include "chrome/test/base/ui_test_utils.h"
+#include "content/public/browser/web_contents.h"
+
+IN_PROC_BROWSER_TEST_F(ExtensionApiTest, PageAction) {
+ ASSERT_TRUE(test_server()->Start());
+ ASSERT_TRUE(RunExtensionTest("page_action/basics")) << message_;
+ const Extension* extension = GetSingleLoadedExtension();
+ ASSERT_TRUE(extension) << message_;
+ {
+ // Tell the extension to update the page action state.
+ ResultCatcher catcher;
+ ui_test_utils::NavigateToURL(browser(),
+ GURL(extension->GetResourceURL("update.html")));
+ ASSERT_TRUE(catcher.GetNextResult());
+ }
+
+ // Test that we received the changes.
+ int tab_id = browser()->GetSelectedTabContentsWrapper()->
+ restore_tab_helper()->session_id().id();
+ ExtensionAction* action = extension->page_action();
+ ASSERT_TRUE(action);
+ EXPECT_EQ("Modified", action->GetTitle(tab_id));
+
+ {
+ // Simulate the page action being clicked.
+ ResultCatcher catcher;
+ int tab_id =
+ ExtensionTabUtil::GetTabId(browser()->GetSelectedWebContents());
+ ExtensionService* service = browser()->profile()->GetExtensionService();
+ service->browser_event_router()->PageActionExecuted(
+ browser()->profile(), extension->id(), "", tab_id, "", 0);
+ EXPECT_TRUE(catcher.GetNextResult());
+ }
+
+ {
+ // Tell the extension to update the page action state again.
+ ResultCatcher catcher;
+ ui_test_utils::NavigateToURL(browser(),
+ GURL(extension->GetResourceURL("update2.html")));
+ ASSERT_TRUE(catcher.GetNextResult());
+ }
+
+ // Test that we received the changes.
+ tab_id = browser()->GetSelectedTabContentsWrapper()->restore_tab_helper()->
+ session_id().id();
+ EXPECT_FALSE(action->GetIcon(tab_id).isNull());
+}
+
+// Test that calling chrome.pageAction.setPopup() can enable a popup.
+IN_PROC_BROWSER_TEST_F(ExtensionApiTest, PageActionAddPopup) {
+ // Load the extension, which has no default popup.
+ ASSERT_TRUE(RunExtensionTest("page_action/add_popup")) << message_;
+ const Extension* extension = GetSingleLoadedExtension();
+ ASSERT_TRUE(extension) << message_;
+
+ int tab_id = ExtensionTabUtil::GetTabId(browser()->GetSelectedWebContents());
+
+ ExtensionAction* page_action = extension->page_action();
+ ASSERT_TRUE(page_action)
+ << "Page action test extension should have a page action.";
+
+ ASSERT_FALSE(page_action->HasPopup(tab_id));
+
+ // Simulate the page action being clicked. The resulting event should
+ // install a page action popup.
+ {
+ ResultCatcher catcher;
+ ExtensionService* service = browser()->profile()->GetExtensionService();
+ service->browser_event_router()->PageActionExecuted(
+ browser()->profile(), extension->id(), "action", tab_id, "", 1);
+ ASSERT_TRUE(catcher.GetNextResult());
+ }
+
+ ASSERT_TRUE(page_action->HasPopup(tab_id))
+ << "Clicking on the page action should have caused a popup to be added.";
+
+ ASSERT_STREQ("/a_popup.html",
+ page_action->GetPopupUrl(tab_id).path().c_str());
+
+ // Now change the popup from a_popup.html to a_second_popup.html .
+ // Load a page which removes the popup using chrome.pageAction.setPopup().
+ {
+ ResultCatcher catcher;
+ ui_test_utils::NavigateToURL(
+ browser(),
+ GURL(extension->GetResourceURL("change_popup.html")));
+ ASSERT_TRUE(catcher.GetNextResult());
+ }
+
+ ASSERT_TRUE(page_action->HasPopup(tab_id));
+ ASSERT_STREQ("/another_popup.html",
+ page_action->GetPopupUrl(tab_id).path().c_str());
+}
+
+// Test that calling chrome.pageAction.setPopup() can remove a popup.
+IN_PROC_BROWSER_TEST_F(ExtensionApiTest, PageActionRemovePopup) {
+ // Load the extension, which has a page action with a default popup.
+ ASSERT_TRUE(RunExtensionTest("page_action/remove_popup")) << message_;
+ const Extension* extension = GetSingleLoadedExtension();
+ ASSERT_TRUE(extension) << message_;
+
+ int tab_id = ExtensionTabUtil::GetTabId(browser()->GetSelectedWebContents());
+
+ ExtensionAction* page_action = extension->page_action();
+ ASSERT_TRUE(page_action)
+ << "Page action test extension should have a page action.";
+
+ ASSERT_TRUE(page_action->HasPopup(tab_id))
+ << "Expect a page action popup before the test removes it.";
+
+ // Load a page which removes the popup using chrome.pageAction.setPopup().
+ {
+ ResultCatcher catcher;
+ ui_test_utils::NavigateToURL(
+ browser(),
+ GURL(extension->GetResourceURL("remove_popup.html")));
+ ASSERT_TRUE(catcher.GetNextResult());
+ }
+
+ ASSERT_FALSE(page_action->HasPopup(tab_id))
+ << "Page action popup should have been removed.";
+}
+
+// Tests old-style pageActions API that is deprecated but we don't want to
+// break.
+IN_PROC_BROWSER_TEST_F(ExtensionApiTest, OldPageActions) {
+ ASSERT_TRUE(RunExtensionTest("page_action/old_api")) << message_;
+ const Extension* extension = GetSingleLoadedExtension();
+ ASSERT_TRUE(extension) << message_;
+
+ // Have the extension enable the page action.
+ {
+ ResultCatcher catcher;
+ ui_test_utils::NavigateToURL(browser(),
+ GURL(extension->GetResourceURL("page.html")));
+ ASSERT_TRUE(catcher.GetNextResult());
+ }
+
+ // Simulate the page action being clicked.
+ {
+ ResultCatcher catcher;
+ int tab_id =
+ ExtensionTabUtil::GetTabId(browser()->GetSelectedWebContents());
+ ExtensionService* service = browser()->profile()->GetExtensionService();
+ service->browser_event_router()->PageActionExecuted(
+ browser()->profile(), extension->id(), "action", tab_id, "", 1);
+ EXPECT_TRUE(catcher.GetNextResult());
+ }
+}
+
+// Tests popups in page actions.
+// Flaky on the trybots. See http://crbug.com/96725.
+IN_PROC_BROWSER_TEST_F(ExtensionApiTest, DISABLED_ShowPageActionPopup) {
+ ASSERT_TRUE(RunExtensionTest("page_action/popup")) << message_;
+ const Extension* extension = GetSingleLoadedExtension();
+ ASSERT_TRUE(extension) << message_;
+
+ ASSERT_TRUE(WaitForPageActionVisibilityChangeTo(1));
+
+ {
+ ResultCatcher catcher;
+ LocationBarTesting* location_bar =
+ browser()->window()->GetLocationBar()->GetLocationBarForTesting();
+ location_bar->TestPageActionPressed(0);
+ ASSERT_TRUE(catcher.GetNextResult());
+ }
+}
+
+// Test http://crbug.com/57333: that two page action extensions using the same
+// icon for the page action icon and the extension icon do not crash.
+IN_PROC_BROWSER_TEST_F(ExtensionApiTest, TestCrash57333) {
+ // Load extension A.
+ ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII("page_action")
+ .AppendASCII("crash_57333")
+ .AppendASCII("Extension1")));
+ // Load extension B.
+ ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII("page_action")
+ .AppendASCII("crash_57333")
+ .AppendASCII("Extension2")));
+}
+
+IN_PROC_BROWSER_TEST_F(ExtensionApiTest, Getters) {
+ ASSERT_TRUE(RunExtensionTest("page_action/getters")) << message_;
+ const Extension* extension = GetSingleLoadedExtension();
+ ASSERT_TRUE(extension) << message_;
+
+ ResultCatcher catcher;
+ ui_test_utils::NavigateToURL(browser(),
+ GURL(extension->GetResourceURL("update.html")));
+ ASSERT_TRUE(catcher.GetNextResult());
+}