From bff7248de4024ff67081a6cde87cf902c58c9a71 Mon Sep 17 00:00:00 2001 From: Maxime Robert Date: Fri, 2 Feb 2024 10:52:42 +0100 Subject: [PATCH 1/3] feat: make the ngOnChanges value type safe --- .../src/lib/ngx-observable-lifecycle.ts | 59 +++++++++++++++---- 1 file changed, 46 insertions(+), 13 deletions(-) diff --git a/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.ts b/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.ts index d6a57c1..33f7967 100644 --- a/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.ts +++ b/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.ts @@ -26,24 +26,55 @@ export type LifecycleHookKey = keyof AllHooks; type AllHookOptions = Record; type DecorateHookOptions = Partial; +export interface TypedSimpleChange { + previousValue: Data; + currentValue: Data; + firstChange: boolean; +} + +/** + * FIRST POINT: + * the key is made optional because an ngOnChanges will only give keys of inputs that have changed + * SECOND POINT: + * the value is associated with `| null` as if an input value is defined but actually retrieved with + * an `async` pipe, we'll initially get a `null` value + * + * For both point, feel free to check the following stackblitz that demo this + * https://stackblitz.com/edit/stackblitz-starters-s5uphw?file=src%2Fmain.ts + */ +export type TypedSimpleChanges = { + [Key in Keys]?: TypedSimpleChange | null; +}; + // none of the hooks have arguments, EXCEPT ngOnChanges which we need to handle differently -export type DecoratedHooks = Record, Observable> & { - ngOnChanges: Observable[0]>; +export type DecoratedHooks = Record< + Exclude, + Observable +> & { + ngOnChanges: Observable>; }; -export type DecoratedHooksSub = { - [k in keyof DecoratedHooks]: DecoratedHooks[k] extends Observable ? Subject : never; +export type DecoratedHooksSub = { + [k in keyof DecoratedHooks]: DecoratedHooks[k] extends Observable + ? Subject + : never; }; -type PatchedComponentInstance = Pick & { - [hookSubject]: Pick; +type PatchedComponentInstance = Pick< + AllHooks, + Hooks +> & { + [hookSubject]: Pick, Hooks>; constructor: { prototype: { - [hooksPatched]: Pick; + [hooksPatched]: Pick; }; }; }; -function getSubjectForHook(componentInstance: PatchedComponentInstance, hook: LifecycleHookKey): Subject { +function getSubjectForHook( + componentInstance: PatchedComponentInstance, + hook: LifecycleHookKey, +): Subject { if (!componentInstance[hookSubject]) { componentInstance[hookSubject] = {}; } @@ -71,7 +102,7 @@ function getSubjectForHook(componentInstance: PatchedComponentInstance, hoo }; const originalOnDestroy = proto.ngOnDestroy; - proto.ngOnDestroy = function (this: PatchedComponentInstance) { + proto.ngOnDestroy = function (this: PatchedComponentInstance) { originalOnDestroy?.call(this); this[hookSubject]?.[hook]?.complete(); delete this[hookSubject]?.[hook]; @@ -87,10 +118,12 @@ function getSubjectForHook(componentInstance: PatchedComponentInstance, hoo /** * Library authors should use this to create their own lifecycle-aware functionality */ -export function getObservableLifecycle(classInstance: any): DecoratedHooks { - return new Proxy({} as DecoratedHooks, { - get(target: DecoratedHooks, p: LifecycleHookKey): Observable { - return getSubjectForHook(classInstance, p).asObservable(); +export function getObservableLifecycle( + classInstance: Component, +): DecoratedHooks { + return new Proxy({} as DecoratedHooks, { + get(target: DecoratedHooks, p: LifecycleHookKey): Observable { + return getSubjectForHook(classInstance as unknown as PatchedComponentInstance, p).asObservable(); }, }); } From 09ec4c71411f9aa6a7942003a96e7ff0e35b11a1 Mon Sep 17 00:00:00 2001 From: Maxime Robert Date: Fri, 2 Feb 2024 11:04:51 +0100 Subject: [PATCH 2/3] refactor: simplify the inner part of the API that deals with ngOnChanges typings --- .../src/lib/ngx-observable-lifecycle.ts | 24 +++++++------------ 1 file changed, 8 insertions(+), 16 deletions(-) diff --git a/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.ts b/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.ts index 33f7967..9f27b0b 100644 --- a/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.ts +++ b/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.ts @@ -47,23 +47,18 @@ export type TypedSimpleChanges = { }; // none of the hooks have arguments, EXCEPT ngOnChanges which we need to handle differently -export type DecoratedHooks = Record< +export type DecoratedHooks = Record< Exclude, Observable > & { ngOnChanges: Observable>; }; -export type DecoratedHooksSub = { - [k in keyof DecoratedHooks]: DecoratedHooks[k] extends Observable - ? Subject - : never; +export type DecoratedHooksSub = { + [k in keyof DecoratedHooks]: DecoratedHooks[k] extends Observable ? Subject : never; }; -type PatchedComponentInstance = Pick< - AllHooks, - Hooks -> & { - [hookSubject]: Pick, Hooks>; +type PatchedComponentInstance = Pick & { + [hookSubject]: Pick; constructor: { prototype: { [hooksPatched]: Pick; @@ -71,10 +66,7 @@ type PatchedComponentInstance( - componentInstance: PatchedComponentInstance, - hook: LifecycleHookKey, -): Subject { +function getSubjectForHook(componentInstance: PatchedComponentInstance, hook: LifecycleHookKey): Subject { if (!componentInstance[hookSubject]) { componentInstance[hookSubject] = {}; } @@ -102,7 +94,7 @@ function getSubjectForHook( }; const originalOnDestroy = proto.ngOnDestroy; - proto.ngOnDestroy = function (this: PatchedComponentInstance) { + proto.ngOnDestroy = function (this: PatchedComponentInstance) { originalOnDestroy?.call(this); this[hookSubject]?.[hook]?.complete(); delete this[hookSubject]?.[hook]; @@ -123,7 +115,7 @@ export function getObservableLifecycle { return new Proxy({} as DecoratedHooks, { get(target: DecoratedHooks, p: LifecycleHookKey): Observable { - return getSubjectForHook(classInstance as unknown as PatchedComponentInstance, p).asObservable(); + return getSubjectForHook(classInstance as unknown as PatchedComponentInstance, p).asObservable(); }, }); } From 6da92b787f1aadeef268fa01dc7d3d6bfcb82afb Mon Sep 17 00:00:00 2001 From: Maxime Robert Date: Fri, 2 Feb 2024 11:10:59 +0100 Subject: [PATCH 3/3] doc: explain how to use the generics to have ngOnChanges type safe --- README.md | 23 ++++++++++++++++++++--- src/app/child/child.component.ts | 23 ++++++++++++++++++++--- 2 files changed, 40 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 9ab133a..a13482b 100644 --- a/README.md +++ b/README.md @@ -94,7 +94,8 @@ import { getObservableLifecycle } from 'ngx-observable-lifecycle'; changeDetection: ChangeDetectionStrategy.OnPush, }) export class ChildComponent { - @Input() input: number | undefined | null; + @Input() input1: number | undefined | null; + @Input() input2: string | undefined | null; constructor() { const { @@ -106,9 +107,12 @@ export class ChildComponent { ngAfterViewInit, ngAfterViewChecked, ngOnDestroy, - } = getObservableLifecycle(this); + } = + // specifying the generics is only needed if you intend to + // use the `ngOnChanges` observable, this way you'll have + // typed input values instead of just a `SimpleChange` + getObservableLifecycle(this); - ngOnChanges.subscribe(() => console.count('onChanges')); ngOnInit.subscribe(() => console.count('onInit')); ngDoCheck.subscribe(() => console.count('doCheck')); ngAfterContentInit.subscribe(() => console.count('afterContentInit')); @@ -116,6 +120,19 @@ export class ChildComponent { ngAfterViewInit.subscribe(() => console.count('afterViewInit')); ngAfterViewChecked.subscribe(() => console.count('afterViewChecked')); ngOnDestroy.subscribe(() => console.count('onDestroy')); + + ngOnChanges.subscribe(changes => { + console.count('onChanges'); + + // do note that we have a type safe object here for `changes` + // with the inputs from our component and their associated values typed accordingly + + changes.input1?.currentValue; // `number | null | undefined` + changes.input1?.previousValue; // `number | null | undefined` + + changes.input2?.currentValue; // `string | null | undefined` + changes.input2?.previousValue; // `string | null | undefined` + }); } } diff --git a/src/app/child/child.component.ts b/src/app/child/child.component.ts index 49a1934..87ccecd 100644 --- a/src/app/child/child.component.ts +++ b/src/app/child/child.component.ts @@ -7,7 +7,8 @@ import { getObservableLifecycle } from 'ngx-observable-lifecycle'; changeDetection: ChangeDetectionStrategy.OnPush, }) export class ChildComponent { - @Input() input: number | undefined | null; + @Input() input1: number | undefined | null; + @Input() input2: string | undefined | null; constructor() { const { @@ -19,9 +20,12 @@ export class ChildComponent { ngAfterViewInit, ngAfterViewChecked, ngOnDestroy, - } = getObservableLifecycle(this); + } = + // specifying the generics is only needed if you intend to + // use the `ngOnChanges` observable, this way you'll have + // typed input values instead of just a `SimpleChange` + getObservableLifecycle(this); - ngOnChanges.subscribe(() => console.count('onChanges')); ngOnInit.subscribe(() => console.count('onInit')); ngDoCheck.subscribe(() => console.count('doCheck')); ngAfterContentInit.subscribe(() => console.count('afterContentInit')); @@ -29,5 +33,18 @@ export class ChildComponent { ngAfterViewInit.subscribe(() => console.count('afterViewInit')); ngAfterViewChecked.subscribe(() => console.count('afterViewChecked')); ngOnDestroy.subscribe(() => console.count('onDestroy')); + + ngOnChanges.subscribe(changes => { + console.count('onChanges'); + + // do note that we have a type safe object here for `changes` + // with the inputs from our component and their associated values typed accordingly + + changes.input1?.currentValue; // `number | null | undefined` + changes.input1?.previousValue; // `number | null | undefined` + + changes.input2?.currentValue; // `string | null | undefined` + changes.input2?.previousValue; // `string | null | undefined` + }); } }