Important: This documentation covers Yarn 1 (Classic).
For Yarn 2+ docs and migration guide, see yarnpkg.com.

Package detail

strictly-typed-events

UselessPickles643MIT0.0.9TypeScript support: included

An Event emitting/subscription library designed for simplicity, convenience, and type-safety in TypeScript projects.

typescript, strict, strictly, typed, event, events, emit, emitter, subscribe, subscription, dispatch, dispatcher, observe, observer, listen, listener

readme

npm version Join the chat at https://gitter.im/ts-enum-util/Lobby Build Status Coverage Status

strictly-typed-events

An Event emitting/subscription library designed for simplicity, convenience, and type-safety in TypeScript projects.

Contents

Installation

Install via npm:

npm i -s strictly-typed-events

Why?

Despite the many event libraries already available on npm, I could not find one that met my desires. My goal is to easily setup a class that emits well-defined events, with type-safety while emitting and subscribing/handling, using minimal boilerplate. It should also be simple and intuitive to emit the events, subscribe to events, and cancel subscriptions to events.

Some design goals:

  • Simplicity of an "event" just being a call to a named handler function, with the "payload" being represented as one or more function parameters.
  • Define all events for a given class in terms of a single interface of event handler function signatures.
  • Support subscribing to multiple events with one call, which subsequently supports cancelling that entire subscription to multiple events with one call.
  • Easily support exposing only the means to subscribe to events, while keeping the means to emit events private.

Example Usage/Patterns

Here's some basic usage examples of strictly-typed-events and suggested patterns to get you started. See documentation in the source code (detailed TSDoc/JSDoc style comments on all types/classes/methods/etc.) for full details.

Subscribing and Cancelling Subscriptions

For the following examples, assume there is a variable source that implements this library's EventSource interface (has the on()/once()/subscribe() methods to subscribe to events).

Subscribe to one event at a time:

// Event name will be type-safe based on valid event names for the
// event source (IDE can auto-complete it!).
// Event handler parameter types will be inferred based on
// signature of event.
// Hold onto the "cancel" function returned when subscribing.
const cancel = source.on("nameChanged", (newname, oldName) => {
    // do stuff
});

// Simply call the cancel function to cancel the subscription
// to the event.
cancel();

Subscribe to one event with a one-time-only handler:

// This handler will self-cancel its own subscription when it is called.
// You can still store the returned cancel function and call it in case you
// need to cancel the subscription before the first time it is called.
source.once("nameChanged", (newname, oldName) => {
    // do stuff
});

Get a Promise that will be resolved the next time an event is emitted:

// Similar to `once()`, except that it returns a promise that resolves to a
// tuple of the event handler arguments.
source.onceAsPromise("nameChanged").then(([newname, oldName]) => {
    // do stuff
});

Or subscribe to multiple events at once:

// Hold onto the "cancel" function returned when subscribing.
const cancel = source.subscribe({
    // Provide handlers for any number of events in this object.
    // All type-safe, of course.
    nameChanged: (newname, oldName) => {
        // do stuff
    },
    // Wrap the handler in the `once()` function to make it a
    // one-time-only handler
    anotherEvent: once((whatever) => {
        // do stuff
    }),
});

// Simply call the cancel function to cancel the subscription
// to ALL events that were originally included in the subscription.
cancel();

Add Events via Inheritence

Here's the simplest, lowest-effort way to add events to a class. This works well for simple situations where your class is not already extending another class, and you want the on() subscription method to be directly on your class.

import { WithEventEmitter } from "strictly-typed-events";

// Simply extend `WithEventEmitter<>`, and define your events
// in the type parameter.
// Your class will now be an implementation of `EventSource`,
class Foo extends WithEventEmitter<{
    /**
     * You can document your event signatures, and IDEs
     * will be able to show this documentation in various
     * contexts where you emit this event or subscribe to
     * this event.
     * @param newName - The new name.
     * @param oldName - The old name.
     */
    nameChanged(newName: string, oldName: string): void;
    /**
     * Another event.
     * Define all events conveniently in one place
     */
    anotherEvent(whatever: number): void;
}> {
    public constructor(private readonly name: string) {
        super();
    }

    public setName(newName: string): void {
        const oldName = this.name;
        this.name = newName;
        // `this.emit` is a special protected property inherited from
        // `WithEventEmitter` with a method for each event.
        // Just call the method (strictly typed for IDE autocomplete, etc.)
        this.emit.nameChanged(newName, oldName);
    }
}

// Sample instance
const foo = new Foo();

// Your class itself is an `EventSource` with the "on()" method
// for subscribing to events.
const cancel = foo.on("nameChanged", (newname, oldName) => {
    // do stuff
});

// Cancel subscription
cancel();

Add Events via Composition

If you either don't want to, or are unable to, use inheritence to add events to your class, then you can do it through composition instead.

This approach guarantees that you have no conflicts between properties/methods on your class and WithEventEmitter.

import { EventEmitter } from "strictly-typed-events";

class Foo {
    // Initialize a private `EventEmitter` instance, and define your
    // events in the type parameter.
    private readonly emitter = new EventEmitter<{
        /**
         * You can document your event signatures, and IDEs
         * will be able to show this documentation in various
         * contexts where you emit this event or subscribe to
         * this event.
         * @param newName - The new name.
         * @param oldName - The old name.
         */
        nameChanged(newName: string, oldName: string): void;
        /**
         * Another event.
         * Define all events conveniently in one place
         */
        anotherEvent(whatever: number): void;
    }>();

    // Also expose your `EventEmitter` publicly, but as an
    // `EventSource` implementationthat only exposes the
    // ability subscribe to events.
    public readonly events = this.emitter.toEventSource();

    public constructor(private readonly name: string) {}

    public setName(newName: string): void {
        const oldName = this.name;
        this.name = newName;
        // Use the private EventEmitter instance to emit
        // events.
        this.emitter.emit.nameChanged(newName, oldName);
    }
}

// Sample instance
const foo = new Foo();

// Use the public `events` property on your class to
// subscribe to events.
const cancel = foo.events.on("nameChanged", (newname, oldName) => {
    // do stuff
});

// Cancel subscription
cancel();

changelog

Changelog

All notable changes to this project will be documented in this file.

The format is based on Keep a Changelog, and this project adheres to Semantic Versioning.

0.0.9 2021-10-25

Fixed

  • Fixed broken Travis CI link in README.
  • Updated README to reflect changes for version 0.0.8.
  • Fix typo in CHANGELOG.

0.0.8 2021-10-25

Added

  • New EventEmitter.toEventSource() method that replaces EventEmitter.asEventSource(). This new method creates and returns a new implementation of EventSource (instead of simply typecasting) to avoid unintentionally exposing the means to emit events at runtime.
  • New EventsType helper type that is similar to EventHandlersType, but returns the original raw Events interface of an EventSource type.

Changed

  • Misc. documentation fixes/improvements.

Removed

  • Removed EventEmitter.asEventSource(). Replaced by EventEmitter.toEventSource().

Fixed

  • Helper types EventHandlersType and EventSourceType now actually work. Verified via new type tests.

0.0.7 2021-03-16

Added

  • onceAsPromise() method that returns a Promise that resolves the next time the specified event is emitted.

0.0.6 2021-03-09

Added

  • Support for unique symbol event names to the EventSource.subscribe() method.

0.0.5 2021-03-08

Changed

  • Reworked/simplified implementation of EventEmitter by extracting some of the code into an abstract base class.
  • Reworked EventSource interface so that there are now 3 distinct methods for subscribing:
    • on(): Subscribe to a single event by name
      • Supports unique symbol event names!
    • once(): Subscribe to a single event by name with a one-time-only handler.
      • Supports unique symbol event names!
    • subscribe(): Subscribe to one or more events via an object keyed by event name.
      • Wrap the handler implementation with the new once() function to mark it as a one-time-only handler.
      • Supports string event names only.
  • Distributed code is now bundled with rollup.
  • Signatures of events MUST have a return type of void now. Previously could be any type.
  • Implementations of handlers may now return a Promise<void> (allows for async implementations).
  • Simplified code related to storing/executing one-time subscriptions.
  • Event handlers are now always called without a this context for simplicity.

Removed

  • onSubscribe handler support in EventEmitter. This was an interesting experiment, but I've deemed it have no real value and not worth the complexity.

0.0.4 2021-03-05

Added

  • EventsType helper type for extracting the Events interface from any EventSource type.
  • Basic usage examples/patterns in README.

Changed

  • Renamed project to strictly-typed-events to match NPM package name.
  • Renamed EventSource.subscribe method to EventSource.on (cascades down to all implementations of the interface too).
  • Renamed EventPublisher to EventEmitter.
    • Renamed property publish to emit.
  • Renamed WithEventPublisher to WithEventEmitter.
    • Renamed property publish to emit.

0.0.3 2021-03-04

Changed

  • Changed NPM package name to strictly-typed-events (v0.0.2 is actually published under this name)

0.0.2 2021-03-04

Changed

  • Rename project and npm package to ts-event-emitter (FAILED; name not accepted by NPM).
  • Retroactively re-published as strictly-typed-events.

0.0.1 2021-03-03

Initial release.

TODO:

  • Finalize naming/terminology.
  • Write good README content (sorry; source code documentation will have to suffice for now).
  • Use rollup to bundle code?