typescript
45 lines · 8 steps
A type-safe event emitter in TypeScript
How generics tie event names to their exact argument types so listeners and emits stay in sync at compile time.
Explained by
highlit
1type EventMap = Record<string, unknown[]>;
2
3type Listener<Args extends unknown[]> = (...args: Args) => void;
4
5export class TypedEmitter<Events extends EventMap> {
6 private listeners: {
7 [K in keyof Events]?: Set<Listener<Events[K]>>;
8 } = {};
9
10 on<K extends keyof Events>(event: K, listener: Listener<Events[K]>): this {
11 (this.listeners[event] ??= new Set()).add(listener);
12 return this;
13 }
14
15 once<K extends keyof Events>(event: K, listener: Listener<Events[K]>): this {
16 const wrapper: Listener<Events[K]> = (...args) => {
17 this.off(event, wrapper);
18 listener(...args);
19 };
20 return this.on(event, wrapper);
21 }
22
23 off<K extends keyof Events>(event: K, listener: Listener<Events[K]>): this {
24 this.listeners[event]?.delete(listener);
25 return this;
26 }
27
28 emit<K extends keyof Events>(event: K, ...args: Events[K]): boolean {
29 const handlers = this.listeners[event];
30 if (!handlers || handlers.size === 0) return false;
31 for (const handler of [...handlers]) {
32 handler(...args);
33 }
34 return true;
35 }
36
37 removeAllListeners<K extends keyof Events>(event?: K): this {
38 if (event === undefined) {
39 this.listeners = {};
40 } else {
41 delete this.listeners[event];
42 }
43 return this;
44 }
45}
01 / 01
STEP 01
‹ swipe to step through ›
Walkthrough
Space play
←→ step
click any line
Three takeaways
- 1Mapping an event-name type to its argument tuple lets the compiler check every listener and emit for you.
- 2Copying the handler set before iterating avoids skipping or double-firing when handlers mutate the set mid-emit.
- 3A `once` listener is just a normal listener that removes itself before delegating to the real callback.
Related explainers
typescript
type RequestState<T, E = string> = | { status: "idle" } | { status: "loading" } | { status: "success"; data: T; fetchedAt: number }
Modeling request state with discriminated unions
discriminated-unions
exhaustiveness-checking
state-machine
Intermediate
8 steps
typescript
import { Directive, Input, TemplateRef,
Building a structural *appUnless directive in Angular
structural-directive
template-rendering
dependency-injection
Intermediate
8 steps
java
public final class Debouncer<T> { private final ScheduledExecutorService scheduler = Executors.newSingleThreadScheduledExecutor(runnable -> {
How a debouncer collapses bursts in Java
debounce
concurrency
scheduling
Intermediate
9 steps
typescript
import { inject } from '@angular/core'; import { ResolveFn, Router, ActivatedRouteSnapshot } from '@angular/router'; import { catchError, of, EMPTY } from 'rxjs'; import { Article } from './models/article';
Prefetching route data with an Angular resolver
route-resolver
dependency-injection
rxjs
Intermediate
6 steps
php
<?php declare(strict_types=1);
Validating registration input with filter_var
validation
sanitization
filter_var
Intermediate
8 steps
typescript
interface Order { id: number; customerId: string; total: number;
A type-safe groupBy in TypeScript
generics
reduce
type-inference
Intermediate
6 steps
Share this explainer
Here's the card — post it anywhere.
Made with highlit — turn any snippet into a walkthrough like this in about a minute.
Explain your code
Embed this explainer
Drop the interactive walkthrough into a blog or docs. Views never cost a credit.
<iframe src="https://highlit.co/explainers/a-type-safe-event-emitter-in-typescript-explained-typescript-3318/embed?autoplay=1" width="100%" height="520" loading="lazy" style="border:0"></iframe>
Autoplay is on by default — add ?autoplay=0 to start paused.