Skip to content

Commit 3be00df

Browse files
committed
fix($browser): detect changes to the browser url that happened in sync
Closes angular#6976.
1 parent 2efe1c2 commit 3be00df

File tree

4 files changed

+43
-4
lines changed

4 files changed

+43
-4
lines changed

src/ng/browser.js

+7
Original file line numberDiff line numberDiff line change
@@ -234,6 +234,13 @@ function Browser(window, document, $log, $sniffer) {
234234
return callback;
235235
};
236236

237+
/**
238+
* Checks whether the url has changed outside of Angular.
239+
* Needs to be exported to be able to check for changes that have been done in sync,
240+
* as hashchange/popstate events fire in async.
241+
*/
242+
self.$$checkUrlChange = fireUrlChange;
243+
237244
//////////////////////////////////////////////////////////////
238245
// Misc API
239246
//////////////////////////////////////////////////////////////

src/ng/rootScope.js

+2
Original file line numberDiff line numberDiff line change
@@ -689,6 +689,8 @@ function $RootScopeProvider(){
689689
logIdx, logMsg, asyncTask;
690690

691691
beginPhase('$digest');
692+
// Check for changes to browser url that happened in sync before the call to $digest
693+
$browser.$$checkUrlChange();
692694

693695
if (this === $rootScope && applyAsyncId !== null) {
694696
// If this is the root scope, and $applyAsync has scheduled a deferred $apply(), then

src/ngMock/angular-mocks.js

+2
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,8 @@ angular.mock.$Browser = function() {
5656
return listener;
5757
};
5858

59+
self.$$checkUrlChange = angular.noop;
60+
5961
self.cookieHash = {};
6062
self.lastCookieHash = {};
6163
self.deferredFns = [];

test/ng/browserSpecs.js

+32-4
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ function MockWindow() {
3636
};
3737

3838
this.location = {
39-
href: 'http://server',
39+
href: 'http://server/',
4040
replace: noop
4141
};
4242

@@ -419,7 +419,7 @@ describe('browser', function() {
419419

420420
expect(replaceState).not.toHaveBeenCalled();
421421
expect(locationReplace).not.toHaveBeenCalled();
422-
expect(fakeWindow.location.href).toEqual('http://server');
422+
expect(fakeWindow.location.href).toEqual('http://server/');
423423
});
424424

425425
it('should use history.replaceState when available', function() {
@@ -431,7 +431,7 @@ describe('browser', function() {
431431

432432
expect(pushState).not.toHaveBeenCalled();
433433
expect(locationReplace).not.toHaveBeenCalled();
434-
expect(fakeWindow.location.href).toEqual('http://server');
434+
expect(fakeWindow.location.href).toEqual('http://server/');
435435
});
436436

437437
it('should set location.href when pushState not available', function() {
@@ -453,7 +453,7 @@ describe('browser', function() {
453453

454454
expect(pushState).not.toHaveBeenCalled();
455455
expect(replaceState).not.toHaveBeenCalled();
456-
expect(fakeWindow.location.href).toEqual('http://server');
456+
expect(fakeWindow.location.href).toEqual('http://server/');
457457
});
458458

459459
it('should return $browser to allow chaining', function() {
@@ -620,4 +620,32 @@ describe('browser', function() {
620620
expect(browser.baseHref()).toEqual('/base/path/');
621621
});
622622
});
623+
624+
describe('integration tests with $location', function() {
625+
626+
beforeEach(module(function($provide, $locationProvider) {
627+
spyOn(fakeWindow.history, 'pushState').andCallFake(function(stateObj, title, newUrl) {
628+
fakeWindow.location.href = newUrl;
629+
});
630+
$provide.value('$browser', browser);
631+
browser.pollFns = [];
632+
633+
$locationProvider.html5Mode(true);
634+
}));
635+
636+
it('should update $location when it was changed outside of Angular in sync '+
637+
'before $digest was called', function() {
638+
inject(function($rootScope, $location) {
639+
fakeWindow.history.pushState(null, '', 'http://server/someTestHash');
640+
641+
// Verify that infinite digest reported in #6976 no longer occurs
642+
expect(function() {
643+
$rootScope.$digest();
644+
}).not.toThrow();
645+
646+
expect($location.path()).toBe('/someTestHash');
647+
});
648+
});
649+
});
650+
623651
});

0 commit comments

Comments
 (0)