aboutsummaryrefslogtreecommitdiffstats
path: root/packages/excalidraw/emitter.ts
diff options
context:
space:
mode:
Diffstat (limited to 'packages/excalidraw/emitter.ts')
-rw-r--r--packages/excalidraw/emitter.ts51
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 = [];
+ }
+}