|
1 | | -import { Actions, EventData } from './types'; |
| 1 | +import { EventType, EventData } from '../actions/types'; |
| 2 | + |
| 3 | +const ALLOWED_ORIGINS: string[] = |
| 4 | + process.env.NODE_ENV === 'development' |
| 5 | + ? ['http://localhost:3000'] |
| 6 | + : ['https://nmrxiv.org']; |
2 | 7 |
|
3 | 8 | const namespace = 'nmr-wrapper'; |
4 | 9 |
|
5 | | -function trigger<T extends Actions>(action: T, data: EventData<T>) { |
6 | | - const event = new CustomEvent(`${namespace}:${action}`, { |
7 | | - detail: data, |
8 | | - }); |
9 | | - window.dispatchEvent(event); |
| 10 | +function parseOrigin(origin: string) { |
| 11 | + const urlSegments = origin.split('://'); |
| 12 | + const hostSegments = urlSegments[1].split('.'); |
| 13 | + |
| 14 | + const url = `${urlSegments[0]}://${hostSegments |
| 15 | + .slice(hostSegments.length > 1 ? 1 : 0) |
| 16 | + .join('.')}`; |
| 17 | + |
| 18 | + return url; |
| 19 | +} |
| 20 | + |
| 21 | +function trigger<T extends EventType>(type: T, data: EventData<T>) { |
| 22 | + window.postMessage({ type: `${namespace}:${type}`, data }); |
10 | 23 | } |
11 | 24 |
|
12 | | -function on<T extends Actions>( |
13 | | - action: T, |
14 | | - dataListener: (data: EventData<T>, event?: Event) => void, |
| 25 | +function on<T extends EventType>( |
| 26 | + type: T, |
| 27 | + dataListener: (data: EventData<T>) => void, |
15 | 28 | options?: boolean | AddEventListenerOptions, |
16 | 29 | ) { |
17 | | - function listener(object: Event) { |
18 | | - const { detail, ...event } = object as CustomEvent; |
19 | | - dataListener?.(detail, event); |
| 30 | + function listener(event: MessageEvent) { |
| 31 | + const { |
| 32 | + origin, |
| 33 | + data: { type: targetType, data }, |
| 34 | + } = event; |
| 35 | + |
| 36 | + if (!ALLOWED_ORIGINS.includes(parseOrigin(origin))) { |
| 37 | + throw new Error(`Invalid Origin ${origin}`); |
| 38 | + } |
| 39 | + |
| 40 | + if (`${namespace}:${type}` === targetType) { |
| 41 | + dataListener?.(data); |
| 42 | + } |
20 | 43 | } |
21 | | - window.addEventListener(`${namespace}:${action}`, listener, options); |
22 | | -} |
| 44 | + window.addEventListener(`message`, listener, options); |
23 | 45 |
|
24 | | -function clean(action: Actions, listener: (object: Event) => void) { |
25 | | - window.removeEventListener(`${namespace}:${action}`, listener); |
| 46 | + return () => window.removeEventListener(`message`, listener); |
26 | 47 | } |
27 | 48 |
|
28 | | -export default { trigger, on, clean }; |
| 49 | +export default { trigger, on }; |
0 commit comments