fix(elements): update the view of an `OnPush` component when inputs change (#39452)

As with regular Angular components, Angular elements are expected to
have their views update when inputs change.

Previously, Angular Elements views were not updated if the underlying
component used the `OnPush` change detection strategy.

This commit fixes this by calling `markForCheck()` on the component
view's `ChangeDetectorRef`.

NOTE:
This is similar to how `@angular/upgrade` does it:
3236ae0ee1/packages/upgrade/src/common/src/downgrade_component_adapter.ts (L146).

Fixes #38948

PR Close #39452
This commit is contained in:
George Kalpakas 2020-11-04 20:46:59 +02:00 committed by Misko Hevery
parent c1907809a8
commit bdce7698fc
8 changed files with 190 additions and 25 deletions

View File

@ -12,7 +12,7 @@
"master": { "master": {
"uncompressed": { "uncompressed": {
"runtime-es2015": 3037, "runtime-es2015": 3037,
"main-es2015": 448615, "main-es2015": 448922,
"polyfills-es2015": 52415 "polyfills-es2015": 52415
} }
} }
@ -21,7 +21,7 @@
"master": { "master": {
"uncompressed": { "uncompressed": {
"runtime-es2015": 3157, "runtime-es2015": 3157,
"main-es2015": 431750, "main-es2015": 432199,
"polyfills-es2015": 52415 "polyfills-es2015": 52415
} }
} }

View File

@ -5,7 +5,7 @@ describe('Element E2E Tests', function () {
describe('Hello World Elements', () => { describe('Hello World Elements', () => {
beforeEach(() => browser.get('hello-world.html')); beforeEach(() => browser.get('hello-world.html'));
describe('(with default view encapsulation)', () => { describe('(with default CD strategy and view encapsulation)', () => {
const helloWorldEl = element(by.css('hello-world-el')); const helloWorldEl = element(by.css('hello-world-el'));
it('should display "Hello World!"', function () { it('should display "Hello World!"', function () {
@ -21,6 +21,22 @@ describe('Element E2E Tests', function () {
}); });
}); });
describe('(with `OnPush` CD strategy)', () => {
const helloWorldOnpushEl = element(by.css('hello-world-onpush-el'));
it('should display "Hello World!"', function () {
expect(helloWorldOnpushEl.getText()).toBe('Hello World!');
});
it('should display "Hello Foo!" via name attribute', function () {
const input = element(by.css('input[type=text]'));
input.sendKeys('Foo');
// Make tests less flaky on CI by waiting up to 5s for the element text to be updated.
browser.wait(EC.textToBePresentInElement(helloWorldOnpushEl, 'Hello Foo!'), 5000);
});
});
describe('(with `ShadowDom` view encapsulation)', () => { describe('(with `ShadowDom` view encapsulation)', () => {
const helloWorldShadowEl = element(by.css('hello-world-shadow-el')); const helloWorldShadowEl = element(by.css('hello-world-shadow-el'));
const getShadowDomText = (el: ElementFinder) => const getShadowDomText = (el: ElementFinder) =>

View File

@ -2,17 +2,29 @@ import {Injector, NgModule} from '@angular/core';
import {createCustomElement} from '@angular/elements'; import {createCustomElement} from '@angular/elements';
import {BrowserModule} from '@angular/platform-browser'; import {BrowserModule} from '@angular/platform-browser';
import {HelloWorldComponent, HelloWorldShadowComponent, TestCardComponent} from './elements'; import {HelloWorldComponent, HelloWorldOnpushComponent, HelloWorldShadowComponent, TestCardComponent} from './elements';
@NgModule({ @NgModule({
declarations: [HelloWorldComponent, HelloWorldShadowComponent, TestCardComponent], declarations: [
entryComponents: [HelloWorldComponent, HelloWorldShadowComponent, TestCardComponent], HelloWorldComponent,
HelloWorldOnpushComponent,
HelloWorldShadowComponent,
TestCardComponent,
],
entryComponents: [
HelloWorldComponent,
HelloWorldOnpushComponent,
HelloWorldShadowComponent,
TestCardComponent,
],
imports: [BrowserModule], imports: [BrowserModule],
}) })
export class AppModule { export class AppModule {
constructor(injector: Injector) { constructor(injector: Injector) {
customElements.define('hello-world-el', createCustomElement(HelloWorldComponent, {injector})); customElements.define('hello-world-el', createCustomElement(HelloWorldComponent, {injector}));
customElements.define(
'hello-world-onpush-el', createCustomElement(HelloWorldOnpushComponent, {injector}));
customElements.define( customElements.define(
'hello-world-shadow-el', createCustomElement(HelloWorldShadowComponent, {injector})); 'hello-world-shadow-el', createCustomElement(HelloWorldShadowComponent, {injector}));
customElements.define('test-card', createCustomElement(TestCardComponent, {injector})); customElements.define('test-card', createCustomElement(TestCardComponent, {injector}));

View File

@ -1,4 +1,4 @@
import {Component, Input, ViewEncapsulation} from '@angular/core'; import {ChangeDetectionStrategy, Component, Input, ViewEncapsulation} from '@angular/core';
@Component({ @Component({
selector: 'hello-world-el', selector: 'hello-world-el',
@ -8,6 +8,15 @@ export class HelloWorldComponent {
@Input() name: string = 'World'; @Input() name: string = 'World';
} }
@Component({
selector: 'hello-world-onpush-el',
template: 'Hello {{name}}!',
changeDetection: ChangeDetectionStrategy.OnPush,
})
export class HelloWorldOnpushComponent {
@Input() name: string = 'World';
}
@Component({ @Component({
selector: 'hello-world-shadow-el', selector: 'hello-world-shadow-el',
template: 'Hello {{name}}!', template: 'Hello {{name}}!',

View File

@ -10,6 +10,7 @@
<body> <body>
<input type="text"> <input type="text">
<hello-world-el></hello-world-el> <hello-world-el></hello-world-el>
<hello-world-onpush-el></hello-world-onpush-el>
<hello-world-shadow-el></hello-world-shadow-el> <hello-world-shadow-el></hello-world-shadow-el>
<script src="dist/bundle.js"></script> <script src="dist/bundle.js"></script>
</body> </body>

View File

@ -5,9 +5,11 @@ platformBrowser().bootstrapModuleFactory(AppModuleNgFactory, {ngZone: 'noop'});
const input = document.querySelector('input')!; const input = document.querySelector('input')!;
const helloWorld = document.querySelector('hello-world-el')!; const helloWorld = document.querySelector('hello-world-el')!;
const helloWorldOnpush = document.querySelector('hello-world-onpush-el')!;
const helloWorldShadow = document.querySelector('hello-world-shadow-el')!; const helloWorldShadow = document.querySelector('hello-world-shadow-el')!;
input.addEventListener('input', () => { input.addEventListener('input', () => {
helloWorld.setAttribute('name', input.value); helloWorld.setAttribute('name', input.value);
helloWorldOnpush.setAttribute('name', input.value);
helloWorldShadow.setAttribute('name', input.value); helloWorldShadow.setAttribute('name', input.value);
}); });

View File

@ -6,7 +6,7 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {ApplicationRef, ComponentFactory, ComponentFactoryResolver, ComponentRef, EventEmitter, Injector, NgZone, OnChanges, SimpleChange, SimpleChanges, Type} from '@angular/core'; import {ApplicationRef, ChangeDetectorRef, ComponentFactory, ComponentFactoryResolver, ComponentRef, EventEmitter, Injector, NgZone, OnChanges, SimpleChange, SimpleChanges, Type} from '@angular/core';
import {merge, Observable, ReplaySubject} from 'rxjs'; import {merge, Observable, ReplaySubject} from 'rxjs';
import {map, switchMap} from 'rxjs/operators'; import {map, switchMap} from 'rxjs/operators';
@ -52,10 +52,19 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
/** Reference to the component that was created on connect. */ /** Reference to the component that was created on connect. */
private componentRef: ComponentRef<any>|null = null; private componentRef: ComponentRef<any>|null = null;
/** Changes that have been made to the component ref since the last time onChanges was called. */ /** Reference to the component view's `ChangeDetectorRef`. */
private viewChangeDetectorRef: ChangeDetectorRef|null = null;
/**
* Changes that have been made to component inputs since the last change detection run.
* (NOTE: These are only recorded if the component implements the `OnChanges` interface.)
*/
private inputChanges: SimpleChanges|null = null; private inputChanges: SimpleChanges|null = null;
/** Whether the created component implements the onChanges function. */ /** Whether changes have been made to component inputs since the last change detection run. */
private hasInputChanges = false;
/** Whether the created component implements the `OnChanges` interface. */
private implementsOnChanges = false; private implementsOnChanges = false;
/** Whether a change detection has been scheduled to run on the component. */ /** Whether a change detection has been scheduled to run on the component. */
@ -68,10 +77,12 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
private readonly initialInputValues = new Map<string, any>(); private readonly initialInputValues = new Map<string, any>();
/** /**
* Set of component inputs that have not yet changed, i.e. for which `ngOnChanges()` has not * Set of component inputs that have not yet changed, i.e. for which `recordInputChange()` has not
* fired. (This is used to determine the value of `fistChange` in `SimpleChange` instances.) * fired.
* (This helps detect the first change of an input, even if it is explicitly set to `undefined`.)
*/ */
private readonly unchangedInputs = new Set<string>(); private readonly unchangedInputs =
new Set<string>(this.componentFactory.inputs.map(({propName}) => propName));
/** Service for setting zone context. */ /** Service for setting zone context. */
private readonly ngZone = this.injector.get<NgZone>(NgZone); private readonly ngZone = this.injector.get<NgZone>(NgZone);
@ -119,6 +130,7 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
if (this.componentRef !== null) { if (this.componentRef !== null) {
this.componentRef.destroy(); this.componentRef.destroy();
this.componentRef = null; this.componentRef = null;
this.viewChangeDetectorRef = null;
} }
}, DESTROY_DELAY); }, DESTROY_DELAY);
}); });
@ -157,7 +169,13 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
return; return;
} }
// Record the changed value and update internal state to reflect the fact that this input has
// changed.
this.recordInputChange(property, value); this.recordInputChange(property, value);
this.unchangedInputs.delete(property);
this.hasInputChanges = true;
// Update the component instance and schedule change detection.
this.componentRef.instance[property] = value; this.componentRef.instance[property] = value;
this.scheduleDetectChanges(); this.scheduleDetectChanges();
}); });
@ -172,6 +190,7 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
const projectableNodes = const projectableNodes =
extractProjectableNodes(element, this.componentFactory.ngContentSelectors); extractProjectableNodes(element, this.componentFactory.ngContentSelectors);
this.componentRef = this.componentFactory.create(childInjector, projectableNodes, element); this.componentRef = this.componentFactory.create(childInjector, projectableNodes, element);
this.viewChangeDetectorRef = this.componentRef.injector.get(ChangeDetectorRef);
this.implementsOnChanges = isFunction((this.componentRef.instance as OnChanges).ngOnChanges); this.implementsOnChanges = isFunction((this.componentRef.instance as OnChanges).ngOnChanges);
@ -187,12 +206,6 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
/** Set any stored initial inputs on the component's properties. */ /** Set any stored initial inputs on the component's properties. */
protected initializeInputs(): void { protected initializeInputs(): void {
this.componentFactory.inputs.forEach(({propName}) => { this.componentFactory.inputs.forEach(({propName}) => {
if (this.implementsOnChanges) {
// If the component implements `ngOnChanges()`, keep track of which inputs have never
// changed so far.
this.unchangedInputs.add(propName);
}
if (this.initialInputValues.has(propName)) { if (this.initialInputValues.has(propName)) {
// Call `setInputValue()` now that the component has been instantiated to update its // Call `setInputValue()` now that the component has been instantiated to update its
// properties and fire `ngOnChanges()`. // properties and fire `ngOnChanges()`.
@ -227,6 +240,17 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
(componentRef.instance as OnChanges).ngOnChanges(inputChanges); (componentRef.instance as OnChanges).ngOnChanges(inputChanges);
} }
/**
* Marks the component view for check, if necessary.
* (NOTE: This is required when the `ChangeDetectionStrategy` is set to `OnPush`.)
*/
protected markViewForCheck(viewChangeDetectorRef: ChangeDetectorRef): void {
if (this.hasInputChanges) {
this.hasInputChanges = false;
viewChangeDetectorRef.markForCheck();
}
}
/** /**
* Schedules change detection to run on the component. * Schedules change detection to run on the component.
* Ignores subsequent calls if already scheduled. * Ignores subsequent calls if already scheduled.
@ -247,8 +271,7 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
*/ */
protected recordInputChange(property: string, currentValue: any): void { protected recordInputChange(property: string, currentValue: any): void {
// Do not record the change if the component does not implement `OnChanges`. // Do not record the change if the component does not implement `OnChanges`.
// (We can only determine that after the component has been instantiated.) if (!this.implementsOnChanges) {
if (this.componentRef !== null && !this.implementsOnChanges) {
return; return;
} }
@ -257,7 +280,7 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
} }
// If there already is a change, modify the current value to match but leave the values for // If there already is a change, modify the current value to match but leave the values for
// previousValue and isFirstChange. // `previousValue` and `isFirstChange`.
const pendingChange = this.inputChanges[property]; const pendingChange = this.inputChanges[property];
if (pendingChange) { if (pendingChange) {
pendingChange.currentValue = currentValue; pendingChange.currentValue = currentValue;
@ -265,8 +288,6 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
} }
const isFirstChange = this.unchangedInputs.has(property); const isFirstChange = this.unchangedInputs.has(property);
this.unchangedInputs.delete(property);
const previousValue = isFirstChange ? undefined : this.getInputValue(property); const previousValue = isFirstChange ? undefined : this.getInputValue(property);
this.inputChanges[property] = new SimpleChange(previousValue, currentValue, isFirstChange); this.inputChanges[property] = new SimpleChange(previousValue, currentValue, isFirstChange);
} }
@ -278,6 +299,7 @@ export class ComponentNgElementStrategy implements NgElementStrategy {
} }
this.callNgOnChanges(this.componentRef); this.callNgOnChanges(this.componentRef);
this.markViewForCheck(this.viewChangeDetectorRef!);
this.componentRef.changeDetectorRef.detectChanges(); this.componentRef.changeDetectorRef.detectChanges();
} }

View File

@ -6,7 +6,7 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import {ApplicationRef, ComponentFactory, ComponentFactoryResolver, ComponentRef, Injector, NgModuleRef, NgZone, SimpleChange, SimpleChanges, Type} from '@angular/core'; import {ApplicationRef, ChangeDetectorRef, ComponentFactory, ComponentFactoryResolver, ComponentRef, Injector, NgModuleRef, NgZone, SimpleChange, SimpleChanges, Type} from '@angular/core';
import {fakeAsync, tick} from '@angular/core/testing'; import {fakeAsync, tick} from '@angular/core/testing';
import {Subject} from 'rxjs'; import {Subject} from 'rxjs';
@ -180,8 +180,11 @@ describe('ComponentFactoryNgElementStrategy', () => {
}); });
describe('when inputs change and is connected', () => { describe('when inputs change and is connected', () => {
let viewChangeDetectorRef: ChangeDetectorRef;
beforeEach(() => { beforeEach(() => {
strategy.connect(document.createElement('div')); strategy.connect(document.createElement('div'));
viewChangeDetectorRef = componentRef.injector.get(ChangeDetectorRef);
}); });
it('should be set on the component instance', () => { it('should be set on the component instance', () => {
@ -209,6 +212,22 @@ describe('ComponentFactoryNgElementStrategy', () => {
expect(componentRef.changeDetectorRef.detectChanges).toHaveBeenCalledTimes(2); expect(componentRef.changeDetectorRef.detectChanges).toHaveBeenCalledTimes(2);
})); }));
it('should not detect changes if the input is set to the same value', fakeAsync(() => {
(componentRef.changeDetectorRef.detectChanges as jasmine.Spy).calls.reset();
strategy.setInputValue('fooFoo', 'fooFoo-1');
strategy.setInputValue('barBar', 'barBar-1');
tick(16); // scheduler waits 16ms if RAF is unavailable
expect(componentRef.changeDetectorRef.detectChanges).toHaveBeenCalledTimes(1);
(componentRef.changeDetectorRef.detectChanges as jasmine.Spy).calls.reset();
strategy.setInputValue('fooFoo', 'fooFoo-1');
strategy.setInputValue('barBar', 'barBar-1');
tick(16); // scheduler waits 16ms if RAF is unavailable
expect(componentRef.changeDetectorRef.detectChanges).not.toHaveBeenCalled();
}));
it('should call ngOnChanges', fakeAsync(() => { it('should call ngOnChanges', fakeAsync(() => {
strategy.setInputValue('fooFoo', 'fooFoo-1'); strategy.setInputValue('fooFoo', 'fooFoo-1');
tick(16); // scheduler waits 16ms if RAF is unavailable tick(16); // scheduler waits 16ms if RAF is unavailable
@ -246,6 +265,22 @@ describe('ComponentFactoryNgElementStrategy', () => {
}); });
})); }));
it('should not call ngOnChanges if the inout is set to the same value', fakeAsync(() => {
const ngOnChangesSpy = spyOn(componentRef.instance, 'ngOnChanges');
strategy.setInputValue('fooFoo', 'fooFoo-1');
strategy.setInputValue('barBar', 'barBar-1');
tick(16); // scheduler waits 16ms if RAF is unavailable
expect(ngOnChangesSpy).toHaveBeenCalledTimes(1);
ngOnChangesSpy.calls.reset();
strategy.setInputValue('fooFoo', 'fooFoo-1');
strategy.setInputValue('barBar', 'barBar-1');
tick(16); // scheduler waits 16ms if RAF is unavailable
expect(ngOnChangesSpy).not.toHaveBeenCalled();
}));
it('should not try to call ngOnChanges if not present on the component', fakeAsync(() => { it('should not try to call ngOnChanges if not present on the component', fakeAsync(() => {
const factory2 = new FakeComponentFactory(FakeComponentWithoutNgOnChanges); const factory2 = new FakeComponentFactory(FakeComponentWithoutNgOnChanges);
const strategy2 = new ComponentNgElementStrategy(factory2, injector); const strategy2 = new ComponentNgElementStrategy(factory2, injector);
@ -261,6 +296,71 @@ describe('ComponentFactoryNgElementStrategy', () => {
// been thrown and `changeDetectorRef2.detectChanges()` would not have been called. // been thrown and `changeDetectorRef2.detectChanges()` would not have been called.
expect(changeDetectorRef2.detectChanges).toHaveBeenCalledTimes(1); expect(changeDetectorRef2.detectChanges).toHaveBeenCalledTimes(1);
})); }));
it('should mark the view for check', fakeAsync(() => {
expect(viewChangeDetectorRef.markForCheck).not.toHaveBeenCalled();
strategy.setInputValue('fooFoo', 'fooFoo-1');
tick(16); // scheduler waits 16ms if RAF is unavailable
expect(viewChangeDetectorRef.markForCheck).toHaveBeenCalledTimes(1);
}));
it('should mark the view for check once for multiple input changes', fakeAsync(() => {
strategy.setInputValue('fooFoo', 'fooFoo-1');
strategy.setInputValue('barBar', 'barBar-1');
tick(16); // scheduler waits 16ms if RAF is unavailable
expect(viewChangeDetectorRef.markForCheck).toHaveBeenCalledTimes(1);
}));
it('should mark the view for check twice for changes in different rounds with previous values',
fakeAsync(() => {
strategy.setInputValue('fooFoo', 'fooFoo-1');
strategy.setInputValue('barBar', 'barBar-1');
tick(16); // scheduler waits 16ms if RAF is unavailable
expect(viewChangeDetectorRef.markForCheck).toHaveBeenCalledTimes(1);
strategy.setInputValue('fooFoo', 'fooFoo-2');
strategy.setInputValue('barBar', 'barBar-2');
tick(16); // scheduler waits 16ms if RAF is unavailable
expect(viewChangeDetectorRef.markForCheck).toHaveBeenCalledTimes(2);
}));
it('should mark the view for check even if ngOnChanges is not present on the component',
fakeAsync(() => {
const factory2 = new FakeComponentFactory(FakeComponentWithoutNgOnChanges);
const strategy2 = new ComponentNgElementStrategy(factory2, injector);
const viewChangeDetectorRef2 = factory2.componentRef.injector.get(ChangeDetectorRef);
strategy2.connect(document.createElement('div'));
(viewChangeDetectorRef2.markForCheck as jasmine.Spy).calls.reset();
strategy2.setInputValue('fooFoo', 'fooFoo-1');
expect(() => tick(16)).not.toThrow(); // scheduler waits 16ms if RAF is unavailable
// If the strategy would have tried to call `component.ngOnChanges()`, an error would have
// been thrown and `viewChangeDetectorRef2.markForCheck()` would not have been called.
expect(viewChangeDetectorRef2.markForCheck).toHaveBeenCalledTimes(1);
}));
it('should not mark the view for check if the input is set to the same value', fakeAsync(() => {
(viewChangeDetectorRef.markForCheck as jasmine.Spy).calls.reset();
strategy.setInputValue('fooFoo', 'fooFoo-1');
strategy.setInputValue('barBar', 'barBar-1');
tick(16); // scheduler waits 16ms if RAF is unavailable
expect(viewChangeDetectorRef.markForCheck).toHaveBeenCalledTimes(1);
(viewChangeDetectorRef.markForCheck as jasmine.Spy).calls.reset();
strategy.setInputValue('fooFoo', 'fooFoo-1');
strategy.setInputValue('barBar', 'barBar-1');
tick(16); // scheduler waits 16ms if RAF is unavailable
expect(viewChangeDetectorRef.markForCheck).not.toHaveBeenCalled();
}));
}); });
describe('disconnect', () => { describe('disconnect', () => {
@ -345,6 +445,9 @@ export class FakeComponentFactory<T extends Type<any>> extends ComponentFactory<
{ {
changeDetectorRef: jasmine.createSpyObj('changeDetectorRef', ['detectChanges']), changeDetectorRef: jasmine.createSpyObj('changeDetectorRef', ['detectChanges']),
hostView: {}, hostView: {},
injector: jasmine.createSpyObj('injector', {
get: jasmine.createSpyObj('viewChangeDetectorRef', ['markForCheck']),
}),
instance: new this.ComponentClass(), instance: new this.ComponentClass(),
}); });