diff --git a/src/ng/rootScope.js b/src/ng/rootScope.js index 9922e1380819..508d56374600 100644 --- a/src/ng/rootScope.js +++ b/src/ng/rootScope.js @@ -778,20 +778,21 @@ function $RootScopeProvider(){ * clean up DOM bindings before an element is removed from the DOM. */ $destroy: function() { - // we can't destroy the root scope or a scope that has been already destroyed + // we can't destroy a scope that has been already destroyed if (this.$$destroyed) return; var parent = this.$parent; this.$broadcast('$destroy'); this.$$destroyed = true; - if (this === $rootScope) return; forEach(this.$$listenerCount, bind(null, decrementListenerCount, this)); // sever all the references to parent scopes (after this cleanup, the current scope should // not be retained by any of our references and should be eligible for garbage collection) - if (parent.$$childHead == this) parent.$$childHead = this.$$nextSibling; - if (parent.$$childTail == this) parent.$$childTail = this.$$prevSibling; + if (parent) { + if (parent.$$childHead == this) parent.$$childHead = this.$$nextSibling; + if (parent.$$childTail == this) parent.$$childTail = this.$$prevSibling; + } if (this.$$prevSibling) this.$$prevSibling.$$nextSibling = this.$$nextSibling; if (this.$$nextSibling) this.$$nextSibling.$$prevSibling = this.$$prevSibling; diff --git a/test/ng/rootScopeSpec.js b/test/ng/rootScopeSpec.js index c7717dc57469..ba72f14087dd 100644 --- a/test/ng/rootScopeSpec.js +++ b/test/ng/rootScopeSpec.js @@ -867,16 +867,32 @@ describe('Scope', function() { })); - it('should broadcast $destroy on rootScope', inject(function($rootScope) { + it('should allow $destroy of rootScope', inject(function($rootScope) { var spy = spyOn(angular, 'noop'); $rootScope.$on('$destroy', angular.noop); $rootScope.$destroy(); $rootScope.$digest(); - expect(log).toEqual('123'); + expect(log).toEqual(''); expect(spy).toHaveBeenCalled(); expect($rootScope.$$destroyed).toBe(true); })); + it('should remove all listeners after $destroy of rootScope', inject(function($rootScope) { + var spy = jasmine.createSpy('dummy'); + $rootScope.$on('dummy', spy); + $rootScope.$destroy(); + $rootScope.$broadcast('dummy'); + expect(spy).not.toHaveBeenCalled(); + })); + + it('should remove all watchers after $destroy of rootScope', inject(function($rootScope) { + var spy = jasmine.createSpy('dummy'); + var digest = $rootScope.$digest; + $rootScope.$watch(spy); + $rootScope.$destroy(); + digest.call($rootScope); + expect(spy).not.toHaveBeenCalled(); + })); it('should remove first', inject(function($rootScope) { first.$destroy();