I'm not sure how to do it, and the documentation doesn't seem to make this quite obvious.
I tried making a background_page
and putting chrome.pageAction.show(tab.id);
inside it, but that doesn't seem to work.
I don't want to use the browser_action
icon because the icon merely reflects the status of the plugin, but it is not a button for performing any actions.
How do I add the page_action's icon inside the address bar for any page/tab at all times?
EDIT: Here's my manifest.json:
{
"name": "My Very First Extension :D",
"version": "0.0.1",
"description": "Awesomeness",
"background_page": "background.html",
"page_action": {
"default_icon": "icon.png"
},
"content_scripts": [{
"matches": ["http://*/*", "https://*/*"],
"js": ["mmm.js"]
}]
}
Where icon.png is a 19x19 pixel PNG graphic. Here's the background.html source, where I'm trying to make the page_action icon appear for all tabs:
<!DOCTYPE html>
<html>
<head>
<script>
chrome.pageAction.show(tab.id);
</script>
</head>
</html>
I see two issues with your code.
You cannot use inline script blocks in your background page -- js code can only be run from a linked script (like so: <script type="text/javascript" src="background.js"></script>
). This is because of the Content Security Policy.
As noted in abraham's answer, since you want the icon to appear for all pages, you should use browser_action, not page_action. According to Google's pageAction documentation page:
Don't use page actions for features that make sense for most pages.
Use browser actions instead.
Icons for page actions appear inside the omnibar (Chrome's address bar), while icons for browser actions appear as a toolbar button next to the omnibar. Either can be used to reflect the status of the plugin and as an action button.
Solutions
To get the icon to appear, you have to call: chrome.pageAction.show(tabId)
. However, you cannot just call this from a content script, because chrome.pageAction.show() must be passed the tabId, and tabId is retrieved from chrome.tabs, and chrome.tabs is not accessible from content scripts.
To add a browser action button and define it's behavior when clicked, define the browser_action element in your manifest.json file ...
"browser_action": {
"default_title": "action description",
"default_icon": "icon.png"
}
... and place this code in your background script:
// Define behavior when browser action icon is clicked
chrome.browserAction.onClicked.addListener(function(tab) {
// executed when the user clicks on the browser action icon
});
Or, if you want to go against convention and use a pageAction, you can use the manifest.json code from your question and place the following into your background script:
// Show page action icon in omnibar.
function showPageAction( tabId, changeInfo, tab ) {
chrome.pageAction.show(tabId);
};
// Call the above function when the url of a tab changes.
chrome.tabs.onUpdated.addListener(showPageAction);
For further research, check out the source code of some of their sample extensions.
If you want a content-script to show a page action icon, he (the content-script) can't do that, but, he can send a message to the background.js to do it (background.js is allowed to show page action icons)
content script (eg. foo.js):
chrome.runtime.sendMessage({action: "showIcon"}, function(response) {});
background.js:
// Received a message from content script
chrome.runtime.onMessage.addListener(function(request, sender, sendResponse) {
if (request.action == "showIcon") {
chrome.pageAction.show(sender.tab.id);
}
});
Use a browser_action
icon. page_action
and browser_action
are the same and provide the same functionality except that page_action
s can be enabled only for specific pages. If you don't want the browser_action
to act like a button than don't have it do anything.