From 69177fbe11e5c50a6de7717719c639069f198666 Mon Sep 17 00:00:00 2001 From: iyegoroff Date: Sat, 1 Aug 2020 15:18:05 +0300 Subject: [PATCH 1/6] improved typing --- .gitignore | 1 + README.md | 20 ++++++--- package.json | 12 ++++-- src/index.ts | 75 ++++++++++++++++++++++------------ test/index_test.ts | 25 ++++++++---- test/test-types-compilation.ts | 75 +++++++++++++++++++++++++--------- tsconfig.json | 6 +-- 7 files changed, 147 insertions(+), 67 deletions(-) diff --git a/.gitignore b/.gitignore index eb570d3..bf29d4f 100644 --- a/.gitignore +++ b/.gitignore @@ -6,3 +6,4 @@ package-lock.json .DS_Store .idea +.vscode diff --git a/README.md b/README.md index 08a21bf..f81166e 100644 --- a/README.md +++ b/README.md @@ -80,9 +80,17 @@ emitter.off('foo', onFoo) // unlisten ### Typescript +Set `"strict": true` in your tsconfig.json to get improved type inferrence for `mitt` instance methods. + ```ts import mitt from 'mitt'; -const emitter: mitt.Emitter = mitt(); + +type Events = { + foo: string + bar?: number +} + +const emitter: mitt.Emitter = mitt(); ``` ## Examples & Demos @@ -114,7 +122,7 @@ const emitter: mitt.Emitter = mitt(); Mitt: Tiny (~200b) functional event emitter / pubsub. -Returns **Mitt** +Returns **Mitt** ### all @@ -126,7 +134,7 @@ Register an event handler for the given type. #### Parameters -- `type` **([string](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String) \| [symbol](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Symbol))** Type of event to listen for, or `"*"` for all events +- `type` **([string](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String) \| [symbol](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Symbol))** Type of event to listen for, or `'*'` for all events - `handler` **[Function](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Statements/function)** Function to call in response to given event ### off @@ -135,15 +143,15 @@ Remove an event handler for the given type. #### Parameters -- `type` **([string](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String) \| [symbol](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Symbol))** Type of event to unregister `handler` from, or `"*"` +- `type` **([string](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String) \| [symbol](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Symbol))** Type of event to unregister `handler` from, or `'*'` - `handler` **[Function](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Statements/function)** Handler function to remove ### emit Invoke all handlers for the given type. -If present, `"*"` handlers are invoked after type-matched handlers. +If present, `'*'` handlers are invoked after type-matched handlers. -Note: Manually firing "\*" handlers is not supported. +Note: Manually firing '\*' handlers is not supported. #### Parameters diff --git a/package.json b/package.json index 51b5cf8..55dd6c5 100644 --- a/package.json +++ b/package.json @@ -12,7 +12,7 @@ "scripts": { "test": "npm-run-all --silent typecheck lint mocha test-types", "mocha": "mocha test", - "test-types": "tsc test/test-types-compilation.ts --noEmit", + "test-types": "tsc test/test-types-compilation.ts --noEmit --strict", "lint": "eslint src test --ext ts --ext js", "typecheck": "tsc --noEmit", "bundle": "microbundle", @@ -78,7 +78,8 @@ "@typescript-eslint/no-explicit-any": 0, "@typescript-eslint/explicit-function-return-type": 0, "@typescript-eslint/explicit-module-boundary-types": 0, - "@typescript-eslint/no-empty-function": 0 + "@typescript-eslint/no-empty-function": 0, + "@typescript-eslint/no-non-null-assertion": 0 } }, "eslintIgnore": [ @@ -104,6 +105,9 @@ "sinon": "^9.0.2", "sinon-chai": "^3.5.0", "ts-node": "^8.10.2", - "typescript": "^3.9.3" + "typescript": "^3.9.7" + }, + "dependencies": { + "ts-toolbelt": "^6.13.37" } -} \ No newline at end of file +} diff --git a/src/index.ts b/src/index.ts index ae85607..2293eb3 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,28 +1,38 @@ +import { Union } from 'ts-toolbelt'; + export type EventType = string | symbol; // An event handler can take an optional event argument // and should not return a value -export type Handler = (event?: T) => void; -export type WildcardHandler = (type: EventType, event?: any) => void; +export type Handler = (event: T) => void; +export type WildcardHandler> = ( + type: keyof T, + event: T[keyof T] +) => void; // An array of all currently registered event handlers for a type -export type EventHandlerList = Array; -export type WildCardEventHandlerList = Array; +export type EventHandlerList = Array>; +export type WildCardEventHandlerList> = Array>; // A map of event types and their corresponding event handlers. -export type EventHandlerMap = Map; +export type EventHandlerMap> = Map< + keyof Events | '*', + EventHandlerList | WildCardEventHandlerList +>; -export interface Emitter { - all: EventHandlerMap; +export interface Emitter> { + all: EventHandlerMap; - on(type: EventType, handler: Handler): void; - on(type: '*', handler: WildcardHandler): void; + on(type: Key, handler: Handler): void; + on(type: '*', handler: WildcardHandler): void; - off(type: EventType, handler: Handler): void; - off(type: '*', handler: WildcardHandler): void; + off(type: Key, handler: Handler): void; + off(type: '*', handler: WildcardHandler): void; - emit(type: EventType, event?: T): void; - emit(type: '*', event?: any): void; + emit(type: Key, event: Events[Key]): void; + emit( + type: Union.Has extends 1 ? Key : never + ): void; } /** @@ -30,7 +40,12 @@ export interface Emitter { * @name mitt * @returns {Mitt} */ -export default function mitt(all?: EventHandlerMap): Emitter { +export default function mitt>( + all?: EventHandlerMap +): Emitter { + type GenericEventHandler = + | Handler + | WildcardHandler; all = all || new Map(); return { @@ -42,26 +57,26 @@ export default function mitt(all?: EventHandlerMap): Emitter { /** * Register an event handler for the given type. - * @param {string|symbol} type Type of event to listen for, or `"*"` for all events + * @param {string|symbol} type Type of event to listen for, or `'*'` for all events * @param {Function} handler Function to call in response to given event * @memberOf mitt */ - on(type: EventType, handler: Handler) { - const handlers = all.get(type); + on(type: Key, handler: GenericEventHandler) { + const handlers: Array | undefined = all!.get(type); const added = handlers && handlers.push(handler); if (!added) { - all.set(type, [handler]); + all!.set(type, [handler] as EventHandlerList); } }, /** * Remove an event handler for the given type. - * @param {string|symbol} type Type of event to unregister `handler` from, or `"*"` + * @param {string|symbol} type Type of event to unregister `handler` from, or `'*'` * @param {Function} handler Handler function to remove * @memberOf mitt */ - off(type: EventType, handler: Handler) { - const handlers = all.get(type); + off(type: Key, handler: GenericEventHandler) { + const handlers: Array | undefined = all!.get(type); if (handlers) { handlers.splice(handlers.indexOf(handler) >>> 0, 1); } @@ -69,17 +84,25 @@ export default function mitt(all?: EventHandlerMap): Emitter { /** * Invoke all handlers for the given type. - * If present, `"*"` handlers are invoked after type-matched handlers. + * If present, `'*'` handlers are invoked after type-matched handlers. * - * Note: Manually firing "*" handlers is not supported. + * Note: Manually firing '*' handlers is not supported. * * @param {string|symbol} type The event type to invoke * @param {Any} [evt] Any value (object is recommended and powerful), passed to each handler * @memberOf mitt */ - emit(type: EventType, evt: T) { - ((all.get(type) || []) as EventHandlerList).slice().map((handler) => { handler(evt); }); - ((all.get('*') || []) as WildCardEventHandlerList).slice().map((handler) => { handler(type, evt); }); + emit(type: Key, evt?: Events[Key]) { + ((all!.get(type) || []) as EventHandlerList) + .slice() + .map((handler) => { + handler(evt!); + }); + ((all!.get('*') || []) as WildCardEventHandlerList) + .slice() + .map((handler) => { + handler(type, evt!); + }); } }; } diff --git a/test/index_test.ts b/test/index_test.ts index 5c25b55..ba1dbff 100644 --- a/test/index_test.ts +++ b/test/index_test.ts @@ -1,4 +1,4 @@ -import mitt, { Emitter } from '..'; +import mitt, { Emitter, EventHandlerMap } from '..'; import chai, { expect } from 'chai'; import { spy } from 'sinon'; import sinonChai from 'sinon-chai'; @@ -15,7 +15,7 @@ describe('mitt', () => { const a = spy(); const b = spy(); map.set('foo', [a, b]); - const events = mitt(map); + const events = mitt<{ foo: undefined }>(map); events.emit('foo'); expect(a).to.have.been.calledOnce; expect(b).to.have.been.calledOnce; @@ -23,9 +23,21 @@ describe('mitt', () => { }); describe('mitt#', () => { - let events, inst: Emitter; - - beforeEach( () => { + const eventType = Symbol('eventType'); + type Events = { + foo: unknown; + constructor: unknown; + FOO: unknown; + bar: unknown; + Bar: unknown; + 'baz:bat!': unknown; + 'baz:baT!': unknown; + Foo: unknown; + [eventType]: unknown; + }; + let events: EventHandlerMap, inst: Emitter; + + beforeEach(() => { events = new Map(); inst = mitt(events); }); @@ -83,7 +95,6 @@ describe('mitt#', () => { it('can take symbols for event types', () => { const foo = () => {}; - const eventType = Symbol('eventType'); inst.on(eventType, foo); expect(events.get(eventType)).to.deep.equal([foo]); }); @@ -151,7 +162,7 @@ describe('mitt#', () => { it('should invoke handler for type', () => { const event = { a: 'b' }; - inst.on('foo', (one, two?) => { + inst.on('foo', (one, two?: unknown) => { expect(one).to.deep.equal(event); expect(two).to.be.an('undefined'); }); diff --git a/test/test-types-compilation.ts b/test/test-types-compilation.ts index 00510da..5bf81d6 100644 --- a/test/test-types-compilation.ts +++ b/test/test-types-compilation.ts @@ -2,42 +2,77 @@ import mitt from '..'; -const emitter = mitt(); +interface SomeEventData { + name: string; +} + +const emitter = mitt<{ + foo: string; + someEvent: SomeEventData; + bar?: number; +}>(); + +const barHandler = (x?: number) => {}; +const fooHandler = (x?: string) => {}; +const wildcardHandler = ( + _type: 'foo' | 'bar' | 'someEvent', + _event: string | SomeEventData | number | undefined +) => {}; /* - * Check that if on is provided a generic, it only accepts handlers of that type + * Check that 'on' args are inferred correctly */ { - const badHandler = (x: number) => {}; - const goodHandler = (x: string) => {}; + // @ts-expect-error + emitter.on('foo', barHandler); + emitter.on('foo', fooHandler); + emitter.on('bar', barHandler); // @ts-expect-error - emitter.on('foo', badHandler); - emitter.on('foo', goodHandler); + emitter.on('bar', fooHandler); + + emitter.on('*', wildcardHandler); + // fooHandler is ok, because ('foo' | 'bar' | 'someEvent') extends string + emitter.on('*', fooHandler); + // @ts-expect-error + emitter.on('*', barHandler); } /* - * Check that if off is provided a generic, it only accepts handlers of that type + * Check that 'off' args are inferred correctly */ { - const badHandler = (x: number) => {}; - const goodHandler = (x: string) => {}; + // @ts-expect-error + emitter.off('foo', barHandler); + emitter.off('foo', fooHandler); + emitter.off('bar', barHandler); // @ts-expect-error - emitter.off('foo', badHandler); - emitter.off('foo', goodHandler); -} + emitter.off('bar', fooHandler); + emitter.off('*', wildcardHandler); + // fooHandler is ok, because ('foo' | 'bar' | 'someEvent') extends string + emitter.off('*', fooHandler); + // @ts-expect-error + emitter.off('*', barHandler); +} /* - * Check that if emitt is provided a generic, it only accepts event data of that type + * Check that 'emit' args are inferred correctly */ { - interface SomeEventData { - name: string; - } - // @ts-expect-error - emitter.emit('foo', 'NOT VALID'); - emitter.emit('foo', { name: 'jack' }); -} + // @ts-expect-error + emitter.emit('someEvent', 'NOT VALID'); + emitter.emit('someEvent', { name: 'jack' }); + + // @ts-expect-error + emitter.emit('foo'); + // @ts-expect-error + emitter.emit('foo', 1); + emitter.emit('foo', 'string'); + emitter.emit('bar'); + emitter.emit('bar', 1); + // @ts-expect-error + emitter.emit('bar', 'string'); +} diff --git a/tsconfig.json b/tsconfig.json index acab4f5..563d58e 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -1,13 +1,11 @@ { "compileOnSave": false, "compilerOptions": { + "strict": true, "noEmit": true, "declaration": true, "moduleResolution": "node", "esModuleInterop": true }, - "include": [ - "src/*.ts", - "test/*.ts", - ] + "include": ["src/*.ts", "test/*.ts"] } From b0e10ba71b4be35730a46d5e9b3f5da615d4cbf9 Mon Sep 17 00:00:00 2001 From: iyegoroff Date: Sat, 1 Aug 2020 15:23:39 +0300 Subject: [PATCH 2/6] upd readme --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index f81166e..5ad8ce6 100644 --- a/README.md +++ b/README.md @@ -122,7 +122,7 @@ const emitter: mitt.Emitter = mitt(); Mitt: Tiny (~200b) functional event emitter / pubsub. -Returns **Mitt** +Returns **Mitt** ### all From 3cee15343decb169d09a819c1861106370561097 Mon Sep 17 00:00:00 2001 From: iyegoroff Date: Sat, 1 Aug 2020 15:26:39 +0300 Subject: [PATCH 3/6] upd readme --- README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 5ad8ce6..2cfddeb 100644 --- a/README.md +++ b/README.md @@ -80,7 +80,7 @@ emitter.off('foo', onFoo) // unlisten ### Typescript -Set `"strict": true` in your tsconfig.json to get improved type inferrence for `mitt` instance methods. +Set `"strict": true` in your tsconfig.json to get improved type inference for `mitt` instance methods. ```ts import mitt from 'mitt'; @@ -122,7 +122,7 @@ const emitter: mitt.Emitter = mitt(); Mitt: Tiny (~200b) functional event emitter / pubsub. -Returns **Mitt** +Returns **Mitt** ### all From cda7eb71141412d018554c8b3fe0138c02595ad7 Mon Sep 17 00:00:00 2001 From: iyegoroff Date: Sat, 1 Aug 2020 15:29:45 +0300 Subject: [PATCH 4/6] upd test & readme --- README.md | 2 +- test/test-types-compilation.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 2cfddeb..0fe8dc4 100644 --- a/README.md +++ b/README.md @@ -122,7 +122,7 @@ const emitter: mitt.Emitter = mitt(); Mitt: Tiny (~200b) functional event emitter / pubsub. -Returns **Mitt** +Returns **Mitt** ### all diff --git a/test/test-types-compilation.ts b/test/test-types-compilation.ts index 5bf81d6..476e631 100644 --- a/test/test-types-compilation.ts +++ b/test/test-types-compilation.ts @@ -13,7 +13,7 @@ const emitter = mitt<{ }>(); const barHandler = (x?: number) => {}; -const fooHandler = (x?: string) => {}; +const fooHandler = (x: string) => {}; const wildcardHandler = ( _type: 'foo' | 'bar' | 'someEvent', _event: string | SomeEventData | number | undefined From 4b6aa9d5c7bcbc813b109f4f72698615babb8d6c Mon Sep 17 00:00:00 2001 From: iyegoroff Date: Sat, 1 Aug 2020 17:36:51 +0300 Subject: [PATCH 5/6] removed magic 1 --- src/index.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/index.ts b/src/index.ts index 2293eb3..9940084 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,4 +1,4 @@ -import { Union } from 'ts-toolbelt'; +import { Union, B } from 'ts-toolbelt'; export type EventType = string | symbol; @@ -31,7 +31,7 @@ export interface Emitter> { emit(type: Key, event: Events[Key]): void; emit( - type: Union.Has extends 1 ? Key : never + type: Union.Has extends B.True ? Key : never ): void; } From d994e0c867d8141be707e8ad7548966b08362b22 Mon Sep 17 00:00:00 2001 From: iyegoroff Date: Tue, 8 Sep 2020 23:21:48 +0300 Subject: [PATCH 6/6] removed ts-toolbelt dependency --- package.json | 3 --- src/index.ts | 6 +----- 2 files changed, 1 insertion(+), 8 deletions(-) diff --git a/package.json b/package.json index 55dd6c5..b34f8e9 100644 --- a/package.json +++ b/package.json @@ -106,8 +106,5 @@ "sinon-chai": "^3.5.0", "ts-node": "^8.10.2", "typescript": "^3.9.7" - }, - "dependencies": { - "ts-toolbelt": "^6.13.37" } } diff --git a/src/index.ts b/src/index.ts index 9940084..00286bc 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,5 +1,3 @@ -import { Union, B } from 'ts-toolbelt'; - export type EventType = string | symbol; // An event handler can take an optional event argument @@ -30,9 +28,7 @@ export interface Emitter> { off(type: '*', handler: WildcardHandler): void; emit(type: Key, event: Events[Key]): void; - emit( - type: Union.Has extends B.True ? Key : never - ): void; + emit(type: undefined extends Events[Key] ? Key : never): void; } /**