From 492f01301108b427796fe200f26e9ed4966571b9 Mon Sep 17 00:00:00 2001 From: Maxime Robert Date: Sun, 4 Feb 2024 11:28:04 +0100 Subject: [PATCH] fix: ensure the definition of the hook ngOnChanges if the corresponding observable is used. See https://stackoverflow.com/a/77930589/2398593 BREAKING CHANGE: Make sure to define the ngOnChanges hook on your component if you use the ngOnChanges observable given by the library otherwise it'll throw a runtime error. More info about why here https://stackoverflow.com/a/77930589/2398593 --- README.md | 9 +++- .../src/lib/ngx-observable-lifecycle.spec.ts | 43 ++++++++++++++++++- .../src/lib/ngx-observable-lifecycle.ts | 6 +++ src/app/child/child.component.ts | 9 +++- 4 files changed, 62 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index 6458ebd..0e89183 100644 --- a/README.md +++ b/README.md @@ -85,7 +85,7 @@ Here's an example component that hooks onto the full set of available hooks. ```ts // ./src/app/child/child.component.ts -import { ChangeDetectionStrategy, Component, Input } from '@angular/core'; +import { ChangeDetectionStrategy, Component, Input, OnChanges } from '@angular/core'; import { getObservableLifecycle } from 'ngx-observable-lifecycle'; @Component({ @@ -93,7 +93,7 @@ import { getObservableLifecycle } from 'ngx-observable-lifecycle'; templateUrl: './child.component.html', changeDetection: ChangeDetectionStrategy.OnPush, }) -export class ChildComponent { +export class ChildComponent implements OnChanges { @Input() input1: number | undefined | null; @Input() input2: string | undefined | null; @@ -134,6 +134,11 @@ export class ChildComponent { changes.input2?.previousValue; // `string | null | undefined` }); } + + // when using the ngOnChanges hook, you have to define the hook in your class even if it's empty + // See https://stackoverflow.com/a/77930589/2398593 for more info + // eslint-disable-next-line @angular-eslint/no-empty-lifecycle-method + public ngOnChanges() {} } ``` diff --git a/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.spec.ts b/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.spec.ts index d8bb5cf..bb7a171 100644 --- a/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.spec.ts +++ b/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.spec.ts @@ -1,4 +1,4 @@ -import { Component, OnDestroy, OnInit } from '@angular/core'; +import { Component, OnChanges, OnDestroy, OnInit } from '@angular/core'; import { isObservable } from 'rxjs'; import { getObservableLifecycle } from './ngx-observable-lifecycle'; @@ -106,5 +106,46 @@ describe('ngx-observable-lifecycle', () => { expect(originalOnDestroySpy).toHaveBeenCalled(); }); + + // see https://stackoverflow.com/a/77930589/2398593 + it(`should throw if ngOnChanges isn't defined in the component if ngOnChanges observable is used`, () => { + class LocalTestComponent { + constructor() { + // all except ngOnChanges + // even without having the original `ngOnChanges` hook it should be ok + const { + ngOnInit, + ngDoCheck, + ngAfterContentInit, + ngAfterContentChecked, + ngAfterViewInit, + ngAfterViewChecked, + ngOnDestroy, + } = getObservableLifecycle(this); + } + } + + expect(() => new LocalTestComponent()).not.toThrowError(); + + class LocalTestWithNgOnChangesNoOriginalHookComponent { + constructor() { + // without having the original `ngOnChanges` hook it should throw + const { ngOnChanges } = getObservableLifecycle(this); + } + } + expect(() => new LocalTestWithNgOnChangesNoOriginalHookComponent()).toThrowError( + `When using the ngOnChanges hook, you have to define the hook in your class even if it's empty. See https://stackoverflow.com/a/77930589/2398593`, + ); + + class LocalTestWithNgOnChangesAndOriginalHookComponent implements OnChanges { + constructor() { + // when we have the original `ngOnChanges` hook it should **not** throw + const { ngOnChanges } = getObservableLifecycle(this); + } + + public ngOnChanges(): void {} + } + expect(() => new LocalTestWithNgOnChangesAndOriginalHookComponent()).not.toThrowError(); + }); }); }); 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 9f27b0b..d944cc3 100644 --- a/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.ts +++ b/projects/ngx-observable-lifecycle/src/lib/ngx-observable-lifecycle.ts @@ -67,6 +67,12 @@ type PatchedComponentInstance = Pick { + if (hook === 'ngOnChanges' && !componentInstance.constructor.prototype[hook]) { + throw new Error( + `When using the ngOnChanges hook, you have to define the hook in your class even if it's empty. See https://stackoverflow.com/a/77930589/2398593`, + ); + } + if (!componentInstance[hookSubject]) { componentInstance[hookSubject] = {}; } diff --git a/src/app/child/child.component.ts b/src/app/child/child.component.ts index 87ccecd..fe6d6fc 100644 --- a/src/app/child/child.component.ts +++ b/src/app/child/child.component.ts @@ -1,4 +1,4 @@ -import { ChangeDetectionStrategy, Component, Input } from '@angular/core'; +import { ChangeDetectionStrategy, Component, Input, OnChanges } from '@angular/core'; import { getObservableLifecycle } from 'ngx-observable-lifecycle'; @Component({ @@ -6,7 +6,7 @@ import { getObservableLifecycle } from 'ngx-observable-lifecycle'; templateUrl: './child.component.html', changeDetection: ChangeDetectionStrategy.OnPush, }) -export class ChildComponent { +export class ChildComponent implements OnChanges { @Input() input1: number | undefined | null; @Input() input2: string | undefined | null; @@ -47,4 +47,9 @@ export class ChildComponent { changes.input2?.previousValue; // `string | null | undefined` }); } + + // when using the ngOnChanges hook, you have to define the hook in your class even if it's empty + // See https://stackoverflow.com/a/77930589/2398593 for more info + // eslint-disable-next-line @angular-eslint/no-empty-lifecycle-method + public ngOnChanges() {} }