A cross-frame communication library
PairedFrame is a utility for when you have no choice but to embed part of your site in an iframe. It allows you to:
- Avoid scrollbars, clipping, and loading FOUC
- Synchronize SPA navigation
- Proxy modals and toast messages to the parent window
- Send arbitary messages between frames
Iframes impact both security and performance. PairedFrame does what it can to provide a seamless integration without making these problems worse.
Each PairedFrame instance acts as a stand-in for an iframe or parent frame.
const parent = new PairedFrame({
targetWindow: window.parent,
targetOrigin: 'https://child.loc:3000',
...options
});
PairedFrame is a Node-style event emitter.
parent.once('ready', () => {
myIframeElement.style.opacity = 1;
});
parent.on('heartbeat', console.log);
parent.emit('some-event-name', someEventData);
You can also send arbitrary messages to the counterpart frame, which will be emitted as events there.
parent.send('some-event-name', someEventData);
The class includes some low-level, promise-based methods to proxy interactive experiences (such as modal windows) from a child frame to a parent frame.
const confirmed = await parent.dialog({ type: 'modal', text: 'Are you sure?' });
The PairedFrame constructor accepts a dictionary of options:
Required
The origin of the frame being targeted, including protocol, hostname, and port (if different than the default expected for the protocol). Only messages hailing from a counterpart at this origin will be trusted.
Required
A reference to the counterpart's window. When connecting from a host frame to
an iframe, this is the iframe element's contentWindow
. When connecting from
an iframe to its host, this is window.parent
.
Boolean. If true, the local scrollHeight will be monitored and broadcast to the counterpart. This should be enabled for the child frame to enable automatic height syncing.
Boolean. If true, the height of resizeElement
will be automatically updated
to match the scrollHeight of the counterpart frame. This should be enabled for
the parent frame to enable automatic height syncing.
Required if autoResize is true
A reference to the element that should be resized to match the counterpart's scrollHeight. For best performance, this should be a wrapper div around the iframe with overflow set to "hidden". The iframe itself should have a static height large enough to accommodate the maximum height of its content.
If your iframe's maximum content height is not predictable (for example, if it has infinitely scrolling content), you can also pass a reference to the iframe element itself. This will work, but may degrade performance during animations.
Boolean. If true, the local pathname will be monitored and broadcast to the counterpart. This should be enabled for both frames to enable automatic navigation syncing.
Boolean. If true, the local pathname will be automatically updated (using
history.replaceState) in response to pathname changes in the counterpart. Then
a synthetic popstate
event (which typically represents browser navigation) is
fired to trigger any active routing library to examine the new pathname and, if
necessary, render a new state.
(remotePath, requestedPath) => localPath
Function. If provided, will be used to convert the counterpart's new pathname into the equivalent local pathname. This allows the parent and child frames to have uniquely named routes that can still be mapped to each other.
(localPath) => remotePath
Function. If provided, will be used to translate new local pathnames into the
requestedPath
argument that will be fed to the counterpart's translatePath.
The counterpart may optionally defer to requestedPath
rather than deriving a
path from remotePath
. This allows one frame to manage route syncing for both
frames.
Notifies the counterpart to initiate some kind of interactive dialog. Returns a promise that will be resolved with the result of that dialog.
PairedFrame only manages communication of the dialog state; it is up to the
user to implement the actual modal, toast message, etc. To that end, config
can be structured however you like.
A special handler for accepting and responding to dialog requests. callback
will receive the dialog config
as its first argument and should return a
promise that resolves with the result of the interaction. Returns the
PairedFrame instance for chaining.
Registers a callback to fire for the given event. Returns the PairedFrame instance for chaining.
Registers a callback to fire once for the given event, then deregister itself. Returns the PairedFrame instance for chaining.
Deregisters a callback for the given event. Returns the PairedFrame instance for chaining.
Emits an event in the local frame, firing any registered callbacks. Returns the number of callbacks that were fired in response.
Emits an event in the counterpart frame, firing any registered callbacks. This
can be used to send arbitrary messages between frames. Returns true
.
Returns an array of all event names that currently have registered callbacks.
Returns an array of callbacks currently registered for the given event.
Returns the number of callbacks currently registered for the given event.
Removes the postMessage listener. Any callbacks attached to the "destroy" event will be fired, after which no callbacks will be fired or postMessages sent.
load
: Indicates that both frames have loaded, but the connection has not yet been set up.ping
: Indicates that the counterpart wishes to establish a connection.pong
: Indicates that the counterpart has approved the connection.ready
: Indicates that both frames are ready to communicate.postmessage-sent
: Indicates that a postMessage has been sent to the counterpart.postmessage-received
: Indicates that a postMessage has been received from the counterpart.resize
: Indicates that the counterpart's scrollHeight has changed.height-updated
: Indicates that the local frame's height has been auto-updated in response to aresize
event.navigate
: Indicates that the counterpart's pathname has changed.path-updated
: Indicates that the local frame's path has been auto-updated in response to anavigate
event.dialog-opened
: Indicates that a dialog has been requested (by either frame).dialog-closed
: Indicates that a dialog has been resolved (by either frame).heartbeat
: Sent every 250ms by each frame to monitor connection.connection-lost
: Indicates that 500ms have passed without a counterpart heartbeat event.connection-restored
: Indicates that a counterpart heartbeat event has been received following a lost connection.destroy
: Fired just before the instance removes its postMessage handler and stops sending or responding to events.*
: All events. Can be used for logging.
- Add parent.loc and child.loc to /etc/hosts pointing at 127.0.0.1
- Run
yarn demo
- Navigate to http://parent.loc:3000 in your browser