Skip to content

Commit f5872c9

Browse files
JeanMechethePunderWoman
authored andcommittedNov 13, 2023
fix(animations): prevent the AsyncAnimationRenderer from calling the delegate when there is no element. (#52570)
This happens when `issueAnimationCommand` is invoked fixes #52538 PR Close #52570
1 parent 7c066a4 commit f5872c9

File tree

6 files changed

+70
-12
lines changed

6 files changed

+70
-12
lines changed
 

‎packages/animations/test/BUILD.bazel

+3
Original file line numberDiff line numberDiff line change
@@ -18,10 +18,13 @@ ts_library(
1818
"//packages/animations",
1919
"//packages/animations/browser",
2020
"//packages/animations/browser/testing",
21+
"//packages/common",
2122
"//packages/core",
2223
"//packages/core/testing",
24+
"//packages/platform-browser",
2325
"//packages/platform-browser-dynamic/testing",
2426
"//packages/platform-browser/animations",
27+
"//packages/platform-browser/animations/async",
2528
"//packages/platform-browser/testing",
2629
],
2730
)

‎packages/animations/test/browser_animation_builder_spec.ts

+47-2
Original file line numberDiff line numberDiff line change
@@ -8,11 +8,13 @@
88
import {animate, AnimationBuilder, style, ɵBrowserAnimationBuilder as BrowserAnimationBuilder} from '@angular/animations';
99
import {AnimationDriver} from '@angular/animations/browser';
1010
import {MockAnimationDriver} from '@angular/animations/browser/testing';
11-
import {Component, ViewChild} from '@angular/core';
11+
import {DOCUMENT} from '@angular/common';
12+
import {Component, NgZone, RendererFactory2, ViewChild} from '@angular/core';
1213
import {fakeAsync, flushMicrotasks, TestBed} from '@angular/core/testing';
14+
import {ɵDomRendererFactory2 as DomRendererFactory2} from '@angular/platform-browser';
1315
import {BrowserDynamicTestingModule, platformBrowserDynamicTesting} from '@angular/platform-browser-dynamic/testing';
1416
import {NoopAnimationsModule} from '@angular/platform-browser/animations';
15-
17+
import {ɵAsyncAnimationRendererFactory as AsyncAnimationRendererFactory} from '@angular/platform-browser/animations/async';
1618

1719
describe('BrowserAnimationBuilder', () => {
1820
if (isNode) {
@@ -236,4 +238,47 @@ describe('BrowserAnimationBuilder', () => {
236238
[BrowserDynamicTestingModule, NoopAnimationsModule], platformBrowserDynamicTesting());
237239
});
238240
});
241+
242+
describe('with Animations async', () => {
243+
beforeEach(() => {
244+
TestBed.configureTestingModule({
245+
providers: [
246+
{
247+
provide: RendererFactory2,
248+
useFactory: (doc: Document, renderer: DomRendererFactory2, zone: NgZone) => {
249+
// Using a empty promise to prevent switching to the delegate to AnimationRenderer
250+
return new AsyncAnimationRendererFactory(
251+
doc, renderer, zone, 'noop', new Promise<any>(() => {}));
252+
},
253+
deps: [DOCUMENT, DomRendererFactory2, NgZone],
254+
},
255+
]
256+
});
257+
});
258+
259+
it('should be able to build', () => {
260+
@Component({
261+
selector: 'ani-cmp',
262+
template: '...',
263+
})
264+
class Cmp {
265+
@ViewChild('target') public target: any;
266+
267+
constructor(public builder: AnimationBuilder) {}
268+
269+
build() {
270+
const definition = this.builder.build([style({'transform': `rotate(0deg)`})]);
271+
272+
return definition.create(this.target);
273+
}
274+
}
275+
276+
TestBed.configureTestingModule({declarations: [Cmp]});
277+
278+
const fixture = TestBed.createComponent(Cmp);
279+
const cmp = fixture.componentInstance;
280+
fixture.detectChanges();
281+
cmp.build();
282+
});
283+
});
239284
});

‎packages/platform-browser/animations/async/src/async-animations.ts

+1
Original file line numberDiff line numberDiff line change
@@ -12,3 +12,4 @@
1212
* Entry point for all animation APIs of the animation browser package.
1313
*/
1414
export {provideAnimationsAsync} from './providers';
15+
export * from './private_export';

‎packages/platform-browser/animations/async/src/async_animation_renderer.ts

+6-10
Original file line numberDiff line numberDiff line change
@@ -6,15 +6,9 @@
66
* found in the LICENSE file at https://angular.io/license
77
*/
88

9-
import type {ɵAnimationRendererFactory as AnimationRendererFactory, ɵAnimationRenderer as AnimationRenderer, ɵBaseAnimationRenderer as BaseAnimationRenderer} from '@angular/animations/browser';
10-
import {NgZone, Renderer2, RendererFactory2, RendererStyleFlags2, RendererType2, ɵRuntimeError as RuntimeError, ɵAnimationRendererType as AnimationRendererType} from '@angular/core';
9+
import {ɵAnimationEngine as AnimationEngine, ɵAnimationRenderer as AnimationRenderer, ɵAnimationRendererFactory as AnimationRendererFactory} from '@angular/animations/browser';
10+
import {NgZone, Renderer2, RendererFactory2, RendererStyleFlags2, RendererType2, ɵAnimationRendererType as AnimationRendererType, ɵRuntimeError as RuntimeError} from '@angular/core';
1111
import {ɵRuntimeErrorCode as RuntimeErrorCode} from '@angular/platform-browser';
12-
/**
13-
* This alias narrows down to only the properties we need when lazy loading (or mock) the module
14-
*/
15-
type AnimationBrowserModuleImports =
16-
Pick<typeof import('@angular/animations/browser'), 'ɵcreateEngine'|'ɵAnimationRendererFactory'>;
17-
1812

1913
const ANIMATION_PREFIX = '@';
2014

@@ -27,8 +21,10 @@ export class AsyncAnimationRendererFactory implements RendererFactory2 {
2721
*/
2822
constructor(
2923
private doc: Document, private delegate: RendererFactory2, private zone: NgZone,
30-
private animationType: 'animations'|'noop',
31-
private moduleImpl?: Promise<AnimationBrowserModuleImports>) {}
24+
private animationType: 'animations'|'noop', private moduleImpl?: Promise<{
25+
ɵcreateEngine: (type: 'animations'|'noop', doc: Document) => AnimationEngine,
26+
ɵAnimationRendererFactory: typeof AnimationRendererFactory
27+
}>) {}
3228

3329
/**
3430
* @internal
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
/**
2+
* @license
3+
* Copyright Google LLC All Rights Reserved.
4+
*
5+
* Use of this source code is governed by an MIT-style license that can be
6+
* found in the LICENSE file at https://angular.io/license
7+
*/
8+
9+
export {AsyncAnimationRendererFactory as ɵAsyncAnimationRendererFactory} from './async_animation_renderer';

‎packages/platform-browser/src/dom/dom_renderer.ts

+4
Original file line numberDiff line numberDiff line change
@@ -279,6 +279,10 @@ class DefaultDomRenderer2 implements Renderer2 {
279279
}
280280

281281
setProperty(el: any, name: string, value: any): void {
282+
if (el == null) {
283+
return;
284+
}
285+
282286
(typeof ngDevMode === 'undefined' || ngDevMode) && this.throwOnSyntheticProps &&
283287
checkNoSyntheticProp(name, 'property');
284288
el[name] = value;

0 commit comments

Comments
 (0)
Please sign in to comment.