diff --git a/angularFiles.js b/angularFiles.js index f40b03ec241c..9d18fd831b6c 100755 --- a/angularFiles.js +++ b/angularFiles.js @@ -93,6 +93,7 @@ var angularFiles = { 'ngAnimate': [ 'src/ngAnimate/shared.js', 'src/ngAnimate/body.js', + 'src/ngAnimate/rafScheduler.js', 'src/ngAnimate/animateChildrenDirective.js', 'src/ngAnimate/animateCss.js', 'src/ngAnimate/animateCssDriver.js', diff --git a/src/ngAnimate/animateCss.js b/src/ngAnimate/animateCss.js index 01140640f041..f15929614246 100644 --- a/src/ngAnimate/animateCss.js +++ b/src/ngAnimate/animateCss.js @@ -328,8 +328,10 @@ var $AnimateCssProvider = ['$animateProvider', function($animateProvider) { var gcsLookup = createLocalCacheLookup(); var gcsStaggerLookup = createLocalCacheLookup(); - this.$get = ['$window', '$$jqLite', '$$AnimateRunner', '$timeout', '$$forceReflow', '$sniffer', '$$rAF', '$animate', - function($window, $$jqLite, $$AnimateRunner, $timeout, $$forceReflow, $sniffer, $$rAF, $animate) { + this.$get = ['$window', '$$jqLite', '$$AnimateRunner', '$timeout', + '$$forceReflow', '$sniffer', '$$rAFScheduler', '$animate', + function($window, $$jqLite, $$AnimateRunner, $timeout, + $$forceReflow, $sniffer, $$rAFScheduler, $animate) { var applyAnimationClasses = applyAnimationClassesFactory($$jqLite); @@ -389,12 +391,8 @@ var $AnimateCssProvider = ['$animateProvider', function($animateProvider) { var cancelLastRAFRequest; var rafWaitQueue = []; function waitUntilQuiet(callback) { - if (cancelLastRAFRequest) { - cancelLastRAFRequest(); //cancels the request - } rafWaitQueue.push(callback); - cancelLastRAFRequest = $$rAF(function() { - cancelLastRAFRequest = null; + $$rAFScheduler.waitUntilQuiet(function() { gcsLookup.flush(); gcsStaggerLookup.flush(); @@ -485,7 +483,6 @@ var $AnimateCssProvider = ['$animateProvider', function($animateProvider) { // there actually is a detected transition or keyframe animation if (options.applyClassesEarly && addRemoveClassName.length) { applyAnimationClasses(element, options); - addRemoveClassName = ''; } var preparationClasses = [structuralClassName, addRemoveClassName].join(' ').trim(); diff --git a/src/ngAnimate/animateCssDriver.js b/src/ngAnimate/animateCssDriver.js index c47a27a85c23..12afe34abc76 100644 --- a/src/ngAnimate/animateCssDriver.js +++ b/src/ngAnimate/animateCssDriver.js @@ -22,13 +22,13 @@ var $$AnimateCssDriverProvider = ['$$animationProvider', function($$animationPro var applyAnimationClasses = applyAnimationClassesFactory($$jqLite); - return function initDriverFn(animationDetails, onBeforeClassesAppliedCb) { + return function initDriverFn(animationDetails) { return animationDetails.from && animationDetails.to ? prepareFromToAnchorAnimation(animationDetails.from, animationDetails.to, animationDetails.classes, animationDetails.anchors) - : prepareRegularAnimation(animationDetails, onBeforeClassesAppliedCb); + : prepareRegularAnimation(animationDetails); }; function filterCssClasses(classes) { @@ -224,21 +224,14 @@ var $$AnimateCssDriverProvider = ['$$animationProvider', function($$animationPro }; } - function prepareRegularAnimation(animationDetails, onBeforeClassesAppliedCb) { + function prepareRegularAnimation(animationDetails) { var element = animationDetails.element; var options = animationDetails.options || {}; - // since the ng-EVENT, class-ADD and class-REMOVE classes are applied inside - // of the animateQueue pre and postDigest stages then there is no need to add - // then them here as well. - options.$$skipPreparationClasses = true; - - // during the pre/post digest stages inside of animateQueue we also performed - // the blocking (transition:-9999s) so there is no point in doing that again. - options.skipBlocking = true; - if (animationDetails.structural) { options.event = animationDetails.event; + options.structural = true; + options.applyClassesEarly = true; // we special case the leave animation since we want to ensure that // the element is removed as soon as the animation is over. Otherwise @@ -248,11 +241,6 @@ var $$AnimateCssDriverProvider = ['$$animationProvider', function($$animationPro } } - // we apply the classes right away since the pre-digest took care of the - // preparation classes. - onBeforeClassesAppliedCb(element); - applyAnimationClasses(element, options); - // We assign the preparationClasses as the actual animation event since // the internals of $animateCss will just suffix the event token values // with `-active` to trigger the animation. diff --git a/src/ngAnimate/animateQueue.js b/src/ngAnimate/animateQueue.js index c76376d1948d..6337ec0c03c2 100644 --- a/src/ngAnimate/animateQueue.js +++ b/src/ngAnimate/animateQueue.js @@ -381,9 +381,6 @@ var $$AnimateQueueProvider = ['$animateProvider', function($animateProvider) { return runner; } - applyGeneratedPreparationClasses(element, isStructural ? event : null, options); - blockTransitions(node, SAFE_FAST_FORWARD_DURATION_VALUE); - // the counter keeps track of cancelled animations var counter = (existingAnimation.counter || 0) + 1; newAnimation.counter = counter; @@ -442,10 +439,7 @@ var $$AnimateQueueProvider = ['$animateProvider', function($animateProvider) { : animationDetails.event; markElementAnimationState(element, RUNNING_STATE); - var realRunner = $$animation(element, event, animationDetails.options, function(e) { - $$forceReflow(); - blockTransitions(getDomNode(e), false); - }); + var realRunner = $$animation(element, event, animationDetails.options); realRunner.done(function(status) { close(!status); diff --git a/src/ngAnimate/animation.js b/src/ngAnimate/animation.js index afb2ac5e060d..c0deb035f790 100644 --- a/src/ngAnimate/animation.js +++ b/src/ngAnimate/animation.js @@ -19,8 +19,8 @@ var $$AnimationProvider = ['$animateProvider', function($animateProvider) { return element.data(RUNNER_STORAGE_KEY); } - this.$get = ['$$jqLite', '$rootScope', '$injector', '$$AnimateRunner', '$$HashMap', - function($$jqLite, $rootScope, $injector, $$AnimateRunner, $$HashMap) { + this.$get = ['$$jqLite', '$rootScope', '$injector', '$$AnimateRunner', '$$HashMap', '$$rAFScheduler', + function($$jqLite, $rootScope, $injector, $$AnimateRunner, $$HashMap, $$rAFScheduler) { var animationQueue = []; var applyAnimationClasses = applyAnimationClassesFactory($$jqLite); @@ -88,11 +88,11 @@ var $$AnimationProvider = ['$animateProvider', function($animateProvider) { if (remainingLevelEntries <= 0) { remainingLevelEntries = nextLevelEntries; nextLevelEntries = 0; - result = result.concat(row); + result.push(row); row = []; } row.push(entry.fn); - forEach(entry.children, function(childEntry) { + entry.children.forEach(function(childEntry) { nextLevelEntries++; queue.push(childEntry); }); @@ -100,14 +100,15 @@ var $$AnimationProvider = ['$animateProvider', function($animateProvider) { } if (row.length) { - result = result.concat(row); + result.push(row); } + return result; } } // TODO(matsko): document the signature in a better way - return function(element, event, options, onBeforeClassesAppliedCb) { + return function(element, event, options) { options = prepareAnimationOptions(options); var isStructural = ['enter', 'move', 'leave'].indexOf(event) >= 0; @@ -159,8 +160,7 @@ var $$AnimationProvider = ['$animateProvider', function($animateProvider) { // the element was destroyed early on which removed the runner // form its storage. This means we can't animate this element // at all and it already has been closed due to destruction. - var elm = entry.element; - if (getRunner(elm) && getDomNode(elm).parentNode) { + if (getRunner(entry.element)) { animations.push(entry); } else { entry.close(); @@ -191,7 +191,7 @@ var $$AnimationProvider = ['$animateProvider', function($animateProvider) { : animationEntry.element; if (getRunner(targetElement)) { - var operation = invokeFirstDriver(animationEntry, onBeforeClassesAppliedCb); + var operation = invokeFirstDriver(animationEntry); if (operation) { startAnimationFn = operation.start; } @@ -211,11 +211,9 @@ var $$AnimationProvider = ['$animateProvider', function($animateProvider) { }); // we need to sort each of the animations in order of parent to child - // relationships. This ensures that the parent to child classes are - // applied at the right time. - forEach(sortAnimations(toBeSortedAnimations), function(triggerAnimation) { - triggerAnimation(); - }); + // relationships. This ensures that the child classes are applied at the + // right time. + $$rAFScheduler(sortAnimations(toBeSortedAnimations)); }); return runner; @@ -285,7 +283,7 @@ var $$AnimationProvider = ['$animateProvider', function($animateProvider) { var lookupKey = from.animationID.toString(); if (!anchorGroups[lookupKey]) { var group = anchorGroups[lookupKey] = { - // TODO(matsko): double-check this code + structural: true, beforeStart: function() { fromAnimation.beforeStart(); toAnimation.beforeStart(); @@ -339,7 +337,7 @@ var $$AnimationProvider = ['$animateProvider', function($animateProvider) { return matches.join(' '); } - function invokeFirstDriver(animationDetails, onBeforeClassesAppliedCb) { + function invokeFirstDriver(animationDetails) { // we loop in reverse order since the more general drivers (like CSS and JS) // may attempt more elements, but custom drivers are more particular for (var i = drivers.length - 1; i >= 0; i--) { @@ -347,7 +345,7 @@ var $$AnimationProvider = ['$animateProvider', function($animateProvider) { if (!$injector.has(driverName)) continue; // TODO(matsko): remove this check var factory = $injector.get(driverName); - var driver = factory(animationDetails, onBeforeClassesAppliedCb); + var driver = factory(animationDetails); if (driver) { return driver; } diff --git a/src/ngAnimate/module.js b/src/ngAnimate/module.js index 6f7e9e1118f4..d48dd73e8d9c 100644 --- a/src/ngAnimate/module.js +++ b/src/ngAnimate/module.js @@ -4,6 +4,7 @@ $$BodyProvider, $$AnimateAsyncRunFactory, + $$rAFSchedulerFactory, $$AnimateChildrenDirective, $$AnimateRunnerFactory, $$AnimateQueueProvider, @@ -744,6 +745,7 @@ angular.module('ngAnimate', []) .provider('$$body', $$BodyProvider) .directive('ngAnimateChildren', $$AnimateChildrenDirective) + .factory('$$rAFScheduler', $$rAFSchedulerFactory) .factory('$$AnimateRunner', $$AnimateRunnerFactory) .factory('$$animateAsyncRun', $$AnimateAsyncRunFactory) diff --git a/src/ngAnimate/rafScheduler.js b/src/ngAnimate/rafScheduler.js new file mode 100644 index 000000000000..84cf6c4b594c --- /dev/null +++ b/src/ngAnimate/rafScheduler.js @@ -0,0 +1,50 @@ +'use strict'; + +var $$rAFSchedulerFactory = ['$$rAF', function($$rAF) { + var queue, cancelFn; + + function scheduler(tasks) { + // we make a copy since RAFScheduler mutates the state + // of the passed in array variable and this would be difficult + // to track down on the outside code + queue = queue.concat(tasks); + nextTick(); + } + + queue = scheduler.queue = []; + + /* waitUntilQuiet does two things: + * 1. It will run the FINAL `fn` value only when an uncancelled RAF has passed through + * 2. It will delay the next wave of tasks from running until the quiet `fn` has run. + * + * The motivation here is that animation code can request more time from the scheduler + * before the next wave runs. This allows for certain DOM properties such as classes to + * be resolved in time for the next animation to run. + */ + scheduler.waitUntilQuiet = function(fn) { + if (cancelFn) cancelFn(); + + cancelFn = $$rAF(function() { + cancelFn = null; + fn(); + nextTick(); + }); + }; + + return scheduler; + + function nextTick() { + if (!queue.length) return; + + var items = queue.shift(); + for (var i = 0; i < items.length; i++) { + items[i](); + } + + if (!cancelFn) { + $$rAF(function() { + if (!cancelFn) nextTick(); + }); + } + } +}]; diff --git a/test/ngAnimate/animateCssDriverSpec.js b/test/ngAnimate/animateCssDriverSpec.js index 62184a7893eb..589d5932e3ff 100644 --- a/test/ngAnimate/animateCssDriverSpec.js +++ b/test/ngAnimate/animateCssDriverSpec.js @@ -105,7 +105,7 @@ describe("ngAnimate $$animateCssDriver", function() { expect(capturedAnimation[1].applyClassesEarly).toBeFalsy(); driver({ element: element, structural: true }); - expect(capturedAnimation[1].applyClassesEarly).toBeFalsy(); + expect(capturedAnimation[1].applyClassesEarly).toBeTruthy(); })); it("should only set the event value if the animation is structural", inject(function() { diff --git a/test/ngAnimate/animateSpec.js b/test/ngAnimate/animateSpec.js index ce897fa0b48e..06e06a867cfa 100644 --- a/test/ngAnimate/animateSpec.js +++ b/test/ngAnimate/animateSpec.js @@ -492,74 +492,6 @@ describe("animations", function() { expect(element).not.toHaveClass('green'); })); - they('should apply the $prop CSS class to the element before digest for the given event and remove when complete', - {'ng-enter': 'enter', 'ng-leave': 'leave', 'ng-move': 'move'}, function(event) { - - inject(function($animate, $rootScope, $document, $rootElement) { - $animate.enabled(true); - - var element = jqLite('
'); - var parent = jqLite('
'); - - $rootElement.append(parent); - jqLite($document[0].body).append($rootElement); - - var runner; - if (event === 'leave') { - parent.append(element); - runner = $animate[event](element); - } else { - runner = $animate[event](element, parent); - } - - var expectedClassName = 'ng-' + event; - - expect(element).toHaveClass(expectedClassName); - - $rootScope.$digest(); - expect(element).toHaveClass(expectedClassName); - - runner.end(); - expect(element).not.toHaveClass(expectedClassName); - - dealoc(parent); - }); - }); - - they('should add CSS classes with the $prop suffix when depending on the event and remove when complete', - {'-add': 'add', '-remove': 'remove'}, function(event) { - - inject(function($animate, $rootScope, $document, $rootElement) { - $animate.enabled(true); - - var element = jqLite('
'); - - $rootElement.append(element); - jqLite($document[0].body).append($rootElement); - - var classes = 'one two'; - var expectedClasses = ['one-',event,' ','two-', event].join(''); - - var runner; - if (event === 'add') { - runner = $animate.addClass(element, classes); - } else { - element.addClass(classes); - runner = $animate.removeClass(element, classes); - } - - expect(element).toHaveClass(expectedClasses); - - $rootScope.$digest(); - expect(element).toHaveClass(expectedClasses); - - runner.end(); - expect(element).not.toHaveClass(expectedClasses); - - dealoc(element); - }); - }); - they('$prop() should operate using a native DOM element', ['enter', 'move', 'leave', 'addClass', 'removeClass', 'setClass', 'animate'], function(event) { diff --git a/test/ngAnimate/animationSpec.js b/test/ngAnimate/animationSpec.js index b1062ff8893c..8a2dc13f52a6 100644 --- a/test/ngAnimate/animationSpec.js +++ b/test/ngAnimate/animationSpec.js @@ -297,6 +297,90 @@ describe('$$animation', function() { }; })); + it('should space out multiple ancestorial class-based animations with a RAF in between', + inject(function($rootScope, $$animation, $$rAF) { + + var parent = element; + element = jqLite('
'); + parent.append(element); + + var child = jqLite('
'); + element.append(child); + + $$animation(parent, 'addClass', { addClass: 'blue' }); + $$animation(element, 'addClass', { addClass: 'red' }); + $$animation(child, 'addClass', { addClass: 'green' }); + + $rootScope.$digest(); + + expect(captureLog.length).toBe(1); + expect(capturedAnimation.options.addClass).toBe('blue'); + + $$rAF.flush(); + expect(captureLog.length).toBe(2); + expect(capturedAnimation.options.addClass).toBe('red'); + + $$rAF.flush(); + expect(captureLog.length).toBe(3); + expect(capturedAnimation.options.addClass).toBe('green'); + })); + + it('should properly cancel out pending animations that are spaced with a RAF request before the digest completes', + inject(function($rootScope, $$animation, $$rAF) { + + var parent = element; + element = jqLite('
'); + parent.append(element); + + var child = jqLite('
'); + element.append(child); + + var r1 = $$animation(parent, 'addClass', { addClass: 'blue' }); + var r2 = $$animation(element, 'addClass', { addClass: 'red' }); + var r3 = $$animation(child, 'addClass', { addClass: 'green' }); + + r2.end(); + + $rootScope.$digest(); + + expect(captureLog.length).toBe(1); + expect(capturedAnimation.options.addClass).toBe('blue'); + + $$rAF.flush(); + + expect(captureLog.length).toBe(2); + expect(capturedAnimation.options.addClass).toBe('green'); + })); + + it('should properly cancel out pending animations that are spaced with a RAF request after the digest completes', + inject(function($rootScope, $$animation, $$rAF) { + + var parent = element; + element = jqLite('
'); + parent.append(element); + + var child = jqLite('
'); + element.append(child); + + var r1 = $$animation(parent, 'addClass', { addClass: 'blue' }); + var r2 = $$animation(element, 'addClass', { addClass: 'red' }); + var r3 = $$animation(child, 'addClass', { addClass: 'green' }); + + $rootScope.$digest(); + + r2.end(); + + expect(captureLog.length).toBe(1); + expect(capturedAnimation.options.addClass).toBe('blue'); + + $$rAF.flush(); + expect(captureLog.length).toBe(1); + + $$rAF.flush(); + expect(captureLog.length).toBe(2); + expect(capturedAnimation.options.addClass).toBe('green'); + })); + they('should return a runner that object that contains a $prop() function', ['end', 'cancel', 'then'], function(method) { inject(function($$animation) { @@ -407,7 +491,7 @@ describe('$$animation', function() { })); it('should always sort parent-element animations to run in order of parent-to-child DOM structure', - inject(function($$animation, $rootScope) { + inject(function($$animation, $rootScope, $animate) { var child = jqLite('
'); var grandchild = jqLite('
'); @@ -423,6 +507,9 @@ describe('$$animation', function() { $rootScope.$digest(); + $animate.flush(); // element -> child + $animate.flush(); // child -> grandchild + expect(captureLog[0].element).toBe(element); expect(captureLog[1].element).toBe(child); expect(captureLog[2].element).toBe(grandchild); @@ -542,7 +629,7 @@ describe('$$animation', function() { })); it("should not group animations into an anchored animation if enter/leave events are NOT used", - inject(function($$animation, $rootScope) { + inject(function($$animation, $rootScope, $$rAF) { fromElement.addClass('shared-class'); fromElement.attr('ng-animate-ref', '1'); @@ -557,6 +644,7 @@ describe('$$animation', function() { }); $rootScope.$digest(); + $$rAF.flush(); expect(captureLog.length).toBe(2); })); @@ -701,7 +789,7 @@ describe('$$animation', function() { })); it('should prepare a parent-element animation to run first before the anchored animation', - inject(function($$animation, $rootScope, $rootElement) { + inject(function($$animation, $rootScope, $rootElement, $animate) { fromAnchors[0].attr('ng-animate-ref', 'shared'); toAnchors[0].attr('ng-animate-ref', 'shared'); @@ -724,6 +812,7 @@ describe('$$animation', function() { expect(captureLog.length).toBe(0); $rootScope.$digest(); + $animate.flush(); expect(captureLog[0].element).toBe(parent); expect(captureLog[1].from.element).toBe(fromElement); diff --git a/test/ngAnimate/integrationSpec.js b/test/ngAnimate/integrationSpec.js index 21ab965f0f8e..5e1e335b72f4 100644 --- a/test/ngAnimate/integrationSpec.js +++ b/test/ngAnimate/integrationSpec.js @@ -140,40 +140,7 @@ describe('ngAnimate integration tests', function() { dealoc(element); })); - it('should always synchronously add css classes in order for child animations to animate properly', - inject(function($animate, $compile, $rootScope, $rootElement, $document) { - - ss.addRule('.animations-enabled .animate-me.ng-enter', 'transition:2s linear all;'); - - element = jqLite('
'); - var child = jqLite('
'); - - element.append(child); - $rootElement.append(element); - jqLite($document[0].body).append($rootElement); - - $compile(element)($rootScope); - - $rootScope.exp = true; - $rootScope.$digest(); - - child = element.find('div'); - - expect(element).toHaveClass('animations-enabled'); - expect(child).toHaveClass('ng-enter'); - - $animate.flush(); - - expect(child).toHaveClass('ng-enter-active'); - - browserTrigger(child, 'transitionend', { timeStamp: Date.now(), elapsedTime: 2 }); - $animate.flush(); - - expect(child).not.toHaveClass('ng-enter-active'); - expect(child).not.toHaveClass('ng-enter'); - })); - - it('should synchronously add/remove ng-class expressions in time for other animations to run on the same element', + it('should include the added/removed classes in lieu of the enter animation', inject(function($animate, $compile, $rootScope, $rootElement, $document) { ss.addRule('.animate-me.ng-enter.on', 'transition:2s linear all;'); @@ -254,9 +221,9 @@ describe('ngAnimate integration tests', function() { expect(child).not.toHaveClass('expand-add'); })); - it('should issue a reflow for each element animation on all DOM levels', function() { + it('should issue a RAF for each element animation on all DOM levels', function() { module('ngAnimateMock'); - inject(function($animate, $compile, $rootScope, $rootElement, $document) { + inject(function($animate, $compile, $rootScope, $rootElement, $document, $$rAF) { element = jqLite( '
' + '
' + @@ -272,70 +239,84 @@ describe('ngAnimate integration tests', function() { $compile(element)($rootScope); $rootScope.$digest(); - expect($animate.reflows).toBe(0); + + var outer = element; + var inner = element.find('div'); $rootScope.exp = true; $rootScope.items = [1,2,3,4,5,6,7,8,9,10]; $rootScope.$digest(); + expect(outer).not.toHaveClass('parent'); + expect(inner).not.toHaveClass('parent2'); - // 2 parents + 10 items = 12 - expect($animate.reflows).toBe(12); - }); - }); + assertTotalRepeats(0); - it('should issue a reflow for each element and also its children', function() { - module('ngAnimateMock'); - inject(function($animate, $compile, $rootScope, $rootElement, $document) { - element = jqLite( - '
' + - '
' + - '
' + - '
' + - '
' + - '
' + - '
' + - '
' + - '
' + - '
' - ); + $$rAF.flush(); + expect(outer).toHaveClass('parent'); - $rootElement.append(element); - jqLite($document[0].body).append($rootElement); + assertTotalRepeats(0); - $compile(element)($rootScope); - $rootScope.$digest(); - expect($animate.reflows).toBe(0); + $$rAF.flush(); + expect(inner).toHaveClass('parent2'); - $rootScope.exp = true; - $rootScope.$digest(); + assertTotalRepeats(10); - // there is one element's expression in there that is false - expect($animate.reflows).toBe(6); + function assertTotalRepeats(total) { + expect(inner[0].querySelectorAll('div.ng-enter').length).toBe(total); + } }); }); - it('should always issue atleast one reflow incase there are no parent class-based animations', function() { + it('should pack level elements into their own RAF flush', function() { module('ngAnimateMock'); inject(function($animate, $compile, $rootScope, $rootElement, $document) { + ss.addRule('.inner', 'transition:2s linear all;'); + element = jqLite( - '
' + - '{{ item }}' + + '
' + + '
' + + '
' + + '
' + + '
' + + '
' + + '
' + + '
' + + '
' + + '
' + + '
' + '
' ); $rootElement.append(element); jqLite($document[0].body).append($rootElement); - $compile(element)($rootScope); $rootScope.$digest(); - expect($animate.reflows).toBe(0); + + assertGroupHasClass(query('outer'), 'on', true); + expect(query('inner').length).toBe(0); $rootScope.exp = true; - $rootScope.items = [1,2,3,4,5,6,7,8,9,10]; $rootScope.$digest(); - expect($animate.reflows).toBe(10); + assertGroupHasClass(query('outer'), 'on', true); + assertGroupHasClass(query('inner'), 'ng-enter', true); + + $animate.flush(); + + assertGroupHasClass(query('outer'), 'on'); + assertGroupHasClass(query('inner'), 'ng-enter'); + + function query(className) { + return element[0].querySelectorAll('.' + className); + } + + function assertGroupHasClass(elms, className, not) { + for (var i = 0; i < elms.length; i++) { + var assert = expect(jqLite(elms[i])); + (not ? assert.not : assert).toHaveClass(className); + } + } }); }); }); @@ -470,6 +451,7 @@ describe('ngAnimate integration tests', function() { } $rootScope.$digest(); + $animate.flush(); expect(endParentAnimationFn).toBeTruthy(); @@ -534,6 +516,7 @@ describe('ngAnimate integration tests', function() { } $rootScope.$digest(); + $animate.flush(); expect(endParentAnimationFn).toBeTruthy(); diff --git a/test/ngAnimate/rafSchedulerSpec.js b/test/ngAnimate/rafSchedulerSpec.js new file mode 100644 index 000000000000..16c7906534a4 --- /dev/null +++ b/test/ngAnimate/rafSchedulerSpec.js @@ -0,0 +1,143 @@ +'use strict'; + +describe("$$rAFScheduler", function() { + + beforeEach(module('ngAnimate')); + + it('should accept an array of tasks and run the first task immediately', + inject(function($$rAFScheduler) { + + var taskSpy = jasmine.createSpy(); + var tasks = [taskSpy]; + $$rAFScheduler([tasks]); + expect(taskSpy).toHaveBeenCalled(); + })); + + it('should run tasks based on how many RAFs have run in comparison to the task index', + inject(function($$rAFScheduler, $$rAF) { + + var i, tasks = []; + + for (i = 0; i < 5; i++) { + tasks.push([jasmine.createSpy()]); + } + + $$rAFScheduler(tasks); + + for (i = 1; i < 5; i++) { + var taskSpy = tasks[i][0]; + expect(taskSpy).not.toHaveBeenCalled(); + $$rAF.flush(); + expect(taskSpy).toHaveBeenCalled(); + } + })); + + it('should space out subarrays by a RAF and run the internals in parallel', + inject(function($$rAFScheduler, $$rAF) { + + var spies = { + a: jasmine.createSpy(), + b: jasmine.createSpy(), + c: jasmine.createSpy(), + + x: jasmine.createSpy(), + y: jasmine.createSpy(), + z: jasmine.createSpy() + }; + + var items = [[spies.a, spies.x], + [spies.b, spies.y], + [spies.c, spies.z]]; + + expect(spies.a).not.toHaveBeenCalled(); + expect(spies.x).not.toHaveBeenCalled(); + + $$rAFScheduler(items); + + expect(spies.a).toHaveBeenCalled(); + expect(spies.x).toHaveBeenCalled(); + + + expect(spies.b).not.toHaveBeenCalled(); + expect(spies.y).not.toHaveBeenCalled(); + + $$rAF.flush(); + + expect(spies.b).toHaveBeenCalled(); + expect(spies.y).toHaveBeenCalled(); + + + expect(spies.c).not.toHaveBeenCalled(); + expect(spies.z).not.toHaveBeenCalled(); + + $$rAF.flush(); + + expect(spies.c).toHaveBeenCalled(); + expect(spies.z).toHaveBeenCalled(); + })); + + describe('.waitUntilQuiet', function() { + + it('should run the `last` provided function when a RAF fully passes', + inject(function($$rAFScheduler, $$rAF) { + + var q1 = jasmine.createSpy(); + $$rAFScheduler.waitUntilQuiet(q1); + + expect(q1).not.toHaveBeenCalled(); + + var q2 = jasmine.createSpy(); + $$rAFScheduler.waitUntilQuiet(q2); + + expect(q1).not.toHaveBeenCalled(); + expect(q2).not.toHaveBeenCalled(); + + var q3 = jasmine.createSpy(); + $$rAFScheduler.waitUntilQuiet(q3); + + expect(q1).not.toHaveBeenCalled(); + expect(q2).not.toHaveBeenCalled(); + expect(q3).not.toHaveBeenCalled(); + + $$rAF.flush(); + + expect(q1).not.toHaveBeenCalled(); + expect(q2).not.toHaveBeenCalled(); + expect(q3).toHaveBeenCalled(); + })); + + it('should always execute itself before the next RAF task tick occurs', function() { + module(provideLog); + inject(function($$rAFScheduler, $$rAF, log) { + var quietFn = log.fn('quiet'); + var tasks = [ + [log.fn('task1')], + [log.fn('task2')], + [log.fn('task3')], + [log.fn('task4')] + ]; + + $$rAFScheduler(tasks); + expect(log).toEqual(['task1']); + + $$rAFScheduler.waitUntilQuiet(quietFn); + expect(log).toEqual(['task1']); + + $$rAF.flush(); + + expect(log).toEqual(['task1', 'quiet', 'task2']); + + $$rAF.flush(); + + expect(log).toEqual(['task1', 'quiet', 'task2', 'task3']); + + $$rAFScheduler.waitUntilQuiet(quietFn); + + $$rAF.flush(); + + expect(log).toEqual(['task1', 'quiet', 'task2', 'task3', 'quiet', 'task4']); + }); + }); + }); + +}); diff --git a/test/ngRoute/directive/ngViewSpec.js b/test/ngRoute/directive/ngViewSpec.js index a2601d5dd335..39ab1a934fea 100644 --- a/test/ngRoute/directive/ngViewSpec.js +++ b/test/ngRoute/directive/ngViewSpec.js @@ -749,7 +749,7 @@ describe('ngView animations', function() { ); it('should render ngClass on ngView', - inject(function($compile, $rootScope, $templateCache, $animate, $location, $timeout) { + inject(function($compile, $rootScope, $templateCache, $animate, $location) { var item; $rootScope.tpl = 'one'; @@ -759,6 +759,7 @@ describe('ngView animations', function() { $location.path('/foo'); $rootScope.$digest(); + $animate.flush(); //we don't care about the enter animation $animate.queue.shift(); @@ -775,6 +776,8 @@ describe('ngView animations', function() { expect($animate.queue.shift().event).toBe('addClass'); expect($animate.queue.shift().event).toBe('removeClass'); + $animate.flush(); + expect(item.hasClass('classy')).toBe(false); expect(item.hasClass('boring')).toBe(true);