fix(ivy): don’t publish animation bindings as attributes (#27805)
Some of the animation tests have been failing because animation gets triggered multiple times. The reason for this is that the compiler was generating static attribute bindings in addition to dynamic bindings. This created multiple writes to the animation render which failed the tests. PR Close #27805
This commit is contained in:
parent
880b4aabdb
commit
1f1e77b641
|
@ -120,4 +120,70 @@ describe('r3_view_compiler', () => {
|
||||||
expectEmit(result.source, bV_call, 'Incorrect bV call');
|
expectEmit(result.source, bV_call, 'Incorrect bV call');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('animations', () => {
|
||||||
|
it('should keep @attr but suppress [@attr]', () => {
|
||||||
|
const files: MockDirectory = {
|
||||||
|
app: {
|
||||||
|
'example.ts': `
|
||||||
|
import {Component, NgModule} from '@angular/core';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'my-app',
|
||||||
|
template: '<div @attrOnly [@myAnimation]="exp"></div>'
|
||||||
|
})
|
||||||
|
export class MyApp {
|
||||||
|
}
|
||||||
|
|
||||||
|
@NgModule({declarations: [MyApp]})
|
||||||
|
export class MyModule {}`
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const template = `
|
||||||
|
const _c0 = ["@attrOnly", ""];
|
||||||
|
// ...
|
||||||
|
template: function MyApp_Template(rf, ctx) {
|
||||||
|
if (rf & 1) {
|
||||||
|
$i0$.ɵelement(0, "div", _c0);
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
// ...
|
||||||
|
}`;
|
||||||
|
const result = compile(files, angularFiles);
|
||||||
|
expectEmit(result.source, template, 'Incorrect initialization attributes');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should dedup multiple [@event] listeners', () => {
|
||||||
|
const files: MockDirectory = {
|
||||||
|
app: {
|
||||||
|
'example.ts': `
|
||||||
|
import {Component, NgModule} from '@angular/core';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'my-app',
|
||||||
|
template: '<div (@mySelector.start)="false" (@mySelector.done)="false" [@mySelector]="0"></div>'
|
||||||
|
})
|
||||||
|
export class MyApp {
|
||||||
|
}
|
||||||
|
|
||||||
|
@NgModule({declarations: [MyApp]})
|
||||||
|
export class MyModule {}`
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const template = `
|
||||||
|
const _c0 = [1, "mySelector"];
|
||||||
|
// ...
|
||||||
|
template: function MyApp_Template(rf, ctx) {
|
||||||
|
if (rf & 1) {
|
||||||
|
$i0$.ɵelementStart(0, "div", _c0);
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
// ...
|
||||||
|
}`;
|
||||||
|
const result = compile(files, angularFiles);
|
||||||
|
expectEmit(result.source, template, 'Incorrect initialization attributes');
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -214,7 +214,6 @@ describe('compiler compliance: styling', () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
const template = `
|
const template = `
|
||||||
const $e0_attrs$ = ["@foo", ""];
|
|
||||||
const $e1_attrs$ = ["@bar", ""];
|
const $e1_attrs$ = ["@bar", ""];
|
||||||
const $e2_attrs$ = ["@baz", ""];
|
const $e2_attrs$ = ["@baz", ""];
|
||||||
…
|
…
|
||||||
|
@ -224,7 +223,7 @@ describe('compiler compliance: styling', () => {
|
||||||
vars: 1,
|
vars: 1,
|
||||||
template: function MyComponent_Template(rf, $ctx$) {
|
template: function MyComponent_Template(rf, $ctx$) {
|
||||||
if (rf & 1) {
|
if (rf & 1) {
|
||||||
$r3$.ɵelement(0, "div", $e0_attrs$);
|
$r3$.ɵelement(0, "div");
|
||||||
$r3$.ɵelement(1, "div", $e1_attrs$);
|
$r3$.ɵelement(1, "div", $e1_attrs$);
|
||||||
$r3$.ɵelement(2, "div", $e2_attrs$);
|
$r3$.ɵelement(2, "div", $e2_attrs$);
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,7 @@ import {flatten, sanitizeIdentifier} from '../../compile_metadata';
|
||||||
import {BindingForm, BuiltinFunctionCall, LocalResolver, convertActionBinding, convertPropertyBinding} from '../../compiler_util/expression_converter';
|
import {BindingForm, BuiltinFunctionCall, LocalResolver, convertActionBinding, convertPropertyBinding} from '../../compiler_util/expression_converter';
|
||||||
import {ConstantPool} from '../../constant_pool';
|
import {ConstantPool} from '../../constant_pool';
|
||||||
import * as core from '../../core';
|
import * as core from '../../core';
|
||||||
import {AST, AstMemoryEfficientTransformer, BindingPipe, BindingType, FunctionCall, ImplicitReceiver, Interpolation, LiteralArray, LiteralMap, LiteralPrimitive, ParsedEventType, PropertyRead} from '../../expression_parser/ast';
|
import {AST, ASTWithSource, AstMemoryEfficientTransformer, BindingPipe, BindingType, FunctionCall, ImplicitReceiver, Interpolation, LiteralArray, LiteralMap, LiteralPrimitive, ParsedEventType, PropertyRead} from '../../expression_parser/ast';
|
||||||
import {Lexer} from '../../expression_parser/lexer';
|
import {Lexer} from '../../expression_parser/lexer';
|
||||||
import {Parser} from '../../expression_parser/parser';
|
import {Parser} from '../../expression_parser/parser';
|
||||||
import * as i18n from '../../i18n/i18n_ast';
|
import * as i18n from '../../i18n/i18n_ast';
|
||||||
|
@ -967,6 +967,29 @@ export class TemplateDefinitionBuilder implements t.Visitor<void>, LocalResolver
|
||||||
styles?: StylingBuilder): o.Expression[] {
|
styles?: StylingBuilder): o.Expression[] {
|
||||||
const attrExprs: o.Expression[] = [];
|
const attrExprs: o.Expression[] = [];
|
||||||
const nonSyntheticInputs: t.BoundAttribute[] = [];
|
const nonSyntheticInputs: t.BoundAttribute[] = [];
|
||||||
|
const alreadySeen = new Set<string>();
|
||||||
|
|
||||||
|
function isASTWithSource(ast: AST): ast is ASTWithSource {
|
||||||
|
return ast instanceof ASTWithSource;
|
||||||
|
}
|
||||||
|
|
||||||
|
function isLiteralPrimitive(ast: AST): ast is LiteralPrimitive {
|
||||||
|
return ast instanceof LiteralPrimitive;
|
||||||
|
}
|
||||||
|
|
||||||
|
function addAttrExpr(key: string | number, value?: o.Expression): void {
|
||||||
|
if (typeof key === 'string') {
|
||||||
|
if (!alreadySeen.has(key)) {
|
||||||
|
attrExprs.push(o.literal(key));
|
||||||
|
if (value !== undefined) {
|
||||||
|
attrExprs.push(value);
|
||||||
|
}
|
||||||
|
alreadySeen.add(key);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
attrExprs.push(o.literal(key));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (inputs.length) {
|
if (inputs.length) {
|
||||||
const EMPTY_STRING_EXPR = asLiteral('');
|
const EMPTY_STRING_EXPR = asLiteral('');
|
||||||
|
@ -976,7 +999,13 @@ export class TemplateDefinitionBuilder implements t.Visitor<void>, LocalResolver
|
||||||
// may be supported differently in future versions of angular. However,
|
// may be supported differently in future versions of angular. However,
|
||||||
// @triggers should always just be treated as regular attributes (it's up
|
// @triggers should always just be treated as regular attributes (it's up
|
||||||
// to the renderer to detect and use them in a special way).
|
// to the renderer to detect and use them in a special way).
|
||||||
attrExprs.push(asLiteral(prepareSyntheticAttributeName(input.name)), EMPTY_STRING_EXPR);
|
const valueExp = input.value;
|
||||||
|
if (isASTWithSource(valueExp)) {
|
||||||
|
const literal = valueExp.ast;
|
||||||
|
if (isLiteralPrimitive(literal) && literal.value === undefined) {
|
||||||
|
addAttrExpr(prepareSyntheticAttributeName(input.name), EMPTY_STRING_EXPR);
|
||||||
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
nonSyntheticInputs.push(input);
|
nonSyntheticInputs.push(input);
|
||||||
}
|
}
|
||||||
|
@ -991,9 +1020,9 @@ export class TemplateDefinitionBuilder implements t.Visitor<void>, LocalResolver
|
||||||
}
|
}
|
||||||
|
|
||||||
if (nonSyntheticInputs.length || outputs.length) {
|
if (nonSyntheticInputs.length || outputs.length) {
|
||||||
attrExprs.push(o.literal(core.AttributeMarker.SelectOnly));
|
addAttrExpr(core.AttributeMarker.SelectOnly);
|
||||||
nonSyntheticInputs.forEach((i: t.BoundAttribute) => attrExprs.push(asLiteral(i.name)));
|
nonSyntheticInputs.forEach((i: t.BoundAttribute) => addAttrExpr(i.name));
|
||||||
outputs.forEach((o: t.BoundEvent) => attrExprs.push(asLiteral(o.name)));
|
outputs.forEach((o: t.BoundEvent) => addAttrExpr(o.name));
|
||||||
}
|
}
|
||||||
|
|
||||||
return attrExprs;
|
return attrExprs;
|
||||||
|
|
|
@ -394,8 +394,9 @@ function renderComponentOrTemplate<T>(
|
||||||
hostView: LView, context: T, templateFn?: ComponentTemplate<T>) {
|
hostView: LView, context: T, templateFn?: ComponentTemplate<T>) {
|
||||||
const rendererFactory = hostView[RENDERER_FACTORY];
|
const rendererFactory = hostView[RENDERER_FACTORY];
|
||||||
const oldView = enterView(hostView, hostView[HOST_NODE]);
|
const oldView = enterView(hostView, hostView[HOST_NODE]);
|
||||||
|
const normalExecutionPath = !getCheckNoChangesMode();
|
||||||
try {
|
try {
|
||||||
if (rendererFactory.begin) {
|
if (normalExecutionPath && rendererFactory.begin) {
|
||||||
rendererFactory.begin();
|
rendererFactory.begin();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -414,7 +415,7 @@ function renderComponentOrTemplate<T>(
|
||||||
templateFn && templateFn(RenderFlags.Update, context !);
|
templateFn && templateFn(RenderFlags.Update, context !);
|
||||||
refreshDescendantViews(hostView);
|
refreshDescendantViews(hostView);
|
||||||
} finally {
|
} finally {
|
||||||
if (rendererFactory.end) {
|
if (normalExecutionPath && rendererFactory.end) {
|
||||||
rendererFactory.end();
|
rendererFactory.end();
|
||||||
}
|
}
|
||||||
leaveView(oldView);
|
leaveView(oldView);
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -853,12 +853,11 @@ import {HostListener} from '../../src/metadata/directives';
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
fixmeIvy('unknown').it(
|
it('should cleanup :enter and :leave artifacts from nodes when any animation sequences fail to be built',
|
||||||
'should cleanup :enter and :leave artifacts from nodes when any animation sequences fail to be built',
|
() => {
|
||||||
() => {
|
@Component({
|
||||||
@Component({
|
selector: 'ani-cmp',
|
||||||
selector: 'ani-cmp',
|
template: `
|
||||||
template: `
|
|
||||||
<div [@myAnimation]="items.length" class="parent" #container>
|
<div [@myAnimation]="items.length" class="parent" #container>
|
||||||
<div *ngFor="let item of items" class="child">
|
<div *ngFor="let item of items" class="child">
|
||||||
{{ item }}
|
{{ item }}
|
||||||
|
@ -866,56 +865,56 @@ import {HostListener} from '../../src/metadata/directives';
|
||||||
<div *ngIf="items.length == 0" class="child">Leave!</div>
|
<div *ngIf="items.length == 0" class="child">Leave!</div>
|
||||||
</div>
|
</div>
|
||||||
`,
|
`,
|
||||||
animations: [
|
animations: [
|
||||||
trigger(
|
trigger(
|
||||||
'myAnimation',
|
'myAnimation',
|
||||||
[
|
[
|
||||||
transition('* => 0', []),
|
transition('* => 0', []),
|
||||||
transition(
|
transition(
|
||||||
'* => *',
|
'* => *',
|
||||||
[
|
[
|
||||||
query(
|
query(
|
||||||
'.child:enter',
|
'.child:enter',
|
||||||
[
|
[
|
||||||
style({opacity: 0}),
|
style({opacity: 0}),
|
||||||
animate(1000, style({opacity: 1})),
|
animate(1000, style({opacity: 1})),
|
||||||
]),
|
]),
|
||||||
query(
|
query(
|
||||||
'.incorrect-child:leave',
|
'.incorrect-child:leave',
|
||||||
[
|
[
|
||||||
animate(1000, style({opacity: 0})),
|
animate(1000, style({opacity: 0})),
|
||||||
]),
|
]),
|
||||||
]),
|
]),
|
||||||
]),
|
]),
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
class Cmp {
|
class Cmp {
|
||||||
@ViewChild('container') public container: any;
|
@ViewChild('container') public container: any;
|
||||||
public items: any[] = [];
|
public items: any[] = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
TestBed.configureTestingModule({declarations: [Cmp]});
|
TestBed.configureTestingModule({declarations: [Cmp]});
|
||||||
|
|
||||||
const engine = TestBed.get(ɵAnimationEngine);
|
const engine = TestBed.get(ɵAnimationEngine);
|
||||||
const fixture = TestBed.createComponent(Cmp);
|
const fixture = TestBed.createComponent(Cmp);
|
||||||
const cmp = fixture.componentInstance;
|
const cmp = fixture.componentInstance;
|
||||||
|
|
||||||
cmp.items = [];
|
cmp.items = [];
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
|
|
||||||
cmp.items = [0, 1, 2, 3, 4];
|
cmp.items = [0, 1, 2, 3, 4];
|
||||||
|
|
||||||
expect(() => { fixture.detectChanges(); }).toThrow();
|
expect(() => { fixture.detectChanges(); }).toThrow();
|
||||||
|
|
||||||
const children = cmp.container.nativeElement.querySelectorAll('.child');
|
const children = cmp.container.nativeElement.querySelectorAll('.child');
|
||||||
expect(children.length).toEqual(5);
|
expect(children.length).toEqual(5);
|
||||||
|
|
||||||
for (let i = 0; i < children.length; i++) {
|
for (let i = 0; i < children.length; i++) {
|
||||||
let child = children[i];
|
let child = children[i];
|
||||||
expect(child.classList.contains(ENTER_CLASSNAME)).toBe(false);
|
expect(child.classList.contains(ENTER_CLASSNAME)).toBe(false);
|
||||||
expect(child.classList.contains(LEAVE_CLASSNAME)).toBe(false);
|
expect(child.classList.contains(LEAVE_CLASSNAME)).toBe(false);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should find elements that have been removed via :leave', () => {
|
it('should find elements that have been removed via :leave', () => {
|
||||||
@Component({
|
@Component({
|
||||||
|
@ -2520,9 +2519,8 @@ import {HostListener} from '../../src/metadata/directives';
|
||||||
expect(p4.element.classList.contains('d'));
|
expect(p4.element.classList.contains('d'));
|
||||||
});
|
});
|
||||||
|
|
||||||
fixmeIvy('unknown').it(
|
it('should collect multiple root levels of :enter and :leave nodes', () => {
|
||||||
'should collect multiple root levels of :enter and :leave nodes', () => {
|
@Component({
|
||||||
@Component({
|
|
||||||
selector: 'ani-cmp',
|
selector: 'ani-cmp',
|
||||||
animations: [
|
animations: [
|
||||||
trigger('pageAnimation', [
|
trigger('pageAnimation', [
|
||||||
|
@ -2557,88 +2555,88 @@ import {HostListener} from '../../src/metadata/directives';
|
||||||
`
|
`
|
||||||
})
|
})
|
||||||
class Cmp {
|
class Cmp {
|
||||||
get title() {
|
get title() {
|
||||||
if (this.page1) {
|
if (this.page1) {
|
||||||
return 'hello from page1';
|
return 'hello from page1';
|
||||||
}
|
|
||||||
return 'greetings from page2';
|
|
||||||
}
|
|
||||||
|
|
||||||
page1 = false;
|
|
||||||
page2 = false;
|
|
||||||
loading = false;
|
|
||||||
|
|
||||||
get status() {
|
|
||||||
if (this.loading) return 'loading';
|
|
||||||
if (this.page1) return 'page1';
|
|
||||||
if (this.page2) return 'page2';
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
return 'greetings from page2';
|
||||||
|
}
|
||||||
|
|
||||||
TestBed.configureTestingModule({declarations: [Cmp]});
|
page1 = false;
|
||||||
|
page2 = false;
|
||||||
|
loading = false;
|
||||||
|
|
||||||
const engine = TestBed.get(ɵAnimationEngine);
|
get status() {
|
||||||
const fixture = TestBed.createComponent(Cmp);
|
if (this.loading) return 'loading';
|
||||||
const cmp = fixture.componentInstance;
|
if (this.page1) return 'page1';
|
||||||
cmp.loading = true;
|
if (this.page2) return 'page2';
|
||||||
fixture.detectChanges();
|
return '';
|
||||||
engine.flush();
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let players = getLog();
|
TestBed.configureTestingModule({declarations: [Cmp]});
|
||||||
resetLog();
|
|
||||||
cancelAllPlayers(players);
|
|
||||||
|
|
||||||
cmp.page1 = true;
|
const engine = TestBed.get(ɵAnimationEngine);
|
||||||
cmp.loading = false;
|
const fixture = TestBed.createComponent(Cmp);
|
||||||
fixture.detectChanges();
|
const cmp = fixture.componentInstance;
|
||||||
engine.flush();
|
cmp.loading = true;
|
||||||
|
fixture.detectChanges();
|
||||||
|
engine.flush();
|
||||||
|
|
||||||
let p1: MockAnimationPlayer;
|
let players = getLog();
|
||||||
let p2: MockAnimationPlayer;
|
resetLog();
|
||||||
let p3: MockAnimationPlayer;
|
cancelAllPlayers(players);
|
||||||
|
|
||||||
players = getLog();
|
cmp.page1 = true;
|
||||||
expect(players.length).toEqual(3);
|
cmp.loading = false;
|
||||||
[p1, p2, p3] = players;
|
fixture.detectChanges();
|
||||||
|
engine.flush();
|
||||||
|
|
||||||
expect(p1.element.classList.contains('loading')).toBe(true);
|
let p1: MockAnimationPlayer;
|
||||||
expect(p2.element.classList.contains('title')).toBe(true);
|
let p2: MockAnimationPlayer;
|
||||||
expect(p3.element.classList.contains('page1')).toBe(true);
|
let p3: MockAnimationPlayer;
|
||||||
|
|
||||||
resetLog();
|
players = getLog();
|
||||||
cancelAllPlayers(players);
|
expect(players.length).toEqual(3);
|
||||||
|
[p1, p2, p3] = players;
|
||||||
|
|
||||||
cmp.page1 = false;
|
expect(p1.element.classList.contains('loading')).toBe(true);
|
||||||
cmp.loading = true;
|
expect(p2.element.classList.contains('title')).toBe(true);
|
||||||
fixture.detectChanges();
|
expect(p3.element.classList.contains('page1')).toBe(true);
|
||||||
|
|
||||||
players = getLog();
|
resetLog();
|
||||||
cancelAllPlayers(players);
|
cancelAllPlayers(players);
|
||||||
|
|
||||||
expect(players.length).toEqual(3);
|
cmp.page1 = false;
|
||||||
[p1, p2, p3] = players;
|
cmp.loading = true;
|
||||||
|
fixture.detectChanges();
|
||||||
|
|
||||||
expect(p1.element.classList.contains('title')).toBe(true);
|
players = getLog();
|
||||||
expect(p2.element.classList.contains('page1')).toBe(true);
|
cancelAllPlayers(players);
|
||||||
expect(p3.element.classList.contains('loading')).toBe(true);
|
|
||||||
|
|
||||||
resetLog();
|
expect(players.length).toEqual(3);
|
||||||
cancelAllPlayers(players);
|
[p1, p2, p3] = players;
|
||||||
|
|
||||||
cmp.page2 = true;
|
expect(p1.element.classList.contains('title')).toBe(true);
|
||||||
cmp.loading = false;
|
expect(p2.element.classList.contains('page1')).toBe(true);
|
||||||
fixture.detectChanges();
|
expect(p3.element.classList.contains('loading')).toBe(true);
|
||||||
engine.flush();
|
|
||||||
|
|
||||||
players = getLog();
|
resetLog();
|
||||||
expect(players.length).toEqual(3);
|
cancelAllPlayers(players);
|
||||||
[p1, p2, p3] = players;
|
|
||||||
|
|
||||||
expect(p1.element.classList.contains('loading')).toBe(true);
|
cmp.page2 = true;
|
||||||
expect(p2.element.classList.contains('title')).toBe(true);
|
cmp.loading = false;
|
||||||
expect(p3.element.classList.contains('page2')).toBe(true);
|
fixture.detectChanges();
|
||||||
});
|
engine.flush();
|
||||||
|
|
||||||
|
players = getLog();
|
||||||
|
expect(players.length).toEqual(3);
|
||||||
|
[p1, p2, p3] = players;
|
||||||
|
|
||||||
|
expect(p1.element.classList.contains('loading')).toBe(true);
|
||||||
|
expect(p2.element.classList.contains('title')).toBe(true);
|
||||||
|
expect(p3.element.classList.contains('page2')).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
it('should emulate leave animation callbacks for all sub elements that have leave triggers within the component',
|
it('should emulate leave animation callbacks for all sub elements that have leave triggers within the component',
|
||||||
fakeAsync(() => {
|
fakeAsync(() => {
|
||||||
|
|
|
@ -34,354 +34,349 @@ import {RouterTestingModule} from '@angular/router/testing';
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
fixmeIvy('unknown').it(
|
it('should query the old and new routes via :leave and :enter', fakeAsync(() => {
|
||||||
'should query the old and new routes via :leave and :enter', fakeAsync(() => {
|
@Component({
|
||||||
@Component({
|
animations: [
|
||||||
animations: [
|
trigger(
|
||||||
trigger(
|
'routerAnimations',
|
||||||
'routerAnimations',
|
[
|
||||||
[
|
transition(
|
||||||
transition(
|
'page1 => page2',
|
||||||
'page1 => page2',
|
[
|
||||||
[
|
query(':leave', animateChild()),
|
||||||
query(':leave', animateChild()),
|
query(':enter', animateChild()),
|
||||||
query(':enter', animateChild()),
|
]),
|
||||||
]),
|
]),
|
||||||
]),
|
],
|
||||||
],
|
template: `
|
||||||
template: `
|
|
||||||
<div [@routerAnimations]="prepareRouteAnimation(r)">
|
<div [@routerAnimations]="prepareRouteAnimation(r)">
|
||||||
<router-outlet #r="outlet"></router-outlet>
|
<router-outlet #r="outlet"></router-outlet>
|
||||||
</div>
|
</div>
|
||||||
`
|
`
|
||||||
})
|
})
|
||||||
class ContainerCmp {
|
class ContainerCmp {
|
||||||
constructor(public router: Router) {}
|
constructor(public router: Router) {}
|
||||||
|
|
||||||
prepareRouteAnimation(r: RouterOutlet) {
|
prepareRouteAnimation(r: RouterOutlet) {
|
||||||
const animation = r.activatedRouteData['animation'];
|
const animation = r.activatedRouteData['animation'];
|
||||||
const value = animation ? animation['value'] : null;
|
const value = animation ? animation['value'] : null;
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'page1',
|
selector: 'page1',
|
||||||
template: `page1`,
|
template: `page1`,
|
||||||
animations: [
|
animations: [
|
||||||
trigger(
|
trigger(
|
||||||
'page1Animation',
|
'page1Animation',
|
||||||
[
|
[
|
||||||
transition(
|
transition(
|
||||||
':leave',
|
':leave',
|
||||||
[
|
[
|
||||||
style({width: '200px'}),
|
style({width: '200px'}),
|
||||||
animate(1000, style({width: '0px'})),
|
animate(1000, style({width: '0px'})),
|
||||||
]),
|
]),
|
||||||
]),
|
]),
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
class Page1Cmp {
|
class Page1Cmp {
|
||||||
@HostBinding('@page1Animation') public doAnimate = true;
|
@HostBinding('@page1Animation') public doAnimate = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'page2',
|
selector: 'page2',
|
||||||
template: `page2`,
|
template: `page2`,
|
||||||
animations: [
|
animations: [
|
||||||
trigger(
|
trigger(
|
||||||
'page2Animation',
|
'page2Animation',
|
||||||
[
|
[
|
||||||
transition(
|
transition(
|
||||||
':enter',
|
':enter',
|
||||||
[
|
[
|
||||||
style({opacity: 0}),
|
style({opacity: 0}),
|
||||||
animate(1000, style({opacity: 1})),
|
animate(1000, style({opacity: 1})),
|
||||||
]),
|
]),
|
||||||
]),
|
]),
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
class Page2Cmp {
|
class Page2Cmp {
|
||||||
@HostBinding('@page2Animation') public doAnimate = true;
|
@HostBinding('@page2Animation') public doAnimate = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [Page1Cmp, Page2Cmp, ContainerCmp],
|
declarations: [Page1Cmp, Page2Cmp, ContainerCmp],
|
||||||
imports: [RouterTestingModule.withRoutes([
|
imports: [RouterTestingModule.withRoutes([
|
||||||
{path: 'page1', component: Page1Cmp, data: makeAnimationData('page1')},
|
{path: 'page1', component: Page1Cmp, data: makeAnimationData('page1')},
|
||||||
{path: 'page2', component: Page2Cmp, data: makeAnimationData('page2')}
|
{path: 'page2', component: Page2Cmp, data: makeAnimationData('page2')}
|
||||||
])]
|
])]
|
||||||
});
|
});
|
||||||
|
|
||||||
const engine = TestBed.get(ɵAnimationEngine);
|
const engine = TestBed.get(ɵAnimationEngine);
|
||||||
const fixture = TestBed.createComponent(ContainerCmp);
|
const fixture = TestBed.createComponent(ContainerCmp);
|
||||||
const cmp = fixture.componentInstance;
|
const cmp = fixture.componentInstance;
|
||||||
cmp.router.initialNavigation();
|
cmp.router.initialNavigation();
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
cmp.router.navigateByUrl('/page1');
|
cmp.router.navigateByUrl('/page1');
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
cmp.router.navigateByUrl('/page2');
|
cmp.router.navigateByUrl('/page2');
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
const player = engine.players[0] !;
|
const player = engine.players[0] !;
|
||||||
const groupPlayer = player.getRealPlayer() as AnimationGroupPlayer;
|
const groupPlayer = player.getRealPlayer() as AnimationGroupPlayer;
|
||||||
const players = groupPlayer.players as MockAnimationPlayer[];
|
const players = groupPlayer.players as MockAnimationPlayer[];
|
||||||
|
|
||||||
expect(players.length).toEqual(2);
|
expect(players.length).toEqual(2);
|
||||||
const [p1, p2] = players;
|
const [p1, p2] = players;
|
||||||
|
|
||||||
expect(p1.duration).toEqual(1000);
|
expect(p1.duration).toEqual(1000);
|
||||||
expect(p1.keyframes).toEqual([
|
expect(p1.keyframes).toEqual([
|
||||||
{offset: 0, width: '200px'},
|
{offset: 0, width: '200px'},
|
||||||
{offset: 1, width: '0px'},
|
{offset: 1, width: '0px'},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
expect(p2.duration).toEqual(2000);
|
expect(p2.duration).toEqual(2000);
|
||||||
expect(p2.keyframes).toEqual([
|
expect(p2.keyframes).toEqual([
|
||||||
{offset: 0, opacity: '0'},
|
{offset: 0, opacity: '0'},
|
||||||
{offset: .5, opacity: '0'},
|
{offset: .5, opacity: '0'},
|
||||||
{offset: 1, opacity: '1'},
|
{offset: 1, opacity: '1'},
|
||||||
]);
|
]);
|
||||||
}));
|
}));
|
||||||
|
|
||||||
fixmeIvy('unknown').it(
|
it('should allow inner enter animations to be emulated within a routed item', fakeAsync(() => {
|
||||||
'should allow inner enter animations to be emulated within a routed item', fakeAsync(() => {
|
@Component({
|
||||||
@Component({
|
animations: [
|
||||||
animations: [
|
trigger(
|
||||||
trigger(
|
'routerAnimations',
|
||||||
'routerAnimations',
|
[
|
||||||
[
|
transition(
|
||||||
transition(
|
'page1 => page2',
|
||||||
'page1 => page2',
|
[
|
||||||
[
|
query(':enter', animateChild()),
|
||||||
query(':enter', animateChild()),
|
]),
|
||||||
]),
|
]),
|
||||||
]),
|
],
|
||||||
],
|
template: `
|
||||||
template: `
|
|
||||||
<div [@routerAnimations]="prepareRouteAnimation(r)">
|
<div [@routerAnimations]="prepareRouteAnimation(r)">
|
||||||
<router-outlet #r="outlet"></router-outlet>
|
<router-outlet #r="outlet"></router-outlet>
|
||||||
</div>
|
</div>
|
||||||
`
|
`
|
||||||
})
|
})
|
||||||
class ContainerCmp {
|
class ContainerCmp {
|
||||||
constructor(public router: Router) {}
|
constructor(public router: Router) {}
|
||||||
|
|
||||||
prepareRouteAnimation(r: RouterOutlet) {
|
prepareRouteAnimation(r: RouterOutlet) {
|
||||||
const animation = r.activatedRouteData['animation'];
|
const animation = r.activatedRouteData['animation'];
|
||||||
const value = animation ? animation['value'] : null;
|
const value = animation ? animation['value'] : null;
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({selector: 'page1', template: `page1`, animations: []})
|
@Component({selector: 'page1', template: `page1`, animations: []})
|
||||||
class Page1Cmp {
|
class Page1Cmp {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'page2',
|
selector: 'page2',
|
||||||
template: `
|
template: `
|
||||||
<h1>Page 2</h1>
|
<h1>Page 2</h1>
|
||||||
<div *ngIf="exp" class="if-one" @ifAnimation></div>
|
<div *ngIf="exp" class="if-one" @ifAnimation></div>
|
||||||
<div *ngIf="exp" class="if-two" @ifAnimation></div>
|
<div *ngIf="exp" class="if-two" @ifAnimation></div>
|
||||||
`,
|
`,
|
||||||
animations: [
|
animations: [
|
||||||
trigger(
|
trigger(
|
||||||
'page2Animation',
|
'page2Animation',
|
||||||
[
|
[
|
||||||
transition(
|
transition(
|
||||||
':enter',
|
':enter',
|
||||||
[query('.if-one', animateChild()), query('.if-two', animateChild())]),
|
[query('.if-one', animateChild()), query('.if-two', animateChild())]),
|
||||||
]),
|
]),
|
||||||
trigger(
|
trigger(
|
||||||
'ifAnimation',
|
'ifAnimation',
|
||||||
[transition(
|
[transition(
|
||||||
':enter', [style({opacity: 0}), animate(1000, style({opacity: 1}))])])
|
':enter', [style({opacity: 0}), animate(1000, style({opacity: 1}))])])
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
class Page2Cmp {
|
class Page2Cmp {
|
||||||
@HostBinding('@page2Animation') public doAnimate = true;
|
@HostBinding('@page2Animation') public doAnimate = true;
|
||||||
|
|
||||||
public exp = true;
|
public exp = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [Page1Cmp, Page2Cmp, ContainerCmp],
|
declarations: [Page1Cmp, Page2Cmp, ContainerCmp],
|
||||||
imports: [RouterTestingModule.withRoutes([
|
imports: [RouterTestingModule.withRoutes([
|
||||||
{path: 'page1', component: Page1Cmp, data: makeAnimationData('page1')},
|
{path: 'page1', component: Page1Cmp, data: makeAnimationData('page1')},
|
||||||
{path: 'page2', component: Page2Cmp, data: makeAnimationData('page2')}
|
{path: 'page2', component: Page2Cmp, data: makeAnimationData('page2')}
|
||||||
])]
|
])]
|
||||||
});
|
});
|
||||||
|
|
||||||
const engine = TestBed.get(ɵAnimationEngine);
|
const engine = TestBed.get(ɵAnimationEngine);
|
||||||
const fixture = TestBed.createComponent(ContainerCmp);
|
const fixture = TestBed.createComponent(ContainerCmp);
|
||||||
const cmp = fixture.componentInstance;
|
const cmp = fixture.componentInstance;
|
||||||
cmp.router.initialNavigation();
|
cmp.router.initialNavigation();
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
cmp.router.navigateByUrl('/page1');
|
cmp.router.navigateByUrl('/page1');
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
cmp.router.navigateByUrl('/page2');
|
cmp.router.navigateByUrl('/page2');
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
const player = engine.players[0] !;
|
const player = engine.players[0] !;
|
||||||
const groupPlayer = player.getRealPlayer() as AnimationGroupPlayer;
|
const groupPlayer = player.getRealPlayer() as AnimationGroupPlayer;
|
||||||
const players = groupPlayer.players as MockAnimationPlayer[];
|
const players = groupPlayer.players as MockAnimationPlayer[];
|
||||||
|
|
||||||
expect(players.length).toEqual(2);
|
expect(players.length).toEqual(2);
|
||||||
const [p1, p2] = players;
|
const [p1, p2] = players;
|
||||||
|
|
||||||
expect(p1.keyframes).toEqual([
|
expect(p1.keyframes).toEqual([
|
||||||
{offset: 0, opacity: '0'},
|
{offset: 0, opacity: '0'},
|
||||||
{offset: 1, opacity: '1'},
|
{offset: 1, opacity: '1'},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
expect(p2.keyframes).toEqual([
|
expect(p2.keyframes).toEqual([
|
||||||
{offset: 0, opacity: '0'},
|
{offset: 0, opacity: '0'},
|
||||||
{offset: .5, opacity: '0'},
|
{offset: .5, opacity: '0'},
|
||||||
{offset: 1, opacity: '1'},
|
{offset: 1, opacity: '1'},
|
||||||
]);
|
]);
|
||||||
}));
|
}));
|
||||||
|
|
||||||
fixmeIvy('unknown').it(
|
it('should allow inner leave animations to be emulated within a routed item', fakeAsync(() => {
|
||||||
'should allow inner leave animations to be emulated within a routed item', fakeAsync(() => {
|
@Component({
|
||||||
@Component({
|
animations: [
|
||||||
animations: [
|
trigger(
|
||||||
trigger(
|
'routerAnimations',
|
||||||
'routerAnimations',
|
[
|
||||||
[
|
transition(
|
||||||
transition(
|
'page1 => page2',
|
||||||
'page1 => page2',
|
[
|
||||||
[
|
query(':leave', animateChild()),
|
||||||
query(':leave', animateChild()),
|
]),
|
||||||
]),
|
]),
|
||||||
]),
|
],
|
||||||
],
|
template: `
|
||||||
template: `
|
|
||||||
<div [@routerAnimations]="prepareRouteAnimation(r)">
|
<div [@routerAnimations]="prepareRouteAnimation(r)">
|
||||||
<router-outlet #r="outlet"></router-outlet>
|
<router-outlet #r="outlet"></router-outlet>
|
||||||
</div>
|
</div>
|
||||||
`
|
`
|
||||||
})
|
})
|
||||||
class ContainerCmp {
|
class ContainerCmp {
|
||||||
constructor(public router: Router) {}
|
constructor(public router: Router) {}
|
||||||
|
|
||||||
prepareRouteAnimation(r: RouterOutlet) {
|
prepareRouteAnimation(r: RouterOutlet) {
|
||||||
const animation = r.activatedRouteData['animation'];
|
const animation = r.activatedRouteData['animation'];
|
||||||
const value = animation ? animation['value'] : null;
|
const value = animation ? animation['value'] : null;
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'page1',
|
selector: 'page1',
|
||||||
template: `
|
template: `
|
||||||
<h1>Page 1</h1>
|
<h1>Page 1</h1>
|
||||||
<div *ngIf="exp" class="if-one" @ifAnimation></div>
|
<div *ngIf="exp" class="if-one" @ifAnimation></div>
|
||||||
<div *ngIf="exp" class="if-two" @ifAnimation></div>
|
<div *ngIf="exp" class="if-two" @ifAnimation></div>
|
||||||
`,
|
`,
|
||||||
animations: [
|
animations: [
|
||||||
trigger(
|
trigger(
|
||||||
'page1Animation',
|
'page1Animation',
|
||||||
[
|
[
|
||||||
transition(
|
transition(
|
||||||
':leave',
|
':leave',
|
||||||
[query('.if-one', animateChild()), query('.if-two', animateChild())]),
|
[query('.if-one', animateChild()), query('.if-two', animateChild())]),
|
||||||
]),
|
]),
|
||||||
trigger(
|
trigger(
|
||||||
'ifAnimation',
|
'ifAnimation',
|
||||||
[transition(
|
[transition(':leave', [style({opacity: 1}), animate(1000, style({opacity: 0}))])]),
|
||||||
':leave', [style({opacity: 1}), animate(1000, style({opacity: 0}))])]),
|
]
|
||||||
]
|
})
|
||||||
})
|
class Page1Cmp {
|
||||||
class Page1Cmp {
|
@HostBinding('@page1Animation') public doAnimate = true;
|
||||||
@HostBinding('@page1Animation') public doAnimate = true;
|
|
||||||
|
|
||||||
public exp = true;
|
public exp = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({selector: 'page2', template: `page2`, animations: []})
|
@Component({selector: 'page2', template: `page2`, animations: []})
|
||||||
class Page2Cmp {
|
class Page2Cmp {
|
||||||
}
|
}
|
||||||
|
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [Page1Cmp, Page2Cmp, ContainerCmp],
|
declarations: [Page1Cmp, Page2Cmp, ContainerCmp],
|
||||||
imports: [RouterTestingModule.withRoutes([
|
imports: [RouterTestingModule.withRoutes([
|
||||||
{path: 'page1', component: Page1Cmp, data: makeAnimationData('page1')},
|
{path: 'page1', component: Page1Cmp, data: makeAnimationData('page1')},
|
||||||
{path: 'page2', component: Page2Cmp, data: makeAnimationData('page2')}
|
{path: 'page2', component: Page2Cmp, data: makeAnimationData('page2')}
|
||||||
])]
|
])]
|
||||||
});
|
});
|
||||||
|
|
||||||
const engine = TestBed.get(ɵAnimationEngine);
|
const engine = TestBed.get(ɵAnimationEngine);
|
||||||
const fixture = TestBed.createComponent(ContainerCmp);
|
const fixture = TestBed.createComponent(ContainerCmp);
|
||||||
const cmp = fixture.componentInstance;
|
const cmp = fixture.componentInstance;
|
||||||
cmp.router.initialNavigation();
|
cmp.router.initialNavigation();
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
cmp.router.navigateByUrl('/page1');
|
cmp.router.navigateByUrl('/page1');
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
cmp.router.navigateByUrl('/page2');
|
cmp.router.navigateByUrl('/page2');
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
const player = engine.players[0] !;
|
const player = engine.players[0] !;
|
||||||
const groupPlayer = player.getRealPlayer() as AnimationGroupPlayer;
|
const groupPlayer = player.getRealPlayer() as AnimationGroupPlayer;
|
||||||
const players = groupPlayer.players as MockAnimationPlayer[];
|
const players = groupPlayer.players as MockAnimationPlayer[];
|
||||||
|
|
||||||
expect(players.length).toEqual(2);
|
expect(players.length).toEqual(2);
|
||||||
const [p1, p2] = players;
|
const [p1, p2] = players;
|
||||||
|
|
||||||
expect(p1.keyframes).toEqual([
|
expect(p1.keyframes).toEqual([
|
||||||
{offset: 0, opacity: '1'},
|
{offset: 0, opacity: '1'},
|
||||||
{offset: 1, opacity: '0'},
|
{offset: 1, opacity: '0'},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
expect(p2.keyframes).toEqual([
|
expect(p2.keyframes).toEqual([
|
||||||
{offset: 0, opacity: '1'},
|
{offset: 0, opacity: '1'},
|
||||||
{offset: .5, opacity: '1'},
|
{offset: .5, opacity: '1'},
|
||||||
{offset: 1, opacity: '0'},
|
{offset: 1, opacity: '0'},
|
||||||
]);
|
]);
|
||||||
}));
|
}));
|
||||||
|
|
||||||
fixmeIvy('unknown').it(
|
it('should properly collect :enter / :leave router nodes even when another non-router *template component is within the trigger boundaries',
|
||||||
'should properly collect :enter / :leave router nodes even when another non-router *template component is within the trigger boundaries',
|
fakeAsync(() => {
|
||||||
fakeAsync(() => {
|
@Component({
|
||||||
@Component({
|
selector: 'ani-cmp',
|
||||||
selector: 'ani-cmp',
|
animations: [
|
||||||
animations: [
|
trigger(
|
||||||
trigger(
|
'pageAnimation',
|
||||||
'pageAnimation',
|
[
|
||||||
[
|
transition(
|
||||||
transition(
|
'page1 => page2',
|
||||||
'page1 => page2',
|
[
|
||||||
[
|
query('.router-container :leave', animate('1s', style({opacity: 0}))),
|
||||||
query('.router-container :leave', animate('1s', style({opacity: 0}))),
|
query('.router-container :enter', animate('1s', style({opacity: 1}))),
|
||||||
query('.router-container :enter', animate('1s', style({opacity: 1}))),
|
]),
|
||||||
]),
|
]),
|
||||||
]),
|
],
|
||||||
],
|
template: `
|
||||||
template: `
|
|
||||||
<div [@pageAnimation]="prepRoute(outlet)">
|
<div [@pageAnimation]="prepRoute(outlet)">
|
||||||
<header>
|
<header>
|
||||||
<div class="inner">
|
<div class="inner">
|
||||||
|
@ -394,139 +389,136 @@ import {RouterTestingModule} from '@angular/router/testing';
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
`
|
`
|
||||||
})
|
})
|
||||||
class ContainerCmp {
|
class ContainerCmp {
|
||||||
loading = false;
|
loading = false;
|
||||||
|
|
||||||
constructor(public router: Router) {}
|
constructor(public router: Router) {}
|
||||||
|
|
||||||
prepRoute(outlet: any) { return outlet.activatedRouteData['animation']; }
|
prepRoute(outlet: any) { return outlet.activatedRouteData['animation']; }
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({selector: 'page1', template: `page1`})
|
@Component({selector: 'page1', template: `page1`})
|
||||||
class Page1Cmp {
|
class Page1Cmp {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({selector: 'page2', template: `page2`})
|
@Component({selector: 'page2', template: `page2`})
|
||||||
class Page2Cmp {
|
class Page2Cmp {
|
||||||
}
|
}
|
||||||
|
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [Page1Cmp, Page2Cmp, ContainerCmp],
|
declarations: [Page1Cmp, Page2Cmp, ContainerCmp],
|
||||||
imports: [RouterTestingModule.withRoutes([
|
imports: [RouterTestingModule.withRoutes([
|
||||||
{path: 'page1', component: Page1Cmp, data: makeAnimationData('page1')},
|
{path: 'page1', component: Page1Cmp, data: makeAnimationData('page1')},
|
||||||
{path: 'page2', component: Page2Cmp, data: makeAnimationData('page2')}
|
{path: 'page2', component: Page2Cmp, data: makeAnimationData('page2')}
|
||||||
])]
|
])]
|
||||||
});
|
});
|
||||||
|
|
||||||
const engine = TestBed.get(ɵAnimationEngine);
|
const engine = TestBed.get(ɵAnimationEngine);
|
||||||
const fixture = TestBed.createComponent(ContainerCmp);
|
const fixture = TestBed.createComponent(ContainerCmp);
|
||||||
const cmp = fixture.componentInstance;
|
const cmp = fixture.componentInstance;
|
||||||
cmp.router.initialNavigation();
|
cmp.router.initialNavigation();
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
cmp.router.navigateByUrl('/page1');
|
cmp.router.navigateByUrl('/page1');
|
||||||
tick();
|
tick();
|
||||||
cmp.loading = true;
|
cmp.loading = true;
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
cmp.router.navigateByUrl('/page2');
|
cmp.router.navigateByUrl('/page2');
|
||||||
tick();
|
tick();
|
||||||
cmp.loading = false;
|
cmp.loading = false;
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
engine.flush();
|
engine.flush();
|
||||||
|
|
||||||
const players = engine.players;
|
const players = engine.players;
|
||||||
expect(players.length).toEqual(1);
|
expect(players.length).toEqual(1);
|
||||||
const [p1] = players;
|
const [p1] = players;
|
||||||
|
|
||||||
const innerPlayers = p1.getRealPlayer().players;
|
const innerPlayers = p1.getRealPlayer().players;
|
||||||
expect(innerPlayers.length).toEqual(2);
|
expect(innerPlayers.length).toEqual(2);
|
||||||
|
|
||||||
const [ip1, ip2] = innerPlayers;
|
const [ip1, ip2] = innerPlayers;
|
||||||
expect(ip1.element.innerText).toEqual('page1');
|
expect(ip1.element.innerText).toEqual('page1');
|
||||||
expect(ip2.element.innerText).toEqual('page2');
|
expect(ip2.element.innerText).toEqual('page2');
|
||||||
}));
|
}));
|
||||||
|
|
||||||
fixmeIvy('unknown').it(
|
it('should allow a recursive set of :leave animations to occur for nested routes',
|
||||||
'should allow a recursive set of :leave animations to occur for nested routes',
|
fakeAsync(() => {
|
||||||
fakeAsync(() => {
|
@Component({selector: 'ani-cmp', template: '<router-outlet name="recur"></router-outlet>'})
|
||||||
@Component(
|
class ContainerCmp {
|
||||||
{selector: 'ani-cmp', template: '<router-outlet name="recur"></router-outlet>'})
|
constructor(private _router: Router) {}
|
||||||
class ContainerCmp {
|
log: string[] = [];
|
||||||
constructor(private _router: Router) {}
|
|
||||||
log: string[] = [];
|
|
||||||
|
|
||||||
enter() { this._router.navigateByUrl('/(recur:recur/nested)'); }
|
enter() { this._router.navigateByUrl('/(recur:recur/nested)'); }
|
||||||
|
|
||||||
leave() { this._router.navigateByUrl('/'); }
|
leave() { this._router.navigateByUrl('/'); }
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'recur-page',
|
selector: 'recur-page',
|
||||||
template: 'Depth: {{ depth }} \n <router-outlet></router-outlet>',
|
template: 'Depth: {{ depth }} \n <router-outlet></router-outlet>',
|
||||||
animations: [
|
animations: [
|
||||||
trigger(
|
trigger(
|
||||||
'pageAnimations',
|
'pageAnimations',
|
||||||
[
|
[
|
||||||
transition(
|
transition(':leave', [group([
|
||||||
':leave', [group([
|
sequence([style({opacity: 1}), animate('1s', style({opacity: 0}))]),
|
||||||
sequence([style({opacity: 1}), animate('1s', style({opacity: 0}))]),
|
query('@*', animateChild(), {optional: true})
|
||||||
query('@*', animateChild(), {optional: true})
|
])]),
|
||||||
])]),
|
]),
|
||||||
]),
|
]
|
||||||
]
|
})
|
||||||
})
|
class RecurPageCmp {
|
||||||
class RecurPageCmp {
|
@HostBinding('@pageAnimations') public animatePage = true;
|
||||||
@HostBinding('@pageAnimations') public animatePage = true;
|
|
||||||
|
|
||||||
@HostBinding('attr.data-depth') public depth = 0;
|
@HostBinding('attr.data-depth') public depth = 0;
|
||||||
|
|
||||||
constructor(private container: ContainerCmp, private route: ActivatedRoute) {
|
constructor(private container: ContainerCmp, private route: ActivatedRoute) {
|
||||||
this.route.data.subscribe(data => {
|
this.route.data.subscribe(data => {
|
||||||
this.container.log.push(`DEPTH ${data.depth}`);
|
this.container.log.push(`DEPTH ${data.depth}`);
|
||||||
this.depth = data.depth;
|
this.depth = data.depth;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
declarations: [ContainerCmp, RecurPageCmp],
|
declarations: [ContainerCmp, RecurPageCmp],
|
||||||
imports: [RouterTestingModule.withRoutes([{
|
imports: [RouterTestingModule.withRoutes([{
|
||||||
path: 'recur',
|
path: 'recur',
|
||||||
component: RecurPageCmp,
|
component: RecurPageCmp,
|
||||||
outlet: 'recur',
|
outlet: 'recur',
|
||||||
data: {depth: 0},
|
data: {depth: 0},
|
||||||
children: [{path: 'nested', component: RecurPageCmp, data: {depth: 1}}]
|
children: [{path: 'nested', component: RecurPageCmp, data: {depth: 1}}]
|
||||||
}])]
|
}])]
|
||||||
});
|
});
|
||||||
|
|
||||||
const fixture = TestBed.createComponent(ContainerCmp);
|
const fixture = TestBed.createComponent(ContainerCmp);
|
||||||
const cmp = fixture.componentInstance;
|
const cmp = fixture.componentInstance;
|
||||||
cmp.enter();
|
cmp.enter();
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
flushMicrotasks();
|
flushMicrotasks();
|
||||||
|
|
||||||
expect(cmp.log).toEqual([
|
expect(cmp.log).toEqual([
|
||||||
'DEPTH 0',
|
'DEPTH 0',
|
||||||
'DEPTH 1',
|
'DEPTH 1',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
cmp.leave();
|
cmp.leave();
|
||||||
tick();
|
tick();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
|
|
||||||
const players = getLog();
|
const players = getLog();
|
||||||
expect(players.length).toEqual(2);
|
expect(players.length).toEqual(2);
|
||||||
|
|
||||||
const [p1, p2] = players;
|
const [p1, p2] = players;
|
||||||
expect(p1.element.getAttribute('data-depth')).toEqual('0');
|
expect(p1.element.getAttribute('data-depth')).toEqual('0');
|
||||||
expect(p2.element.getAttribute('data-depth')).toEqual('1');
|
expect(p2.element.getAttribute('data-depth')).toEqual('1');
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -242,10 +242,9 @@ import {fixmeIvy} from '@angular/private/testing';
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
fixmeIvy('unknown').it(
|
it('should treat * styles as ! for queried items that are collected in a container that is being removed',
|
||||||
'should treat * styles as ! for queried items that are collected in a container that is being removed',
|
() => {
|
||||||
() => {
|
@Component({
|
||||||
@Component({
|
|
||||||
selector: 'my-app',
|
selector: 'my-app',
|
||||||
styles: [`
|
styles: [`
|
||||||
.list .outer {
|
.list .outer {
|
||||||
|
@ -286,58 +285,58 @@ import {fixmeIvy} from '@angular/private/testing';
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
class Cmp {
|
class Cmp {
|
||||||
items: any[] = [];
|
items: any[] = [];
|
||||||
|
|
||||||
get exp() { return this.items.length ? 'full' : 'empty'; }
|
get exp() { return this.items.length ? 'full' : 'empty'; }
|
||||||
|
|
||||||
empty() { this.items = []; }
|
empty() { this.items = []; }
|
||||||
|
|
||||||
full() { this.items = [0, 1, 2, 3, 4]; }
|
full() { this.items = [0, 1, 2, 3, 4]; }
|
||||||
}
|
}
|
||||||
|
|
||||||
TestBed.configureTestingModule({declarations: [Cmp]});
|
TestBed.configureTestingModule({declarations: [Cmp]});
|
||||||
|
|
||||||
const engine = TestBed.get(ɵAnimationEngine);
|
const engine = TestBed.get(ɵAnimationEngine);
|
||||||
const fixture = TestBed.createComponent(Cmp);
|
const fixture = TestBed.createComponent(Cmp);
|
||||||
const cmp = fixture.componentInstance;
|
const cmp = fixture.componentInstance;
|
||||||
|
|
||||||
cmp.empty();
|
cmp.empty();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
let player = engine.players[0] !as TransitionAnimationPlayer;
|
let player = engine.players[0] !as TransitionAnimationPlayer;
|
||||||
player.finish();
|
player.finish();
|
||||||
|
|
||||||
cmp.full();
|
cmp.full();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
|
|
||||||
player = engine.players[0] !as TransitionAnimationPlayer;
|
player = engine.players[0] !as TransitionAnimationPlayer;
|
||||||
let queriedPlayers = (player.getRealPlayer() as AnimationGroupPlayer).players;
|
let queriedPlayers = (player.getRealPlayer() as AnimationGroupPlayer).players;
|
||||||
expect(queriedPlayers.length).toEqual(5);
|
expect(queriedPlayers.length).toEqual(5);
|
||||||
|
|
||||||
let i = 0;
|
let i = 0;
|
||||||
for (i = 0; i < queriedPlayers.length; i++) {
|
for (i = 0; i < queriedPlayers.length; i++) {
|
||||||
let player = queriedPlayers[i] as ɵWebAnimationsPlayer;
|
let player = queriedPlayers[i] as ɵWebAnimationsPlayer;
|
||||||
expect(player.keyframes).toEqual([
|
expect(player.keyframes).toEqual([
|
||||||
{height: '0px', offset: 0},
|
{height: '0px', offset: 0},
|
||||||
{height: '50px', offset: 1},
|
{height: '50px', offset: 1},
|
||||||
]);
|
]);
|
||||||
player.finish();
|
player.finish();
|
||||||
}
|
}
|
||||||
|
|
||||||
cmp.empty();
|
cmp.empty();
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
|
|
||||||
player = engine.players[0] !as TransitionAnimationPlayer;
|
player = engine.players[0] !as TransitionAnimationPlayer;
|
||||||
queriedPlayers = (player.getRealPlayer() as AnimationGroupPlayer).players;
|
queriedPlayers = (player.getRealPlayer() as AnimationGroupPlayer).players;
|
||||||
expect(queriedPlayers.length).toEqual(5);
|
expect(queriedPlayers.length).toEqual(5);
|
||||||
|
|
||||||
for (i = 0; i < queriedPlayers.length; i++) {
|
for (i = 0; i < queriedPlayers.length; i++) {
|
||||||
let player = queriedPlayers[i] as ɵWebAnimationsPlayer;
|
let player = queriedPlayers[i] as ɵWebAnimationsPlayer;
|
||||||
expect(player.keyframes).toEqual([
|
expect(player.keyframes).toEqual([
|
||||||
{height: '50px', offset: 0},
|
{height: '50px', offset: 0},
|
||||||
{height: '0px', offset: 1},
|
{height: '0px', offset: 1},
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should compute intermediate styles properly when an animation is cancelled', () => {
|
it('should compute intermediate styles properly when an animation is cancelled', () => {
|
||||||
@Component({
|
@Component({
|
||||||
|
|
|
@ -279,36 +279,35 @@ import {el} from '../../testing/src/browser_util';
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
fixmeIvy(`FW-802: Animation 'start' and 'end' hooks are invoked twice`)
|
it('should provide hooks at the start and end of change detection', () => {
|
||||||
.it('should provide hooks at the start and end of change detection', () => {
|
@Component({
|
||||||
@Component({
|
selector: 'my-cmp',
|
||||||
selector: 'my-cmp',
|
template: `
|
||||||
template: `
|
|
||||||
<div [@myAnimation]="exp"></div>
|
<div [@myAnimation]="exp"></div>
|
||||||
`,
|
`,
|
||||||
animations: [trigger('myAnimation', [])]
|
animations: [trigger('myAnimation', [])]
|
||||||
})
|
})
|
||||||
class Cmp {
|
class Cmp {
|
||||||
public exp: any;
|
public exp: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
TestBed.configureTestingModule({
|
TestBed.configureTestingModule({
|
||||||
providers: [{provide: AnimationEngine, useClass: InjectableAnimationEngine}],
|
providers: [{provide: AnimationEngine, useClass: InjectableAnimationEngine}],
|
||||||
declarations: [Cmp]
|
declarations: [Cmp]
|
||||||
});
|
});
|
||||||
|
|
||||||
const renderer = TestBed.get(RendererFactory2) as ExtendedAnimationRendererFactory;
|
const renderer = TestBed.get(RendererFactory2) as ExtendedAnimationRendererFactory;
|
||||||
const fixture = TestBed.createComponent(Cmp);
|
const fixture = TestBed.createComponent(Cmp);
|
||||||
const cmp = fixture.componentInstance;
|
const cmp = fixture.componentInstance;
|
||||||
|
|
||||||
renderer.log = [];
|
renderer.log = [];
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
expect(renderer.log).toEqual(['begin', 'end']);
|
expect(renderer.log).toEqual(['begin', 'end']);
|
||||||
|
|
||||||
renderer.log = [];
|
renderer.log = [];
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
expect(renderer.log).toEqual(['begin', 'end']);
|
expect(renderer.log).toEqual(['begin', 'end']);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue