From c49b8a2db5d916a9213547125af996d6c853230c Mon Sep 17 00:00:00 2001 From: Vojta Jina Date: Fri, 13 Jan 2012 16:38:08 -0800 Subject: fix($location): do not $digest if browser's url change fired within $apply/$digest Chrome (probably other browsers as well) fires 'hashchange' event synchronously, so if you change raw location from within $apply/$digest, we don't want to $apply twice. (It would throw an exception) --- test/service/locationSpec.js | 37 +++++++++++++++++++++++++++++++++++++ 1 file changed, 37 insertions(+) (limited to 'test') diff --git a/test/service/locationSpec.js b/test/service/locationSpec.js index c7cc9c23..97f2ab63 100644 --- a/test/service/locationSpec.js +++ b/test/service/locationSpec.js @@ -364,6 +364,43 @@ describe('$location', function() { })); + // location.href = '...' fires hashchange event synchronously, so it might happen inside $apply + it('should not $apply when browser url changed inside $apply', inject( + function($rootScope, $browser, $location) { + var OLD_URL = $browser.url(), + NEW_URL = 'http://updated.com/url'; + + + $rootScope.$apply(function() { + $browser.url(NEW_URL); + $browser.poll(); // simulate firing event from browser + expect($location.absUrl()).toBe(OLD_URL); // should be async + }); + + expect($location.absUrl()).toBe(NEW_URL); + })); + + // location.href = '...' fires hashchange event synchronously, so it might happen inside $digest + it('should not $apply when browser url changed inside $digest', inject( + function($rootScope, $browser, $location) { + var OLD_URL = $browser.url(), + NEW_URL = 'http://updated.com/url', + notRunYet = true; + + $rootScope.$watch(function() { + if (notRunYet) { + notRunYet = false; + $browser.url(NEW_URL); + $browser.poll(); // simulate firing event from browser + expect($location.absUrl()).toBe(OLD_URL); // should be async + } + }); + + $rootScope.$digest(); + expect($location.absUrl()).toBe(NEW_URL); + })); + + it('should update browser when $location changes', inject(function($rootScope, $browser, $location) { var $browserUrl = spyOnlyCallsWithArgs($browser, 'url').andCallThrough(); $location.path('/new/path'); -- cgit v1.2.3