export interface EventEmitterOptions { once?: boolean; // call once and remove prepend?: boolean; // put listener to the beginning } export type EventEmitterCallback = (...data: D) => void | boolean; export class EventEmitter { protected listeners: [EventEmitterCallback, EventEmitterOptions][] = []; addListener(callback: EventEmitterCallback, options: EventEmitterOptions = {}) { const fn = options.prepend ? "unshift" : "push"; this.listeners[fn]([callback, options]); } removeListener(callback: EventEmitterCallback) { this.listeners = this.listeners.filter(([cb]) => cb !== callback); } removeAllListeners() { this.listeners.length = 0; } emit(...data: D) { for (const [callback, { once }] of this.listeners) { if (once) { this.removeListener(callback); } if (callback(...data) === false) { break; } } } }