Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

Commit 524650a

Browse files
committed
fix($animate): avoid accidentally matching substrings when resolving the presence of className tokens
1 parent 02a4582 commit 524650a

File tree

2 files changed

+36
-4
lines changed

2 files changed

+36
-4
lines changed

src/ngAnimate/animate.js

+5-4
Original file line numberDiff line numberDiff line change
@@ -647,10 +647,11 @@ angular.module('ngAnimate', ['ng'])
647647
return;
648648
}
649649

650+
var ONE_SPACE = ' ';
650651
//this value will be searched for class-based CSS className lookup. Therefore,
651652
//we prefix and suffix the current className value with spaces to avoid substring
652653
//lookups of className tokens
653-
var futureClassName = ' ' + currentClassName + ' ';
654+
var futureClassName = ONE_SPACE + currentClassName + ONE_SPACE;
654655
if(ngAnimateState.running) {
655656
//if an animation is currently running on the element then lets take the steps
656657
//to cancel that animation and fire any required callbacks
@@ -671,16 +672,16 @@ angular.module('ngAnimate', ['ng'])
671672
//will be invalid. Therefore the same string manipulation that would occur within the
672673
//DOM operation will be performed below so that the class comparison is valid...
673674
futureClassName = ngAnimateState.event == 'removeClass' ?
674-
futureClassName.replace(ngAnimateState.className, '') :
675-
futureClassName + ngAnimateState.className + ' ';
675+
futureClassName.replace(ONE_SPACE + ngAnimateState.className + ONE_SPACE, ONE_SPACE) :
676+
futureClassName + ngAnimateState.className + ONE_SPACE;
676677
}
677678
}
678679

679680
//There is no point in perform a class-based animation if the element already contains
680681
//(on addClass) or doesn't contain (on removeClass) the className being animated.
681682
//The reason why this is being called after the previous animations are cancelled
682683
//is so that the CSS classes present on the element can be properly examined.
683-
var classNameToken = ' ' + className + ' ';
684+
var classNameToken = ONE_SPACE + className + ONE_SPACE;
684685
if((animationEvent == 'addClass' && futureClassName.indexOf(classNameToken) >= 0) ||
685686
(animationEvent == 'removeClass' && futureClassName.indexOf(classNameToken) == -1)) {
686687
fireDOMOperation();

test/ngAnimate/animateSpec.js

+31
Original file line numberDiff line numberDiff line change
@@ -2665,6 +2665,37 @@ describe("ngAnimate", function() {
26652665
expect(element.hasClass('red')).toBe(true);
26662666
}));
26672667

2668+
it("should avoid mixing up substring classes during add and remove operations", function() {
2669+
var currentAnimation, currentFn;
2670+
module(function($animateProvider) {
2671+
$animateProvider.register('.on', function() {
2672+
return {
2673+
beforeAddClass : function(element, className, done) {
2674+
currentAnimation = 'addClass';
2675+
currentFn = done;
2676+
},
2677+
beforeRemoveClass : function(element, className, done) {
2678+
currentAnimation = 'removeClass';
2679+
currentFn = done;
2680+
}
2681+
};
2682+
});
2683+
});
2684+
inject(function($compile, $rootScope, $animate, $sniffer, $timeout) {
2685+
var element = $compile('<div class="animation-enabled only"></div>')($rootScope);
2686+
$rootElement.append(element);
2687+
jqLite($document[0].body).append($rootElement);
2688+
2689+
$animate.addClass(element, 'on');
2690+
expect(currentAnimation).toBe('addClass');
2691+
currentFn();
2692+
2693+
$animate.removeClass(element, 'on');
2694+
$animate.addClass(element, 'on');
2695+
2696+
expect(currentAnimation).toBe('addClass');
2697+
});
2698+
});
26682699

26692700
it('should enable and disable animations properly on the root element', function() {
26702701
var count = 0;

0 commit comments

Comments
 (0)