Cross Tab and Window Communications, 4 options
The Need
Frequently when dealing with large web apps there comes a need to open another window either using window.open() or by embedding an iFrame and changing the iFrame source. At that point any changes made in one window need to be reflected in the original or parent window. But how do you indicate to the original window when and what something changes?
The Options
There are four ways to achieve this that I know about
- Directly
- Via the Server
- localStorage (HTML5)
- postMessage (HTML5)
Directly
If one window opens another window it can retain a reference to the opened window. Similarly, a window opened by another window can reference it's opener. (Assuming they are part of the same domain!). With the window reference in hand- JavaScript methods can be called directly on a different window/tab.
From the parent:
var child = window.open("/new-window/test");
child.doSomething();
From the child:
window.opener.doSomething();
However- IE ruins the day. Although this works great for simple scenarios, in practice it frequently becomes impossible to pass object parameters from one window to another in IE. Why exactly that is I can not say. However too frequently when passing complex object as parameters, especially if they are being referenced in both windows, IE throws an obscure error message. My hunch it's because of how IE was designed the object pointers in the native browser code are unable to be transferred. Whatever the reason- too often direct communication was met with an unwilling IE.
Via the Server
Instead of directly communicating- each window can send a message to the server via AJAX. All other windows can then poll the server to listen for any messages that were passed. A AJAX push mechanism or websockets can be used instead of polling. Every time the polling windows identify a message, they can perform a local function in their own window.
PROS
- Reliable.
- Server can keep logs of what's happening
- Server can tweak the data that's being sent over the wire
CONS
- Server stress of regular AJAX calls, and pings from all the listening windows.
- Developer overhead of creating a messaging system
- Not instant- need to wait for a roundtrip back and force from the server until the listening window takes action
localStorage
This feature is considered part of the new HTML5 API's. However this has full support going back to IE8 making this one of the older of the new API's to be out in the wild. localStorage's raison d'ĂȘtre is as it's name implies- storage. To this end it comes in two different flavors: localStorage and sessionStorage. localStorage allows you to store a string inside a browser based key/value map. This value is retained in storage even after the browser is closed, the computer is turned off, and the user has gone home for the night. sessionStorage has an identical API and does the same things, however, like it's name implies the value is saved only has long as the browser session is alive. If the browser is closed, the data is lost. The values that are saved can be retrieved by any website that is part of the same domain. So if you have a page at www.example.com/one who saves a value, www.example.com/two is also allowed to retrieve that value.
While storage is nice to have and all it's actually the unique behavior of storage events that we can use to our advantage. Every time data is added to the localStorage or sessionStorage an event is fired in all windows that are part of the same domain. So if we want to trigger an event we can simply save some data in localStorage and every other window that wants to respond can listen for the storage event.
localStorage.setItem("myEvent", params);
window.addEventListener("storage", function(event){
switch(event.key){
case "myEvent": //do I care about event?
var params = event.newValue;
//do something with params
break;
}
}, false);
Note: if using jQuery the data will be in the event.originalEvent. For example event.originalEvent.newValue and event.originalEvent.key
As to be expected there are some caveats. For starters IE10 sessionStorage is very quirky. Sometimes after setting an item, calling sessionStorage.getItem(key) on other windows brings the old values! Personally I've given up on sessionStorage and now only use localStorage. That decision will need to be reevaluated as the support improves. Also on some browsers the event only fires when a new value is different than the old value. So if you just want to trigger an event without params, or use the same params over again you will need to add a timestamp to your value object so that it ensures it's unique. Another caveat is that localStorage only saves strings. So all your objects need to be serialized using JSON.stringify(). On the other end the listeners will then need to deserialize the string back into an object. And finally IE11 has several weird bugs related to localStorage events. They just don't fire consistently! I've learned this the hard way :(
PROS
- Can target any browser window without a direct reference
- Can target multiple windows
- Limited to same domain
- Really easy to use
CONS
- Can't target a specific window (see inverse as PRO)
- Values need to be serialized
- IE10
- IE11
- Did I mention IE?
postMessage
A newish HTML5 feature that lets windows send [post] messages between one another. Similar to direct communication the browser windows need a reference to the opened or opener window. With that reference in hand a message can be passed to the other windows. This method is a bit of a cross between direct communications method and the localStorage technique. It's similar to direct communication in that you need a reference to the window that you would like to communicate with. However, on a implementation level it's closer localStorage since you essentially create a message event that needs to be listened for. Here though, the message does not need to be serialized.
Unlike all the other options on the list this works cross-domain! However cross domain messaging turns out to be a mixed bag. While it's great if you need it can become a security concern for that same reason if not handled correctly. To mitigate the risks of cross domain snooping by evil frames when sending a message you can limit the domains that are capable of listening to your message.
var child = window.open("child.html");
child.postMessage({param: value}, "/"); //"/" limits to same domain
window.addEventListener("message", function(event){
var params = event.data;
//do something with params
}, false);
Note: if using jQuery the data will be in the event.originalEvent.data
PROS
- Can pass objects without serialization
- Can pass object without IE having a cow
CONS
- Need a reference to the window that you are sending the message to
- Potentially open to security concerns if not careful.
Conclusion
All four methods have situations where they come in handy. Each have their own set of caveats. Unfortunately IE bugs put a damper of both direct communication and localStorage events.