From 45472d50039479f8e5ab4706705e322040bca26c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Matias=20Niemel=C3=A4?= Date: Mon, 28 Oct 2013 10:43:41 -0600 Subject: [PATCH] fix($animate): ensure former nodes are fully cleaned up when a follow-up structural animation takes place Closes #4435 --- src/ngAnimate/animate.js | 12 ++++++------ test/ngAnimate/animateSpec.js | 36 +++++++++++++++++++++++++++++++++++ 2 files changed, 42 insertions(+), 6 deletions(-) diff --git a/src/ngAnimate/animate.js b/src/ngAnimate/animate.js index b2b88d0ac872..6b72f4a87628 100644 --- a/src/ngAnimate/animate.js +++ b/src/ngAnimate/animate.js @@ -588,6 +588,7 @@ angular.module('ngAnimate', ['ng']) //if an animation is currently running on the element then lets take the steps //to cancel that animation and fire any required callbacks $timeout.cancel(ngAnimateState.flagTimer); + cleanup(element); cancelAnimations(ngAnimateState.animations); (ngAnimateState.done || noop)(); } @@ -700,25 +701,24 @@ angular.module('ngAnimate', ['ng']) return rootAnimateState.disabled || rootAnimateState.running; } - var validState; do { //the element did not reach the root element which means that it //is not apart of the DOM. Therefore there is no reason to do //any animations on it - if(parent.length === 0) return true; + if(parent.length === 0) break; var isRoot = parent[0] == $rootElement[0]; var state = isRoot ? rootAnimateState : parent.data(NG_ANIMATE_STATE); - if(state && (state.disabled != null || state.running != null)) { - validState = state; - break; + var result = state && (!!state.disabled || !!state.running); + if(isRoot || result) { + return result; } if(isRoot) return true; } while(parent = parent.parent()); - return validState ? (validState.disabled || validState.running) : true; + return true; } }]); diff --git a/test/ngAnimate/animateSpec.js b/test/ngAnimate/animateSpec.js index c995cd838989..bc0d334785e2 100644 --- a/test/ngAnimate/animateSpec.js +++ b/test/ngAnimate/animateSpec.js @@ -1817,6 +1817,42 @@ describe("ngAnimate", function() { // expect(element.hasClass('hiding')).toBe(false); // }); // }); + it("should remove all the previous classes when the next animation is applied before a reflow", function() { + var fn, interceptedClass; + module(function($animateProvider) { + $animateProvider.register('.three', function() { + return { + move : function(element, done) { + fn = function() { + done(); + } + return function() { + interceptedClass = element.attr('class'); + } + } + } + }); + }); + inject(function($compile, $rootScope, $animate, $timeout) { + var parent = html($compile('
')($rootScope)); + var one = $compile('
')($rootScope); + var two = $compile('
')($rootScope); + var three = $compile('
')($rootScope); + + parent.append(one); + parent.append(two); + parent.append(three); + + $animate.move(three, null, two); + $rootScope.$digest(); + + $animate.move(three, null, one); + $rootScope.$digest(); + + //this means that the former animation was cleaned up before the new one starts + expect(interceptedClass.indexOf('ng-animate') >= 0).toBe(false); + }); + }); it("should provide the correct CSS class to the addClass and removeClass callbacks within a JS animation", function() { module(function($animateProvider) {