From 83832bf4e5da0b9322ca1c5f09de6177d4b3705d Mon Sep 17 00:00:00 2001 From: Martin Staffa Date: Mon, 30 Nov 2015 12:23:05 +0100 Subject: [PATCH] fix(ngAnimate): don't normalize classes on follow-up animations to joined animations This allows follow-up animations to remove a class that is currently being added. Fixes #13339 Fixes #13380 Closes #13414 --- src/ngAnimate/animateQueue.js | 43 ++++++++++++++++++++++++++++--- test/ngAnimate/integrationSpec.js | 39 ++++++++++++++++++++++++++++ 2 files changed, 78 insertions(+), 4 deletions(-) diff --git a/src/ngAnimate/animateQueue.js b/src/ngAnimate/animateQueue.js index e024c831cf3c..2bd0fc3e067a 100644 --- a/src/ngAnimate/animateQueue.js +++ b/src/ngAnimate/animateQueue.js @@ -12,6 +12,14 @@ var $$AnimateQueueProvider = ['$animateProvider', function($animateProvider) { join: [] }; + function makeTruthyMapFromKeys(keys) { + var map = Object.create(null); + forEach(keys, function(key) { + map[key] = true; + }); + return map; + } + function isAllowed(ruleType, element, currentAnimation, previousAnimation) { return rules[ruleType].some(function(fn) { return fn(element, currentAnimation, previousAnimation); @@ -59,11 +67,38 @@ var $$AnimateQueueProvider = ['$animateProvider', function($animateProvider) { }); rules.cancel.push(function(element, newAnimation, currentAnimation) { - var nO = newAnimation.options; - var cO = currentAnimation.options; + var ONE_SPACE = ' '; + + var nA = newAnimation.options.addClass; + var nR = newAnimation.options.removeClass; + var cA = currentAnimation.options.addClass; + var cR = currentAnimation.options.removeClass; + + // early detection to save the global CPU shortage :) + if ((!isDefined(nA) && !isDefined(nR)) || (!isDefined(cA) && !isDefined(cR))) { + return false; + } + + var cancelSomething = false; + + cA = cA ? makeTruthyMapFromKeys(cA.split(ONE_SPACE)) : null; + cR = cR ? makeTruthyMapFromKeys(cR.split(ONE_SPACE)) : null; + + if (cR && nA) { + cancelSomething = nA.split(ONE_SPACE).some(function(className) { + return cR[className]; + }); + + if (cancelSomething) return true; + } + + if (cA && nR) { + cancelSomething = nR.split(ONE_SPACE).some(function(className) { + return cA[className]; + }); + } - // if the exact same CSS class is added/removed then it's safe to cancel it - return (nO.addClass && nO.addClass === cO.removeClass) || (nO.removeClass && nO.removeClass === cO.addClass); + return cancelSomething; }); this.$get = ['$$rAF', '$rootScope', '$rootElement', '$document', '$$HashMap', diff --git a/test/ngAnimate/integrationSpec.js b/test/ngAnimate/integrationSpec.js index 4a0610f2170a..8fa3e4f11d6f 100644 --- a/test/ngAnimate/integrationSpec.js +++ b/test/ngAnimate/integrationSpec.js @@ -387,6 +387,45 @@ describe('ngAnimate integration tests', function() { dealoc(element); })); + + + it("should not normalize classes in a follow-up animation to a joined class-based animation", + inject(function($animate, $animateCss, $rootScope, $document, $rootElement, $$rAF) { + + ss.addRule('.hide', 'opacity: 0'); + ss.addRule('.hide-add, .hide-remove', 'transition: 1s linear all'); + + jqLite($document[0].body).append($rootElement); + element = jqLite('
'); + $rootElement.append(element); + + // These animations will be joined together + $animate.addClass(element, 'red'); + $animate.addClass(element, 'hide'); + $rootScope.$digest(); + + expect(element).toHaveClass('red-add'); + expect(element).toHaveClass('hide-add'); + + // When a digest has passed, but no $rAF has been issued yet, .hide hasn't been added to + // the element, which means class normalization does not allow .hide to get removed + $animate.removeClass(element, 'hide'); + $rootScope.$digest(); + $$rAF.flush(); + + expect(element).not.toHaveClass('hide-add hide-add-active'); + expect(element).toHaveClass('hide-remove hide-remove-active'); + + //End the animation process + browserTrigger(element, 'transitionend', + { timeStamp: Date.now() + 1000, elapsedTime: 2 }); + $animate.flush(); + + expect(element).not.toHaveClass('hide-add-active red-add-active'); + expect(element).toHaveClass('red'); + expect(element).not.toHaveClass('hide'); + })); + }); describe('JS animations', function() {