@medievalrain/emitter 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,15 @@
1
+ //#region src/types.d.ts
2
+ type EventMap = Record<string, (...args: any[]) => void>;
3
+ type CallbackOptions = {
4
+ once?: boolean;
5
+ signal?: AbortSignal;
6
+ };
7
+ //#endregion
8
+ //#region src/emitter.d.ts
9
+ declare const createEmitter: <Events extends EventMap>() => {
10
+ emit: <EM extends keyof Events>(event: EM, ...args: Parameters<Events[EM]>) => void;
11
+ on: <EM extends keyof Events>(event: EM, callback: Events[EM], options?: CallbackOptions) => void;
12
+ off: <EM extends keyof Events>(event: EM, callback: Events[EM]) => void;
13
+ };
14
+ //#endregion
15
+ export { type CallbackOptions, type EventMap, createEmitter };
package/dist/index.js ADDED
@@ -0,0 +1,51 @@
1
+ //#region src/emitter.ts
2
+ const createEmitter = () => {
3
+ const callbackMap = /* @__PURE__ */ new Map();
4
+ const off = (event, callback) => {
5
+ const callbacks = callbackMap.get(event);
6
+ if (!callbacks) return;
7
+ const controller = callbacks.get(callback)?.controller;
8
+ if (controller) controller.abort();
9
+ callbacks.delete(callback);
10
+ };
11
+ const emit = (event, ...args) => {
12
+ const callbacks = callbackMap.get(event);
13
+ if (!callbacks?.size) return;
14
+ Array.from(callbacks?.entries()).forEach(([callback, { once }]) => {
15
+ if (!callbacks.has(callback)) return;
16
+ try {
17
+ callback(...args);
18
+ } finally {
19
+ if (once) off(event, callback);
20
+ }
21
+ });
22
+ };
23
+ const on = (event, callback, options) => {
24
+ if (options?.signal && options.signal.aborted) return;
25
+ let callbacks = callbackMap.get(event);
26
+ let unsubController = void 0;
27
+ if (!callbacks) {
28
+ callbacks = /* @__PURE__ */ new Map();
29
+ callbackMap.set(event, callbacks);
30
+ } else if (callbacks.get(callback)) return;
31
+ if (options?.signal) {
32
+ unsubController = new AbortController();
33
+ options.signal.addEventListener("abort", () => off(event, callback), {
34
+ once: true,
35
+ signal: unsubController.signal
36
+ });
37
+ }
38
+ callbacks.set(callback, {
39
+ once: options?.once,
40
+ controller: unsubController
41
+ });
42
+ };
43
+ return {
44
+ emit,
45
+ on,
46
+ off
47
+ };
48
+ };
49
+
50
+ //#endregion
51
+ export { createEmitter };
package/package.json ADDED
@@ -0,0 +1,33 @@
1
+ {
2
+ "name": "@medievalrain/emitter",
3
+ "version": "0.1.0",
4
+ "description": "A typesafe event emitter",
5
+ "type": "module",
6
+ "files": [
7
+ "dist"
8
+ ],
9
+ "main": "./dist/index.js",
10
+ "module": "./dist/index.js",
11
+ "types": "./dist/index.d.ts",
12
+ "exports": {
13
+ ".": "./dist/index.js",
14
+ "./package.json": "./package.json"
15
+ },
16
+ "scripts": {
17
+ "build": "tsdown",
18
+ "dev": "tsdown --watch",
19
+ "test": "vitest"
20
+ },
21
+ "devDependencies": {
22
+ "@types/node": "24.5.0",
23
+ "oxlint": "1.15.0",
24
+ "prettier": "3.6.2",
25
+ "tsdown": "0.15.1",
26
+ "typescript": "5.9.2",
27
+ "vitest": "4.0.0-beta.11"
28
+ },
29
+ "repository": {
30
+ "type": "git",
31
+ "url": "git+https://github.com/medievalrain/emitter.git"
32
+ }
33
+ }