Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

Commit 5f1c6cb

Browse files
committed
fix($browser): detect changes to the browser url that happened in sync
Closes #6976.
1 parent 5f90340 commit 5f1c6cb

File tree

4 files changed

+44
-4
lines changed

4 files changed

+44
-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
@@ -687,6 +687,8 @@ function $RootScopeProvider(){
687687
logIdx, logMsg, asyncTask;
688688

689689
beginPhase('$digest');
690+
// Check for changes to browser url that happened in sync before the call to $digest
691+
$browser.$$checkUrlChange();
690692

691693
lastDirtyWatch = null;
692694

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

+33-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() {
@@ -587,6 +587,7 @@ describe('browser', function() {
587587
fakeWindow.fire('hashchange');
588588
expect(callback).not.toHaveBeenCalled();
589589
});
590+
590591
});
591592

592593

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

0 commit comments

Comments
 (0)