I am running into an issue sending data from my background script to the script for my pageAction
. My content script adds an <iframe />
and the JavaScript in the <iframe />
is receiving the data from my background script, but it does not seem to be retrieved in my pageAction
.
In my background script I have something like:
chrome.tabs.sendMessage(senderTab.tab.id,
{
foo:bar
});
where senderTab.tab.id
is the "sender" in onMessage
Listener in my background script.
In the JavaScript loaded by the <iframe />
injected by my content script I have something like:
chrome.runtime.onMessage.addListener(
function(request, sender, sendResponse) {
console.log("received in iframe:", request);
}
});
The <iframe />
receives the message exactly as expected.
I put the same JavaScript in my page_action.js
, but it does not receive any data from the background script. The pageAction is activated with chrome.pageAction.show(senderTab.tab.id);
before I call chrome.tabs.sendMessage(senderTab.tab.id ...
Is the HTML page attached to my pageAction not part of the same tab? Since this tabId
enabled me to activate/"show" the icon, I would think the listener in the JavaScript for the pageAction should also receive from chrome.tabs.sendMessage(senderTab.tab.id ...
In my content script I use the following to send data to the background script:
chrome.runtime.sendMessage({
foo: bar
});
When the content script sends the above message, the pageAction JavaScript is picking it up.
How do I get the background script to properly send data to my pageAction? I do not want to have pageAction request/poll, instead I want pageAction to just listen and receive. E.g., if the pageAction HTML it shown, it should be able to update in real time as the background page makes changes.
Communicating with a page in the background context
Pages which are open in the background context include:
background
pages remain loaded at all times.)Using
tabs.sendMessage()
(MDN) will not send a message to any of them. You would need to useruntime.sendMessage()
(MDN) to send a message to them. The scope for any of them, except background pages and event pages, only exists when it is being displayed. Obviously, you can not communicate with the code when it does not exist. When the scope exists, you can communicate with any of them using:Directly
From the background context, you can directly change variables, or call functions, in another page that is also in the background context (i.e. not content scripts), after having gotten a reference to its global scope, its Window, using
extension.getViews()
(MDN),extension.getBackgroundPage()
(MDN), or other method(MDN).For example, you can call a function created with
function myFunction
in the page of the first returned view by using something like:It should be noted that in your callback from
tabs.create()
(MDN) orwindows.create()
(MDN) the view for the newly opened tab or window will probably not yet exist. You will need to use some methodology to wait for the view to exist.2 See below for recommended ways to communicate with newly opened tabs or windows.Directly manipulating values in the other page's scope allows you to communicate any type of data you desire.
Messaging
Receive messages using
chrome.runtime.onMessage
(MDN), 3 which were sent withchrome.runtime.sendMessage()
(MDN). Each time you receive a message in aruntime.onMessage
listener, there will be asendResponse
function provided as the third argument which allows you to directly respond to the message. If the original sender has not supplied a callback to receive such a response in their call tochrome.runtime.sendMessage()
, then the response is lost. If using Promises (e.g.browser.runtime.sendMessage()
in Firefox), the response is passed as an argument when the Promise is fulfilled. If you want to send the response asynchronously, you will need toreturn true;
from yourruntime.onMessage
listener.Ports
You can also connect ports, using
chrome.runtime.connect()
(MDN) andchrome.runtime.onConnect
(MDN) for longer term messaging.Use
chrome.tabs.sendMessage()
to send to content scriptsIf you want to send from the background context (e.g. background script or popup) to a content script you would use
chrome.tabs.sendMessage()
/chrome.runtime.onMessage
, or connect port(s) usingchrome.tabs.connect()
(MDN)/chrome.runtime.onConnect
.JSON-serializable data only
Using messaging, you can only pass data which is JSON-serializable.
Messages are received by all scripts in the background, except the sender
Messages sent to the background context are received by all scripts in the background context which have registered a listener, except the script which sent it.3 There is no way to specify that it is only to be received by a specific script. Thus, if you have multiple potential recipients, you will need to create a way to be sure that the message received was intended for that script. The ways to do so usually rely on specific properties existing in the message (e.g. use a
destination
orrecipient
property to indicate what script is to receive it, or define that sometype
of messages are always for one recipient or another), or to differentiate based on thesender
(MDN) supplied to the message handler (e.g. if messages from one sender are always only for a specific recipient). There is no set way to do this, you must choose/create a way to do it for use in your extension.For a more detailed discussion of this issue, please see: Messages intended for one script in the background context are received by all
Data in a StorageArea
Store data to a StorageArea(MDN) and be notified of the change in other scripts using
chrome.storage.onChanged
(MDN). Thestorage.onChanged
event can be listened to in both the background context and content scripts.You can only store data which is JSON-serializable into a StorageArea.
Which method is best to use in any particular situation will depends on what you are wanting to communicate (type of data, state change, etc.), and to which portion, or portions, of your extension you are wanting to communicate from and to. For instance, if you want to communicate information which is not JSON-serializable, you would need to do so directly (i.e. not messaging or using a StorageArea). You can use multiple methods in the same extension.
More on popups
None of the popups (e.g. browser action, or page action) are directly associated with the active tab. There is no concept of a shared or separate instance per tab. However, the user can open one popup in each Chrome window. If more than one popup is open (a maximum of one per Chrome window), then each is in a separate instance (separate scope; has its own Window), but are in the same context. When a popup is actually visible, it exists in the background context.
There is only ever one page action or browser action popup open at a time per Chrome window. The HTML file which will be open will be whichever one has been defined for the active tab of the current window and opened by the user by clicking on the page/browser action button. This can be assigned a different HTML document for different tabs by using
chrome.browserAction.setPopup()
(MDN), orchrome.pageAction.setPopup()
(MDN), and specifying atabId
. The popup can/will be destroyed for multiple reasons, but definitely when another tab becomes the active tab in the window in which the popup is open.However, any method of communication used will only communicate to the one(s) which is/are currently open, not ones which are not open. If popups are open for more than one Chrome window at a time, then they are separate instances, with their own scope (i.e. their own Window). You can think of this something like having the same web page open in more than one tab.
If you have a background script, the background script context is persistent across the entire instance of Chrome. If you do not have a background script the context may be created when needed (e.g. a popup is shown) and destroyed when no longer needed.
chrome.tabs.sendMessage()
can not communicate to popupsAs mentioned above, even if the popup did exist, it will exist in the background context. Calling
chrome.tabs.sendMessage()
sends a message to content scripts injected into a tab/frame, not to the background context. Thus, it will not send a message to a non-content script like a popup.Action button: enable/disable (browser action) vs. show/hide (page action)
Calling
chrome.pageAction.show()
(MDN) just causes the page action button to be shown. It does not cause any associated popup to be shown. If the popup/options page/other page is not actually being shown (not just the button), then its scope does not exist. When it does not exist, it, obviously, can not receive any messageInstead of the page action's ability to
show()
(MDN) orhide()
(MDN) the button, browser actions canenable()
(MDN) ordisable()
(MDN) the button.Programmatically opening a tab or window with HTML from your extension
You can use
tabs.create()
(MDN) orwindows.create()
(MDN) to open a tab or window containing an HTML page from within your extension. However, the callback for both of those API calls is executed prior to the page's DOM existing and thus prior to any JavaScript associated with the page existing. Thus, you can not immediately access the DOM created by the contents of that page, nor interact with the JavaScript for the page. Very specifically: noruntime.onMessage()
listeners will have been added, so no messages sent at that time will be received by the newly opening page.The best ways to resolve this issue are:
chrome.extension.getBackgroundPage()
to read the data directly.storage.local
(MDN). The opening page can then read it when its JavaScript is run. For example, you could use a key calledmessageToNewExtensionPage
.runtime.sendMessage()
, then initiate the transfer of the data from your newly opening page by sending a message from the that page's code to the source of the data (usingruntime.sendMessage()
, ortabs.sendMessage()
for content script sources) requesting the data. The script with the data can then send the data back using thesendResponse
(MDN) function provided byruntime.onMessage()
.Additional references
Chrome
Firefox
There are multiple methods which you can use. Which way is best will depend on exactly what you are doing (e.g. when you need to access the view with respect to the code being executed in the view). A simple method would be just to poll waiting for the view to exist. The following code does that for opening a window: