Revert "fix(animations): ensure inner :leave animations do not remove node when skipped (#19532)"
This reverts commit ac50bd678e
.
This commit is contained in:
parent
8b571309ed
commit
d0af45c31a
|
@ -141,7 +141,7 @@ export class AnimationTransitionNamespace {
|
||||||
if (!triggersWithStates.hasOwnProperty(name)) {
|
if (!triggersWithStates.hasOwnProperty(name)) {
|
||||||
addClass(element, NG_TRIGGER_CLASSNAME);
|
addClass(element, NG_TRIGGER_CLASSNAME);
|
||||||
addClass(element, NG_TRIGGER_CLASSNAME + '-' + name);
|
addClass(element, NG_TRIGGER_CLASSNAME + '-' + name);
|
||||||
triggersWithStates[name] = DEFAULT_STATE_VALUE;
|
triggersWithStates[name] = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
|
@ -305,31 +305,35 @@ export class AnimationTransitionNamespace {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private _signalRemovalForInnerTriggers(rootElement: any, context: any, animate: boolean = false) {
|
private _destroyInnerNodes(rootElement: any, context: any, animate: boolean = false) {
|
||||||
// emulate a leave animation for all inner nodes within this node.
|
// emulate a leave animation for all inner nodes within this node.
|
||||||
// If there are no animations found for any of the nodes then clear the cache
|
// If there are no animations found for any of the nodes then clear the cache
|
||||||
// for the element.
|
// for the element.
|
||||||
this._engine.driver.query(rootElement, NG_TRIGGER_SELECTOR, true).forEach(elm => {
|
this._engine.driver.query(rootElement, NG_TRIGGER_SELECTOR, true).forEach(elm => {
|
||||||
const namespaces = this._engine.fetchNamespacesByElement(elm);
|
const namespaces = this._engine.fetchNamespacesByElement(elm);
|
||||||
if (namespaces.size) {
|
if (namespaces.size) {
|
||||||
namespaces.forEach(ns => { ns.triggerLeaveAnimation(elm, context, false, true); });
|
namespaces.forEach(ns => ns.removeNode(elm, context, true));
|
||||||
} else {
|
} else {
|
||||||
this.clearElementCache(elm);
|
this.clearElementCache(elm);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
triggerLeaveAnimation(
|
removeNode(element: any, context: any, doNotRecurse?: boolean): void {
|
||||||
element: any, context: any, destroyAfterComplete?: boolean,
|
const engine = this._engine;
|
||||||
defaultToFallback?: boolean): boolean {
|
|
||||||
const triggerStates = this._engine.statesByElement.get(element);
|
if (!doNotRecurse && element.childElementCount) {
|
||||||
|
this._destroyInnerNodes(element, context, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
const triggerStates = engine.statesByElement.get(element);
|
||||||
if (triggerStates) {
|
if (triggerStates) {
|
||||||
const players: TransitionAnimationPlayer[] = [];
|
const players: TransitionAnimationPlayer[] = [];
|
||||||
Object.keys(triggerStates).forEach(triggerName => {
|
Object.keys(triggerStates).forEach(triggerName => {
|
||||||
// this check is here in the event that an element is removed
|
// this check is here in the event that an element is removed
|
||||||
// twice (both on the host level and the component level)
|
// twice (both on the host level and the component level)
|
||||||
if (this._triggers[triggerName]) {
|
if (this._triggers[triggerName]) {
|
||||||
const player = this.trigger(element, triggerName, VOID_VALUE, defaultToFallback);
|
const player = this.trigger(element, triggerName, VOID_VALUE, false);
|
||||||
if (player) {
|
if (player) {
|
||||||
players.push(player);
|
players.push(player);
|
||||||
}
|
}
|
||||||
|
@ -337,55 +341,11 @@ export class AnimationTransitionNamespace {
|
||||||
});
|
});
|
||||||
|
|
||||||
if (players.length) {
|
if (players.length) {
|
||||||
this._engine.markElementAsRemoved(this.id, element, true, context);
|
engine.markElementAsRemoved(this.id, element, true, context);
|
||||||
if (destroyAfterComplete) {
|
optimizeGroupPlayer(players).onDone(() => engine.processLeaveNode(element));
|
||||||
optimizeGroupPlayer(players).onDone(() => this._engine.processLeaveNode(element));
|
return;
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
prepareLeaveAnimationListeners(element: any) {
|
|
||||||
const listeners = this._elementListeners.get(element);
|
|
||||||
if (listeners) {
|
|
||||||
const visitedTriggers = new Set<string>();
|
|
||||||
listeners.forEach(listener => {
|
|
||||||
const triggerName = listener.name;
|
|
||||||
if (visitedTriggers.has(triggerName)) return;
|
|
||||||
visitedTriggers.add(triggerName);
|
|
||||||
|
|
||||||
const trigger = this._triggers[triggerName];
|
|
||||||
const transition = trigger.fallbackTransition;
|
|
||||||
const elementStates = this._engine.statesByElement.get(element) !;
|
|
||||||
const fromState = elementStates[triggerName] || DEFAULT_STATE_VALUE;
|
|
||||||
const toState = new StateValue(VOID_VALUE);
|
|
||||||
const player = new TransitionAnimationPlayer(this.id, triggerName, element);
|
|
||||||
|
|
||||||
this._engine.totalQueuedPlayers++;
|
|
||||||
this._queue.push({
|
|
||||||
element,
|
|
||||||
triggerName,
|
|
||||||
transition,
|
|
||||||
fromState,
|
|
||||||
toState,
|
|
||||||
player,
|
|
||||||
isFallbackTransition: true
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
removeNode(element: any, context: any): void {
|
|
||||||
const engine = this._engine;
|
|
||||||
|
|
||||||
if (element.childElementCount) {
|
|
||||||
this._signalRemovalForInnerTriggers(element, context, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
// this means that a * => VOID animation was detected and kicked off
|
|
||||||
if (this.triggerLeaveAnimation(element, context, true)) return;
|
|
||||||
|
|
||||||
// find the player that is animating and make sure that the
|
// find the player that is animating and make sure that the
|
||||||
// removal is delayed until that player has completed
|
// removal is delayed until that player has completed
|
||||||
|
@ -416,7 +376,33 @@ export class AnimationTransitionNamespace {
|
||||||
// during flush or will be picked up by a parent query. Either way
|
// during flush or will be picked up by a parent query. Either way
|
||||||
// we need to fire the listeners for this element when it DOES get
|
// we need to fire the listeners for this element when it DOES get
|
||||||
// removed (once the query parent animation is done or after flush)
|
// removed (once the query parent animation is done or after flush)
|
||||||
this.prepareLeaveAnimationListeners(element);
|
const listeners = this._elementListeners.get(element);
|
||||||
|
if (listeners) {
|
||||||
|
const visitedTriggers = new Set<string>();
|
||||||
|
listeners.forEach(listener => {
|
||||||
|
const triggerName = listener.name;
|
||||||
|
if (visitedTriggers.has(triggerName)) return;
|
||||||
|
visitedTriggers.add(triggerName);
|
||||||
|
|
||||||
|
const trigger = this._triggers[triggerName];
|
||||||
|
const transition = trigger.fallbackTransition;
|
||||||
|
const elementStates = engine.statesByElement.get(element) !;
|
||||||
|
const fromState = elementStates[triggerName] || DEFAULT_STATE_VALUE;
|
||||||
|
const toState = new StateValue(VOID_VALUE);
|
||||||
|
const player = new TransitionAnimationPlayer(this.id, triggerName, element);
|
||||||
|
|
||||||
|
this._engine.totalQueuedPlayers++;
|
||||||
|
this._queue.push({
|
||||||
|
element,
|
||||||
|
triggerName,
|
||||||
|
transition,
|
||||||
|
fromState,
|
||||||
|
toState,
|
||||||
|
player,
|
||||||
|
isFallbackTransition: true
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
// whether or not a parent has an animation we need to delay the deferral of the leave
|
// whether or not a parent has an animation we need to delay the deferral of the leave
|
||||||
// operation until we have more information (which we do after flush() has been called)
|
// operation until we have more information (which we do after flush() has been called)
|
||||||
|
@ -479,7 +465,7 @@ export class AnimationTransitionNamespace {
|
||||||
|
|
||||||
destroy(context: any) {
|
destroy(context: any) {
|
||||||
this.players.forEach(p => p.destroy());
|
this.players.forEach(p => p.destroy());
|
||||||
this._signalRemovalForInnerTriggers(this.hostElement, context);
|
this._destroyInnerNodes(this.hostElement, context);
|
||||||
}
|
}
|
||||||
|
|
||||||
elementContainsData(element: any): boolean {
|
elementContainsData(element: any): boolean {
|
||||||
|
@ -682,7 +668,7 @@ export class TransitionAnimationEngine {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
removeNode(namespaceId: string, element: any, context: any): void {
|
removeNode(namespaceId: string, element: any, context: any, doNotRecurse?: boolean): void {
|
||||||
if (!isElementNode(element)) {
|
if (!isElementNode(element)) {
|
||||||
this._onRemovalComplete(element, context);
|
this._onRemovalComplete(element, context);
|
||||||
return;
|
return;
|
||||||
|
@ -690,7 +676,7 @@ export class TransitionAnimationEngine {
|
||||||
|
|
||||||
const ns = namespaceId ? this._fetchNamespace(namespaceId) : null;
|
const ns = namespaceId ? this._fetchNamespace(namespaceId) : null;
|
||||||
if (ns) {
|
if (ns) {
|
||||||
ns.removeNode(element, context);
|
ns.removeNode(element, context, doNotRecurse);
|
||||||
} else {
|
} else {
|
||||||
this.markElementAsRemoved(namespaceId, element, false, context);
|
this.markElementAsRemoved(namespaceId, element, false, context);
|
||||||
}
|
}
|
||||||
|
@ -722,15 +708,7 @@ export class TransitionAnimationEngine {
|
||||||
|
|
||||||
destroyInnerAnimations(containerElement: any) {
|
destroyInnerAnimations(containerElement: any) {
|
||||||
let elements = this.driver.query(containerElement, NG_TRIGGER_SELECTOR, true);
|
let elements = this.driver.query(containerElement, NG_TRIGGER_SELECTOR, true);
|
||||||
elements.forEach(element => this.destroyActiveAnimationsForElement(element));
|
elements.forEach(element => {
|
||||||
|
|
||||||
if (this.playersByQueriedElement.size == 0) return;
|
|
||||||
|
|
||||||
elements = this.driver.query(containerElement, NG_ANIMATING_SELECTOR, true);
|
|
||||||
elements.forEach(element => this.finishActiveQueriedAnimationOnElement(element));
|
|
||||||
}
|
|
||||||
|
|
||||||
destroyActiveAnimationsForElement(element: any) {
|
|
||||||
const players = this.playersByElement.get(element);
|
const players = this.playersByElement.get(element);
|
||||||
if (players) {
|
if (players) {
|
||||||
players.forEach(player => {
|
players.forEach(player => {
|
||||||
|
@ -748,13 +726,19 @@ export class TransitionAnimationEngine {
|
||||||
if (stateMap) {
|
if (stateMap) {
|
||||||
Object.keys(stateMap).forEach(triggerName => stateMap[triggerName] = DELETED_STATE_VALUE);
|
Object.keys(stateMap).forEach(triggerName => stateMap[triggerName] = DELETED_STATE_VALUE);
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
|
|
||||||
finishActiveQueriedAnimationOnElement(element: any) {
|
if (this.playersByQueriedElement.size == 0) return;
|
||||||
|
|
||||||
|
elements = this.driver.query(containerElement, NG_ANIMATING_SELECTOR, true);
|
||||||
|
if (elements.length) {
|
||||||
|
elements.forEach(element => {
|
||||||
const players = this.playersByQueriedElement.get(element);
|
const players = this.playersByQueriedElement.get(element);
|
||||||
if (players) {
|
if (players) {
|
||||||
players.forEach(player => player.finish());
|
players.forEach(player => player.finish());
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
whenRenderingDone(): Promise<any> {
|
whenRenderingDone(): Promise<any> {
|
||||||
|
|
|
@ -2380,62 +2380,6 @@ export function main() {
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should not cause a removal of inner @trigger DOM nodes when a parent animation occurs',
|
|
||||||
fakeAsync(() => {
|
|
||||||
@Component({
|
|
||||||
selector: 'ani-cmp',
|
|
||||||
template: `
|
|
||||||
<div @parent *ngIf="exp" class="parent">
|
|
||||||
this <div @child>child</div>
|
|
||||||
</div>
|
|
||||||
`,
|
|
||||||
animations: [
|
|
||||||
trigger(
|
|
||||||
'parent',
|
|
||||||
[
|
|
||||||
transition(
|
|
||||||
':leave',
|
|
||||||
[
|
|
||||||
style({opacity: 0}),
|
|
||||||
animate('1s', style({opacity: 1})),
|
|
||||||
]),
|
|
||||||
]),
|
|
||||||
trigger(
|
|
||||||
'child',
|
|
||||||
[
|
|
||||||
transition(
|
|
||||||
'* => something',
|
|
||||||
[
|
|
||||||
style({opacity: 0}),
|
|
||||||
animate('1s', style({opacity: 1})),
|
|
||||||
]),
|
|
||||||
]),
|
|
||||||
]
|
|
||||||
})
|
|
||||||
class Cmp {
|
|
||||||
public exp: boolean = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
TestBed.configureTestingModule({declarations: [Cmp]});
|
|
||||||
|
|
||||||
const fixture = TestBed.createComponent(Cmp);
|
|
||||||
const cmp = fixture.componentInstance;
|
|
||||||
|
|
||||||
cmp.exp = true;
|
|
||||||
fixture.detectChanges();
|
|
||||||
flushMicrotasks();
|
|
||||||
|
|
||||||
cmp.exp = false;
|
|
||||||
fixture.detectChanges();
|
|
||||||
flushMicrotasks();
|
|
||||||
|
|
||||||
const players = getLog();
|
|
||||||
expect(players.length).toEqual(1);
|
|
||||||
|
|
||||||
const element = players[0] !.element;
|
|
||||||
expect(element.innerText.trim()).toMatch(/this\s+child/mg);
|
|
||||||
}));
|
|
||||||
|
|
||||||
it('should only mark outermost *directive nodes :enter and :leave when inserts and removals occur',
|
it('should only mark outermost *directive nodes :enter and :leave when inserts and removals occur',
|
||||||
() => {
|
() => {
|
||||||
@Component({
|
@Component({
|
||||||
|
@ -2675,8 +2619,8 @@ export function main() {
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
flushMicrotasks();
|
flushMicrotasks();
|
||||||
expect(cmp.log).toEqual([
|
expect(cmp.log).toEqual([
|
||||||
'c1-start', 'c1-done', 'c2-start', 'c2-done', 'p-start', 'c3-start', 'c3-done',
|
'c1-start', 'c1-done', 'c2-start', 'c2-done', 'p-start', 'p-done', 'c3-start',
|
||||||
'p-done'
|
'c3-done'
|
||||||
]);
|
]);
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue