From 393850086fbff30f5f04026e70f8143c9c21f06f Mon Sep 17 00:00:00 2001 From: Andy Joslin Date: Wed, 26 Mar 2014 20:48:53 -0600 Subject: [PATCH] feat($ionicModal): show/hide/remove return promises for animating --- js/ext/angular/src/service/ionicModal.js | 51 +++++++++++-------- .../angular/test/service/ionicModal.unit.js | 31 ++++++++++- 2 files changed, 60 insertions(+), 22 deletions(-) diff --git a/js/ext/angular/src/service/ionicModal.js b/js/ext/angular/src/service/ionicModal.js index b4cc1b01876..340072fa429 100644 --- a/js/ext/angular/src/service/ionicModal.js +++ b/js/ext/angular/src/service/ionicModal.js @@ -41,8 +41,15 @@ angular.module('ionic.service.modal', ['ionic.service.templateLoad', 'ionic.serv * }); * ``` */ -.factory('$ionicModal', ['$rootScope', '$document', '$compile', '$timeout', '$ionicPlatform', '$ionicTemplateLoader', - function( $rootScope, $document, $compile, $timeout, $ionicPlatform, $ionicTemplateLoader) { +.factory('$ionicModal', [ + '$rootScope', + '$document', + '$compile', + '$timeout', + '$ionicPlatform', + '$ionicTemplateLoader', + '$q', +function($rootScope, $document, $compile, $timeout, $ionicPlatform, $ionicTemplateLoader, $q) { /** * @ngdoc controller @@ -76,17 +83,15 @@ angular.module('ionic.service.modal', ['ionic.service.templateLoad', 'ionic.serv * @ngdoc method * @name ionicModal#show * @description Show this modal instance. + * @returns {promise} A promise which is resolved when the modal is finished animating in. */ show: function() { var self = this; var modalEl = angular.element(self.modalEl); self.el.classList.remove('hide'); - $document[0].body.classList.add('modal-open'); - self._isShown = true; - if(!self.el.parentElement) { modalEl.addClass(self.animation); $document[0].body.appendChild(self.el); @@ -95,28 +100,32 @@ angular.module('ionic.service.modal', ['ionic.service.templateLoad', 'ionic.serv modalEl.addClass('ng-enter active') .removeClass('ng-leave ng-leave-active'); - $timeout(function(){ - modalEl.addClass('ng-enter-active'); - self.scope.$parent && self.scope.$parent.$broadcast('modal.shown'); - self.el.classList.add('active'); - }, 20); + self._isShown = true; self._deregisterBackButton = $ionicPlatform.registerBackButtonAction(function(){ self.hide(); }, 200); + self._isOpenPromise = $q.defer(); ionic.views.Modal.prototype.show.call(self); + $timeout(function(){ + modalEl.addClass('ng-enter-active'); + self.scope.$parent && self.scope.$parent.$broadcast('modal.shown'); + self.el.classList.add('active'); + }, 20); + + return $timeout(angular.noop, 400); }, /** * @ngdoc method * @name ionicModal#hide * @description Hide this modal instance. + * @returns {promise} A promise which is resolved when the modal is finished animating out. */ hide: function() { var self = this; - self._isShown = false; var modalEl = angular.element(self.modalEl); self.el.classList.remove('active'); @@ -127,32 +136,32 @@ angular.module('ionic.service.modal', ['ionic.service.templateLoad', 'ionic.serv .removeClass('ng-enter ng-enter-active active'); }, 20); - $timeout(function(){ - $document[0].body.classList.remove('modal-open'); - self.el.classList.add('hide'); - }, 350); + self._isShown = false; + self.scope.$parent && self.scope.$parent.$broadcast('modal.hidden'); + self._deregisterBackButton && self._deregisterBackButton(); ionic.views.Modal.prototype.hide.call(self); - self.scope.$parent && self.scope.$parent.$broadcast('modal.hidden'); - - self._deregisterBackButton && self._deregisterBackButton(); + return $timeout(function(){ + $document[0].body.classList.remove('modal-open'); + self.el.classList.add('hide'); + }, 350); }, /** * @ngdoc method * @name ionicModal#remove * @description Remove this modal instance from the DOM and clean up. + * @returns {promise} A promise which is resolved when the modal is finished animating out. */ remove: function() { var self = this; - self.hide(); self.scope.$parent && self.scope.$parent.$broadcast('modal.removed'); - $timeout(function(){ + return self.hide().then(function() { self.scope.$destroy(); self.el && self.el.parentElement && self.el.parentElement.removeChild(self.el); - }, 750); + }); }, /** diff --git a/js/ext/angular/test/service/ionicModal.unit.js b/js/ext/angular/test/service/ionicModal.unit.js index e9f9f9eb6bc..58be2eb9140 100644 --- a/js/ext/angular/test/service/ionicModal.unit.js +++ b/js/ext/angular/test/service/ionicModal.unit.js @@ -1,4 +1,4 @@ -describe('Ionic Modal', function() { +ddescribe('Ionic Modal', function() { var modal, q, timeout, ionicPlatform, rootScope; beforeEach(module('ionic.service.modal')); @@ -133,4 +133,33 @@ describe('Ionic Modal', function() { expect(m.scope.$parent.$broadcast).toHaveBeenCalledWith('modal.removed'); timeout.flush(); })); + + it('show should return a promise resolved on hide', function() { + var template = ''; + var m = modal.fromTemplate(template, {}); + var done = false; + + m.hide().then(function() { + done = true; + }); + expect(m.el.classList.contains('hide')).toBe(false); + timeout.flush(); + expect(m.el.classList.contains('hide')).toBe(true); + expect(done).toBe(true); + }); + + it('show should return a promise resolved on remove', function() { + var template = ''; + var m = modal.fromTemplate(template, {}); + var done = false; + + m.remove().then(function() { + done = true; + }); + spyOn(m.scope, '$destroy'); + timeout.flush(); + expect(m.scope.$destroy).toHaveBeenCalled(); + expect(done).toBe(true); + }); + });