diff options
Diffstat (limited to 'packages/excalidraw/emitter.ts')
| -rw-r--r-- | packages/excalidraw/emitter.ts | 51 |
1 files changed, 51 insertions, 0 deletions
diff --git a/packages/excalidraw/emitter.ts b/packages/excalidraw/emitter.ts new file mode 100644 index 0000000..9382697 --- /dev/null +++ b/packages/excalidraw/emitter.ts @@ -0,0 +1,51 @@ +import type { UnsubscribeCallback } from "./types"; + +type Subscriber<T extends any[]> = (...payload: T) => void; + +export class Emitter<T extends any[] = []> { + public subscribers: Subscriber<T>[] = []; + + /** + * Attaches subscriber + * + * @returns unsubscribe function + */ + on(...handlers: Subscriber<T>[] | Subscriber<T>[][]): UnsubscribeCallback { + const _handlers = handlers + .flat() + .filter((item) => typeof item === "function"); + + this.subscribers.push(..._handlers); + + return () => this.off(_handlers); + } + + once(...handlers: Subscriber<T>[] | Subscriber<T>[][]): UnsubscribeCallback { + const _handlers = handlers + .flat() + .filter((item) => typeof item === "function"); + + _handlers.push(() => detach()); + + const detach = this.on(..._handlers); + return detach; + } + + off(...handlers: Subscriber<T>[] | Subscriber<T>[][]) { + const _handlers = handlers.flat(); + this.subscribers = this.subscribers.filter( + (handler) => !_handlers.includes(handler), + ); + } + + trigger(...payload: T) { + for (const handler of this.subscribers) { + handler(...payload); + } + return this; + } + + clear() { + this.subscribers = []; + } +} |
