|
1 | 1 | import { setMaxListeners as nodeSetMaxListeners } from 'events'
|
2 | 2 |
|
3 |
| -export interface EventCallback<EventType> { (evt: EventType): void } |
4 |
| -export interface EventObject<EventType> { handleEvent: EventCallback<EventType> } |
5 |
| -export type EventHandler<EventType> = EventCallback<EventType> | EventObject<EventType> |
6 |
| - |
7 |
| -interface Listener { |
8 |
| - once: boolean |
9 |
| - callback: any |
10 |
| -} |
11 |
| - |
12 |
| -/** |
13 |
| - * Adds types to the EventTarget class. Hopefully this won't be necessary forever. |
14 |
| - * |
15 |
| - * https://github.com/microsoft/TypeScript/issues/28357 |
16 |
| - * https://github.com/microsoft/TypeScript/issues/43477 |
17 |
| - * https://github.com/microsoft/TypeScript/issues/299 |
18 |
| - * etc |
19 |
| - */ |
20 |
| -export interface TypedEventTarget <EventMap extends Record<string, any>> extends EventTarget { |
21 |
| - addEventListener<K extends keyof EventMap>(type: K, listener: EventHandler<EventMap[K]> | null, options?: boolean | AddEventListenerOptions): void |
22 |
| - |
23 |
| - listenerCount (type: string): number |
24 |
| - |
25 |
| - removeEventListener<K extends keyof EventMap>(type: K, listener?: EventHandler<EventMap[K]> | null, options?: boolean | EventListenerOptions): void |
26 |
| - |
27 |
| - removeEventListener (type: string, listener?: EventHandler<Event>, options?: boolean | EventListenerOptions): void |
28 |
| - |
29 |
| - safeDispatchEvent<Detail>(type: keyof EventMap, detail: CustomEventInit<Detail>): boolean |
30 |
| -} |
31 |
| - |
32 |
| -/** |
33 |
| - * An implementation of a typed event target |
34 |
| - * etc |
35 |
| - */ |
36 |
| -export class TypedEventEmitter<EventMap extends Record<string, any>> extends EventTarget implements TypedEventTarget<EventMap> { |
37 |
| - #listeners = new Map<any, Listener[]>() |
38 |
| - |
39 |
| - listenerCount (type: string): number { |
40 |
| - const listeners = this.#listeners.get(type) |
41 |
| - |
42 |
| - if (listeners == null) { |
43 |
| - return 0 |
44 |
| - } |
45 |
| - |
46 |
| - return listeners.length |
47 |
| - } |
48 |
| - |
49 |
| - addEventListener<K extends keyof EventMap>(type: K, listener: EventHandler<EventMap[K]> | null, options?: boolean | AddEventListenerOptions): void |
50 |
| - addEventListener (type: string, listener: EventHandler<Event>, options?: boolean | AddEventListenerOptions): void { |
51 |
| - super.addEventListener(type, listener, options) |
52 |
| - |
53 |
| - let list = this.#listeners.get(type) |
54 |
| - |
55 |
| - if (list == null) { |
56 |
| - list = [] |
57 |
| - this.#listeners.set(type, list) |
58 |
| - } |
59 |
| - |
60 |
| - list.push({ |
61 |
| - callback: listener, |
62 |
| - once: (options !== true && options !== false && options?.once) ?? false |
63 |
| - }) |
64 |
| - } |
65 |
| - |
66 |
| - removeEventListener<K extends keyof EventMap>(type: K, listener?: EventHandler<EventMap[K]> | null, options?: boolean | EventListenerOptions): void |
67 |
| - removeEventListener (type: string, listener?: EventHandler<Event>, options?: boolean | EventListenerOptions): void { |
68 |
| - super.removeEventListener(type.toString(), listener ?? null, options) |
69 |
| - |
70 |
| - let list = this.#listeners.get(type) |
71 |
| - |
72 |
| - if (list == null) { |
73 |
| - return |
74 |
| - } |
75 |
| - |
76 |
| - list = list.filter(({ callback }) => callback !== listener) |
77 |
| - this.#listeners.set(type, list) |
78 |
| - } |
79 |
| - |
80 |
| - dispatchEvent (event: Event): boolean { |
81 |
| - const result = super.dispatchEvent(event) |
82 |
| - |
83 |
| - let list = this.#listeners.get(event.type) |
84 |
| - |
85 |
| - if (list == null) { |
86 |
| - return result |
87 |
| - } |
88 |
| - |
89 |
| - list = list.filter(({ once }) => !once) |
90 |
| - this.#listeners.set(event.type, list) |
91 |
| - |
92 |
| - return result |
93 |
| - } |
94 |
| - |
95 |
| - safeDispatchEvent<Detail>(type: keyof EventMap, detail: CustomEventInit<Detail>): boolean { |
96 |
| - return this.dispatchEvent(new CustomEvent<Detail>(type as string, detail)) |
97 |
| - } |
98 |
| -} |
99 |
| - |
100 |
| -/** |
101 |
| - * CustomEvent is a standard event but it's not supported by node. |
102 |
| - * |
103 |
| - * Remove this when https://github.com/nodejs/node/issues/40678 is closed. |
104 |
| - * |
105 |
| - * Ref: https://developer.mozilla.org/en-US/docs/Web/API/CustomEvent |
106 |
| - */ |
107 |
| -class CustomEventPolyfill<T = any> extends Event { |
108 |
| - /** Returns any custom data event was created with. Typically used for synthetic events. */ |
109 |
| - public detail: T |
110 |
| - |
111 |
| - constructor (message: string, data?: EventInit & { detail: T }) { |
112 |
| - super(message, data) |
113 |
| - // @ts-expect-error could be undefined |
114 |
| - this.detail = data?.detail |
115 |
| - } |
116 |
| -} |
117 |
| - |
118 |
| -export const CustomEvent = globalThis.CustomEvent ?? CustomEventPolyfill |
119 |
| - |
120 | 3 | // create a setMaxListeners that doesn't break browser usage
|
121 | 4 | export const setMaxListeners: typeof nodeSetMaxListeners = (n, ...eventTargets) => {
|
122 | 5 | try {
|
|
0 commit comments