From f0097d2a9d13079329a17f6576f78c7610ef0a70 Mon Sep 17 00:00:00 2001 From: timdeschryver <28659384+timdeschryver@users.noreply.github.com> Date: Sat, 16 Oct 2021 19:14:46 +0200 Subject: [PATCH 1/2] feat: add rerender method BREAKING CHANGE: rerender has been renamed to change change keeps the current fixture intact and invokes ngOnChanges the new rerender method destroys the current component and creates a new instance with the updated properties BEFORE: const { rerender } = render(...) rerender({...}) AFTER: const { change } = render(...) change({...}) --- .../src/app/issues/issue-222.spec.ts | 18 ++- .../examples/16-input-getter-setter.spec.ts | 19 +++- projects/testing-library/src/lib/models.ts | 8 +- .../src/lib/testing-library.ts | 106 ++++++++++-------- projects/testing-library/tests/change.spec.ts | 85 ++++++++++++++ .../testing-library/tests/rerender.spec.ts | 70 ++++-------- 6 files changed, 206 insertions(+), 100 deletions(-) create mode 100644 projects/testing-library/tests/change.spec.ts diff --git a/apps/example-app-karma/src/app/issues/issue-222.spec.ts b/apps/example-app-karma/src/app/issues/issue-222.spec.ts index 740d8184..b6ac5204 100644 --- a/apps/example-app-karma/src/app/issues/issue-222.spec.ts +++ b/apps/example-app-karma/src/app/issues/issue-222.spec.ts @@ -1,6 +1,6 @@ import { render, screen } from '@testing-library/angular'; -it('https://github.com/testing-library/angular-testing-library/issues/222', async () => { +it('https://github.com/testing-library/angular-testing-library/issues/222 with rerender', async () => { const { rerender } = await render(`
Hello {{ name}}
`, { componentProperties: { name: 'Sarah', @@ -8,7 +8,21 @@ it('https://github.com/testing-library/angular-testing-library/issues/222', asyn }); expect(screen.getByText('Hello Sarah')).toBeTruthy(); - rerender({ name: 'Mark' }); + + await rerender({ name: 'Mark' }); + + expect(screen.getByText('Hello Mark')).toBeTruthy(); +}); + +it('https://github.com/testing-library/angular-testing-library/issues/222 with change', async () => { + const { change } = await render(`
Hello {{ name}}
`, { + componentProperties: { + name: 'Sarah', + }, + }); + + expect(screen.getByText('Hello Sarah')).toBeTruthy(); + await change({ name: 'Mark' }); expect(screen.getByText('Hello Mark')).toBeTruthy(); }); diff --git a/apps/example-app/src/app/examples/16-input-getter-setter.spec.ts b/apps/example-app/src/app/examples/16-input-getter-setter.spec.ts index 28d736c6..1d87edf9 100644 --- a/apps/example-app/src/app/examples/16-input-getter-setter.spec.ts +++ b/apps/example-app/src/app/examples/16-input-getter-setter.spec.ts @@ -10,16 +10,29 @@ test('should run logic in the input setter and getter', async () => { expect(getterValueControl).toHaveTextContent('I am value from getter Angular'); }); -test('should run logic in the input setter and getter while re-rendering', async () => { - const { rerender } = await render(InputGetterSetter, { componentProperties: { value: 'Angular' } }); +test('should run logic in the input setter and getter while changing', async () => { + const { change } = await render(InputGetterSetter, { componentProperties: { value: 'Angular' } }); const valueControl = screen.getByTestId('value'); const getterValueControl = screen.getByTestId('value-getter'); expect(valueControl).toHaveTextContent('I am value from setter Angular'); expect(getterValueControl).toHaveTextContent('I am value from getter Angular'); - await rerender({ value: 'React' }); + await change({ value: 'React' }); expect(valueControl).toHaveTextContent('I am value from setter React'); expect(getterValueControl).toHaveTextContent('I am value from getter React'); }); + +test('should run logic in the input setter and getter while re-rendering', async () => { + const { rerender } = await render(InputGetterSetter, { componentProperties: { value: 'Angular' } }); + + expect(screen.getByTestId('value')).toHaveTextContent('I am value from setter Angular'); + expect(screen.getByTestId('value-getter')).toHaveTextContent('I am value from getter Angular'); + + await rerender({ value: 'React' }); + + // note we have to re-query because the elements are not the same anymore + expect(screen.getByTestId('value')).toHaveTextContent('I am value from setter React'); + expect(screen.getByTestId('value-getter')).toHaveTextContent('I am value from getter React'); +}); diff --git a/projects/testing-library/src/lib/models.ts b/projects/testing-library/src/lib/models.ts index 54b5f2ee..abb72bcd 100644 --- a/projects/testing-library/src/lib/models.ts +++ b/projects/testing-library/src/lib/models.ts @@ -56,7 +56,13 @@ export interface RenderResult extend * @description * Re-render the same component with different props. */ - rerender: (componentProperties: Partial) => void; + rerender: (rerenderedProperties: Partial) => void; + + /** + * @description + * Re-render the component while invoking ngOnChanges. + */ + change: (changedProperties: Partial) => void; } export interface RenderComponentOptions { diff --git a/projects/testing-library/src/lib/testing-library.ts b/projects/testing-library/src/lib/testing-library.ts index 95f28fc7..f5ad7de4 100644 --- a/projects/testing-library/src/lib/testing-library.ts +++ b/projects/testing-library/src/lib/testing-library.ts @@ -93,53 +93,30 @@ export async function render( schemas: [...schemas], }); - if (componentProviders) { - componentProviders - .reduce((acc, provider) => acc.concat(provider), []) - .forEach((p) => { - const { provide, ...provider } = p; - TestBed.overrideProvider(provide, provider); - }); - } - - const fixture = await createComponentFixture(sut, { template, wrapper }); - setComponentProperties(fixture, { componentProperties }); - - if (removeAngularAttributes) { - fixture.nativeElement.removeAttribute('ng-version'); - const idAttribute = fixture.nativeElement.getAttribute('id'); - if (idAttribute && idAttribute.startsWith('root')) { - fixture.nativeElement.removeAttribute('id'); - } - } - - mountedFixtures.add(fixture); - await TestBed.compileComponents(); - let isAlive = true; - fixture.componentRef.onDestroy(() => (isAlive = false)); + componentProviders + .reduce((acc, provider) => acc.concat(provider), []) + .forEach((p) => { + const { provide, ...provider } = p; + TestBed.overrideProvider(provide, provider); + }); - function detectChanges() { - if (isAlive) { - fixture.detectChanges(); - } - } + const componentContainer = createComponentFixture(sut, { template, wrapper }); - // Call ngOnChanges on initial render - if (hasOnChangesHook(fixture.componentInstance)) { - const changes = getChangesObj(null, componentProperties); - fixture.componentInstance.ngOnChanges(changes); - } + let fixture: ComponentFixture; + let detectChanges: () => void; - if (detectChangesOnRender) { - detectChanges(); - } + await renderFixture(componentProperties); + + const rerender = async (rerenderedProperties: Partial) => { + await renderFixture(rerenderedProperties); + }; - const rerender = (rerenderedProperties: Partial) => { - const changes = getChangesObj(fixture.componentInstance, rerenderedProperties); + const change = (changedProperties: Partial) => { + const changes = getChangesObj(fixture.componentInstance, changedProperties); - setComponentProperties(fixture, { componentProperties: rerenderedProperties }); + setComponentProperties(fixture, { componentProperties: changedProperties }); if (hasOnChangesHook(fixture.componentInstance)) { fixture.componentInstance.ngOnChanges(changes); @@ -188,9 +165,10 @@ export async function render( return { fixture, - detectChanges, + detectChanges: () => detectChanges(), navigate, rerender, + change, debugElement: typeof sut === 'string' ? fixture.debugElement : fixture.debugElement.query(By.directive(sut)), container: fixture.nativeElement, debug: (element = fixture.nativeElement, maxLength, options) => @@ -199,6 +177,42 @@ export async function render( : console.log(dtlPrettyDOM(element, maxLength, options)), ...replaceFindWithFindAndDetectChanges(dtlGetQueriesForElement(fixture.nativeElement, queries)), }; + + async function renderFixture(properties: Partial) { + if (fixture) { + cleanupAtFixture(fixture); + } + + fixture = await createComponent(componentContainer); + setComponentProperties(fixture, { componentProperties: properties }); + + if (removeAngularAttributes) { + fixture.nativeElement.removeAttribute('ng-version'); + const idAttribute = fixture.nativeElement.getAttribute('id'); + if (idAttribute && idAttribute.startsWith('root')) { + fixture.nativeElement.removeAttribute('id'); + } + } + mountedFixtures.add(fixture); + + let isAlive = true; + fixture.componentRef.onDestroy(() => (isAlive = false)); + + if (hasOnChangesHook(fixture.componentInstance)) { + const changes = getChangesObj(null, componentProperties); + fixture.componentInstance.ngOnChanges(changes); + } + + detectChanges = () => { + if (isAlive) { + fixture.detectChanges(); + } + }; + + if (detectChangesOnRender) { + detectChanges(); + } + } } async function createComponent(component: Type): Promise> { @@ -207,19 +221,19 @@ async function createComponent(component: Type): Promise( +function createComponentFixture( sut: Type | string, { template, wrapper }: Pick, 'template' | 'wrapper'>, -): Promise> { +): Type { if (typeof sut === 'string') { TestBed.overrideTemplate(wrapper, sut); - return createComponent(wrapper); + return wrapper; } if (template) { TestBed.overrideTemplate(wrapper, template); - return createComponent(wrapper); + return wrapper; } - return createComponent(sut); + return sut; } function setComponentProperties( diff --git a/projects/testing-library/tests/change.spec.ts b/projects/testing-library/tests/change.spec.ts new file mode 100644 index 00000000..85cc3667 --- /dev/null +++ b/projects/testing-library/tests/change.spec.ts @@ -0,0 +1,85 @@ +import { ChangeDetectionStrategy, Component, Input, OnChanges, SimpleChanges } from '@angular/core'; +import { render, screen } from '../src/public_api'; + +@Component({ + selector: 'atl-fixture', + template: ` {{ firstName }} {{ lastName }} `, +}) +class FixtureComponent { + @Input() firstName = 'Sarah'; + @Input() lastName; +} + +test('changes the component with updated props', async () => { + const { change } = await render(FixtureComponent); + expect(screen.getByText('Sarah')).toBeInTheDocument(); + + const firstName = 'Mark'; + change({ firstName }); + + expect(screen.getByText(firstName)).toBeInTheDocument(); +}); + +test('changes the component with updated props while keeping other props untouched', async () => { + const firstName = 'Mark'; + const lastName = 'Peeters'; + const { change } = await render(FixtureComponent, { + componentProperties: { + firstName, + lastName, + }, + }); + + expect(screen.getByText(`${firstName} ${lastName}`)).toBeInTheDocument(); + + const firstName2 = 'Chris'; + change({ firstName: firstName2 }); + + expect(screen.getByText(`${firstName2} ${lastName}`)).toBeInTheDocument(); +}); + +@Component({ + selector: 'atl-fixture', + template: ` {{ name }} `, +}) +class FixtureWithNgOnChangesComponent implements OnChanges { + @Input() name = 'Sarah'; + @Input() nameChanged: (name: string, isFirstChange: boolean) => void; + + ngOnChanges(changes: SimpleChanges) { + if (changes.name && this.nameChanged) { + this.nameChanged(changes.name.currentValue, changes.name.isFirstChange()); + } + } +} + +test('will call ngOnChanges on change', async () => { + const nameChanged = jest.fn(); + const componentProperties = { nameChanged }; + const { change } = await render(FixtureWithNgOnChangesComponent, { componentProperties }); + expect(screen.getByText('Sarah')).toBeInTheDocument(); + + const name = 'Mark'; + change({ name }); + + expect(screen.getByText(name)).toBeInTheDocument(); + expect(nameChanged).toHaveBeenCalledWith(name, false); +}); + +@Component({ + changeDetection: ChangeDetectionStrategy.OnPush, + selector: 'atl-fixture', + template: `
Number
`, +}) +class FixtureWithOnPushComponent { + @Input() activeField: string; +} + +test('update properties on change', async () => { + const { change } = await render(FixtureWithOnPushComponent); + const numberHtmlElementRef = screen.queryByTestId('number'); + + expect(numberHtmlElementRef).not.toHaveClass('active'); + change({ activeField: 'number' }); + expect(numberHtmlElementRef).toHaveClass('active'); +}); diff --git a/projects/testing-library/tests/rerender.spec.ts b/projects/testing-library/tests/rerender.spec.ts index c98bd1bc..443560a2 100644 --- a/projects/testing-library/tests/rerender.spec.ts +++ b/projects/testing-library/tests/rerender.spec.ts @@ -1,66 +1,40 @@ -import { ChangeDetectionStrategy, Component, Input, OnChanges, SimpleChanges } from '@angular/core'; +import { Component, Input } from '@angular/core'; import { render, screen } from '../src/public_api'; @Component({ selector: 'atl-fixture', - template: ` {{ name }} `, + template: ` {{ firstName }} {{ lastName }} `, }) class FixtureComponent { - @Input() name = 'Sarah'; + @Input() firstName = 'Sarah'; + @Input() lastName; } -test('will rerender the component with updated props', async () => { +test('rerenders the component with updated props', async () => { const { rerender } = await render(FixtureComponent); expect(screen.getByText('Sarah')).toBeInTheDocument(); - const name = 'Mark'; - rerender({ name }); + const firstName = 'Mark'; + await rerender({ firstName }); - expect(screen.getByText(name)).toBeInTheDocument(); + expect(screen.getByText(firstName)).toBeInTheDocument(); }); -@Component({ - selector: 'atl-fixture', - template: ` {{ name }} `, -}) -class FixtureWithNgOnChangesComponent implements OnChanges { - @Input() name = 'Sarah'; - @Input() nameChanged: (name: string, isFirstChange: boolean) => void; - - ngOnChanges(changes: SimpleChanges) { - if (changes.name && this.nameChanged) { - this.nameChanged(changes.name.currentValue, changes.name.isFirstChange()); - } - } -} - -test('will call ngOnChanges on rerender', async () => { - const nameChanged = jest.fn(); - const componentProperties = { nameChanged }; - const { rerender } = await render(FixtureWithNgOnChangesComponent, { componentProperties }); - expect(screen.getByText('Sarah')).toBeInTheDocument(); +test('rerenders the component with updated props and resets other props', async () => { + const firstName = 'Mark'; + const lastName = 'Peeters'; + const { rerender } = await render(FixtureComponent, { + componentProperties: { + firstName, + lastName, + }, + }); - const name = 'Mark'; - rerender({ name }); - - expect(screen.getByText(name)).toBeInTheDocument(); - expect(nameChanged).toHaveBeenCalledWith(name, false); -}); - -@Component({ - changeDetection: ChangeDetectionStrategy.OnPush, - selector: 'atl-fixture', - template: `
Number
`, -}) -class FixtureWithOnPushComponent { - @Input() activeField: string; -} + expect(screen.getByText(`${firstName} ${lastName}`)).toBeInTheDocument(); -test('update properties on rerender', async () => { - const { rerender } = await render(FixtureWithOnPushComponent); - const numberHtmlElementRef = screen.queryByTestId('number'); + const firstName2 = 'Chris'; + rerender({ firstName: firstName2 }); - expect(numberHtmlElementRef).not.toHaveClass('active'); - rerender({ activeField: 'number' }); - expect(numberHtmlElementRef).toHaveClass('active'); + expect(screen.queryByText(`${firstName2} ${lastName}`)).not.toBeInTheDocument(); + expect(screen.queryByText(firstName2)).not.toBeInTheDocument(); }); From ed8c348fdb23d36df153189f1db7b37aeb31826d Mon Sep 17 00:00:00 2001 From: Tim Deschryver <28659384+timdeschryver@users.noreply.github.com> Date: Wed, 24 Nov 2021 21:32:36 +0100 Subject: [PATCH 2/2] Update models.ts --- projects/testing-library/src/lib/models.ts | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/projects/testing-library/src/lib/models.ts b/projects/testing-library/src/lib/models.ts index abb72bcd..bda39543 100644 --- a/projects/testing-library/src/lib/models.ts +++ b/projects/testing-library/src/lib/models.ts @@ -54,13 +54,14 @@ export interface RenderResult extend navigate: (elementOrPath: Element | string, basePath?: string) => Promise; /** * @description - * Re-render the same component with different props. + * Re-render the same component with different properties. + * This creates a new instance of the component. */ rerender: (rerenderedProperties: Partial) => void; /** * @description - * Re-render the component while invoking ngOnChanges. + * Keeps the current fixture intact and invokes ngOnChanges with the updated properties. */ change: (changedProperties: Partial) => void; }