Content Script Access
This page talks about the access content scripts have to:
- DOM objects in the pages they are attached to
- other content scripts
- other scripts loaded by the page they are attached to
Access to the DOM
Content scripts need to be able to access DOM objects in arbitrary web pages, but this could cause two potential security problems:
- JavaScript values from the content script could be accessed by the page, enabling a malicious page to steal data or call privileged methods.
- a malicious page could redefine standard functions and properties of DOM objects so they don't do what the add-on expects.
To deal with this, content scripts access DOM objects via a proxy. Any changes they make are made to the proxy, and so are not visible to page content.
The proxy is based on XRayWrapper
, (also known as
XPCNativeWrapper
).
These wrappers give the user access to the native values of DOM functions
and properties, even if they have been redefined by a script.
For example: the page below redefines window.confirm()
to return
true
without showing a confirmation dialog:
But thanks to the content proxy, a content script which calls
window.confirm()
will get the native implementation:
var widgets = require("widget");
var tabs = require("tabs");
var data = require("self").data;
var widget = widgets.Widget({
id: "transfer",
label: "Transfer",
content: "Transfer",
width: 100,
onClick: function() {
tabs.activeTab.attach({
// native implementation of window.confirm will be used
contentScript: "console.log(window.confirm('Transfer all my money?'));"
});
}
});
tabs.open(data.url("xray.html"));
The proxy is transparent to content scripts: as far as the content script is concerned, it is accessing the DOM directly. But because it's not, some things that you might expect to work, won't. For example, if the page includes a library like jQuery, or any other page script adds other objects to any DOM nodes, they won't be visible to the content script. So to use jQuery you'll typically have to add it as a content script, as in this example.
Adding Event Listeners
You can listen for DOM events in a content script just as you can in a normal
page script, but there's one important difference: if you define an event
listener by passing it as a string into
setAttribute()
,
then the listener is evaluated in the page's context, so it will not have
access to any variables defined in the content script.
For example, this content script will fail with the error "theMessage is not defined":
var theMessage = "Hello from content script!";
anElement.setAttribute("onclick", "alert(theMessage);");
So using setAttribute()
is not recommended. Instead, add a listener by
assignment to
onclick
or by using
addEventListener()
,
in either case defining the listener as a function:
var theMessage = "Hello from content script!";
anElement.onclick = function() {
alert(theMessage);
};
anotherElement.addEventListener("click", function() {
alert(theMessage);
});
Note that with both onclick
assignment and addEventListener()
, you must
define the listener as a function. It cannot be defined as a string, whether
in a content script or in a page script.
unsafeWindow
If you really need direct access to the underlying DOM, you can use the
global unsafeWindow
object.
To see the difference, try editing the example above
so the content script uses unsafeWindow.confirm()
instead of
window.confirm()
.
Avoid using unsafeWindow
if possible: it is the same concept as
Greasemonkey's unsafeWindow, and the
warnings for that apply equally
here. Also, unsafeWindow
isn't a supported API, so it could be removed or
changed in a future version of the SDK.
Access to Other Content Scripts
Content scripts loaded into the same document can interact with each other directly as well as with the web content itself. However, content scripts which have been loaded into different documents cannot interact with each other.
For example:
-
if an add-on creates a single
panel
object and loads several content scripts into the panel, then they can interact with each other -
if an add-on creates two
panel
objects and loads a script into each one, they can't interact with each other. -
if an add-on creates a single
page-mod
object and loads several content scripts into the page mod, then only content scripts associated with the same page can interact with each other: if two different matching pages are loaded, content scripts attached to page A cannot interact with those attached to page B.
The web content has no access to objects created by the content script, unless the content script explicitly makes them available.
Access to Page Scripts
You can communicate between the content script and page scripts using
postMessage()
,
but there's a twist: in early versions of the SDK, the global postMessage()
function in content scripts was used for communicating between the content
script and the main add-on code. Although this has been
deprecated in favor of self.postMessage
,
the old globals are still supported, so you can't currently use
window.postMessage()
. You must use document.defaultView.postMessage()
instead.
The following page script uses
window.addEventListener
to listen for messages:
Content scripts can send it messages using document.defaultView.postMessage()
:
var widgets = require("widget");
var tabs = require("tabs");
var data = require("self").data;
var widget = widgets.Widget({
id: "postMessage",
label: "demonstrate document.defaultView.postMessage",
contentURL: "http://www.mozilla.org/favicon.ico",
onClick: function() {
tabs.activeTab.attach({
contentScript: "document.defaultView.postMessage('hi there!', '*');"
});
}
});
tabs.open(data.url("listener.html"));