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( '