mirror of
https://github.com/angular/angular
synced 2026-05-24 09:28:37 +00:00
This PR also simplifies the imperative ordering of async methods. It moves the logic into a generic abstraction `PriorityAwareMessageBus`.
73 lines
2.1 KiB
TypeScript
73 lines
2.1 KiB
TypeScript
import { MessageBus, Events, Parameters } from 'protocol';
|
|
|
|
type AnyEventCallback<Ev> = <E extends keyof Ev>(topic: E, args: Parameters<Ev[E]>) => void;
|
|
|
|
export class IFrameMessageBus extends MessageBus<Events> {
|
|
private _listeners: any[] = [];
|
|
|
|
constructor(private _source: string, private _destination: string, private _docWindow: () => Window) {
|
|
super();
|
|
}
|
|
|
|
onAny(cb: AnyEventCallback<Events>): () => void {
|
|
const listener = (e: MessageEvent) => {
|
|
if (!e.data || !e.data.topic || e.data.source !== this._destination) {
|
|
return;
|
|
}
|
|
cb(e.data.topic, e.data.args);
|
|
};
|
|
window.addEventListener('message', listener);
|
|
this._listeners.push(listener);
|
|
return () => {
|
|
this._listeners.splice(this._listeners.indexOf(listener), 1);
|
|
window.removeEventListener('message', listener);
|
|
};
|
|
}
|
|
|
|
on<E extends keyof Events>(topic: E, cb: Events[E]): () => void {
|
|
const listener = (e: MessageEvent) => {
|
|
if (!e.data || e.data.source !== this._destination || !e.data.topic) {
|
|
return;
|
|
}
|
|
if (e.data.topic === topic) {
|
|
cb.apply(null, e.data.args);
|
|
}
|
|
};
|
|
window.addEventListener('message', listener);
|
|
this._listeners.push(listener);
|
|
return () => {
|
|
this._listeners.splice(this._listeners.indexOf(listener), 1);
|
|
window.removeEventListener('message', listener);
|
|
};
|
|
}
|
|
|
|
once<E extends keyof Events>(topic: E, cb: Events[E]): void {
|
|
const listener = (e: MessageEvent) => {
|
|
if (!e.data || e.data.source !== this._destination || !e.data.topic) {
|
|
return;
|
|
}
|
|
if (e.data.topic === topic) {
|
|
cb.apply(null, e.data.args);
|
|
window.removeEventListener('message', listener);
|
|
}
|
|
};
|
|
window.addEventListener('message', listener);
|
|
}
|
|
|
|
emit<E extends keyof Events>(topic: E, args?: Parameters<Events[E]>): boolean {
|
|
this._docWindow().postMessage(
|
|
{
|
|
source: this._source,
|
|
topic,
|
|
args,
|
|
},
|
|
'*'
|
|
);
|
|
return true;
|
|
}
|
|
|
|
destroy(): void {
|
|
this._listeners.forEach((l) => window.removeEventListener('message', l));
|
|
this._listeners = [];
|
|
}
|
|
}
|