Fork me on GitHub

JSChannel Documentation

Verifying the correctness of the code on this page...

All the code on this page is correct.

Some code on this page doesn't work.



Introduction

JSChannel is a small JavaScript abstraction layer on top of HTML5 cross-document messaging. It builds rich messaging semantics out of window.postMessage().

About The Code Examples

This page uses the doctestjs library by Ian Bicking. The code examples are actually run on your browser to verify their correctness.

All the code examples in this documentation assume communication between a parent page and a child frame. Example code executed in the child frame looks like this:

/* Here's a comment in the child frame. */

The examples also assume that a few global functions exist in the parent page:

Page Setup

JSChannel has no dependencies, so there's very little setup involved. You can get started by making a simple HTML page that includes the JSChannel script:
<script src="jschannel.js"></script>
And a trivial child frame:
<iframe id="childId" src="child.html"></iframe>

The contents of the child frame don't matter, so long as it includes the JSChannel script (so that it can communicate with its parent).

Note that we're just using a child frame for the sake of example; as a rule, JSChannel can be used anywhere that window.postMessage() can.

Creating a Channel

Let's create a channel in our child frame:

var chan = Channel.build({window: window.parent, origin: "*", scope: "testScope"});

And now we'll build one in our parent page to communicate with it:

$ chan = Channel.build({
>   window: document.getElementById("childId").contentWindow,
>   origin: "*",
>   scope: "testScope",
>   onReady: function() {
>     emit("channel is ready!");
>   }
> });
> wait();
out("channel is ready!")

Note that the onReady callback is called once actual communication has been established between the parent page and child frame. If the child frame hadn't set up its end of the channel, for instance, onReady would never get called.

Remote Methods

Now we'll define a simple function in our child frame:

chan.bind("reverse", function(trans, s) { return s.split("").reverse().join(""); });

And call it in our parent frame.

$ chan.call({method: "reverse",
>            params: "hello world!",
>            success: function(v) { emit(v.toString()); }});
> wait();
out("!dlrow olleh")

Remote methods can throw errors, too. Here's one in our child frame:

chan.bind("throwObject", function(trans, s) { throw {error: "object_error_code", message: 'object error message'}; });

Calling that produces this:

$ chan.call({method: "throwObject",
>            error: emit,
>            success: emit});
> wait();
out("object_error_code", "object error message")

Transactions

If your method's implementation is asychronous, you'll have to use the transaction object that's automatically passed as the first argument to a remote method, like so:

chan.bind("twiddleThumbs", function(trans) { setTimeout(function() { trans.complete("thumbs twiddled!"); }, 50); trans.delayReturn(true); });

The caller doesn't do anything differently, since it's always calling the remote method asynchronously:

$ chan.call({method: "twiddleThumbs", success: emit});
> wait();
out("thumbs twiddled!")

You can use a transaction's error() method to propagate errors, too.

chan.bind("getStuff", function(trans) { setTimeout(function() { trans.error("mega_fail", "i could not get your stuff."); }, 50); trans.delayReturn(true); });
$ chan.call({method: "getStuff",
>            error: emit,
>            success: emit});
> wait();
out("mega_fail", "i could not get your stuff.")

Callbacks

Clients can pass functions as parameters to remote methods, and they can be called by the implementer. Here's a trivial example:

chan.bind("simple_cb", function(trans, params) { params.cb("you called?"); trans.complete("complete"); });
$ chan.call({method: "simple_cb", params: { cb: emit }, success: emit});
> wait();
out("you called?")
out("complete")

Timeouts

Clients can also be notified if a remote method invocation takes too long to complete by passing a timeout parameter to a channel's call() method.

For instance, we can try calling a nonexistent method with a timeout:

$ chan.call({method: "nonexistent",
>            timeout: 50,
>            error: emit,
>            success: emit});
> wait();
out("timeout_error", "timeout (50ms) exceeded on method 'testScope::nonexistent'")

Notifications

Notifications are similar to method calls, but are "fire-and-forget", lacking any concept of an error or return value. For example:

chan.bind("ping", function(context, params) { // context.origin contains the origin also for notifications. if (!context.origin) throw "where's your origin?"; chan.notify({method: "pong", params: params}); });
$ chan.bind("pong", function(context, params) { emit(params); });
> chan.notify({method: "ping", params: "hai2u"});
> wait();
out("hai2u")