-
Notifications
You must be signed in to change notification settings - Fork 27.4k
feat($location): allow to change location during $locationChangeStart #9678
Conversation
@tbosch - While I'm afraid adding custom methods to scope events is currently not a mechanism that WDYT? |
4dd5a20
to
998c61c
Compare
@@ -1690,6 +1690,31 @@ describe('$location', function() { | |||
expect($browser.url()).toEqual('http://server/'); | |||
})); | |||
|
|||
it('should allow redirect during $locationChangeStart', inject(function($location, $browser, $rootScope, $log) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This line (and a few more beloew) is too long (> 100 characters)
:(
what I'd like to see is a case where a nested $locationChangeStart handler prevents default. if that case works right, then it's cool with me |
@caitp I've added the test case you requested |
Assigning to @pkozlowski-opensource as he is working on #9607 |
Looking at this PR right now and so far so good but I wonder if we shouldn't have a test where the $locationChangeStart is triggered by the routing system. Normally we should be fine, but since the original issue was opened for the routing, maybe we should play it on the safe side? |
OK, went over this patch once again and it LGTM based on my (limited) knowledge of the $location code. Having said this I would love to see 2 more tests:
it('should allow redirects while handling $routeChangeStart', function () {
module(function($routeProvider) {
$routeProvider.when('/some', {
id: 'some', template: 'Some functionality'
});
$routeProvider.when('/redirect', {
id: 'redirect'
});
});
module(provideLog);
inject(function($route, $location, $rootScope, $compile, log) {
$rootScope.$on('$routeChangeStart', function(event, next, current) {
if (next.id === 'some') {
$location.path('/redirect');
}
});
$compile('<div><div ng-view></div></div>')($rootScope);
$rootScope.$on('$routeChangeStart', log.fn('routeChangeStart'));
$rootScope.$on('$routeChangeError', log.fn('routeChangeError'));
$rootScope.$on('$routeChangeSuccess', log.fn('routeChangeSuccess'));
$rootScope.$apply(function() {
$location.path('/some');
});
expect($route.current.id).toBe('redirect');
expect($location.path()).toBe('/redirect');
expect(log).toEqual([ 'routeChangeStart', 'routeChangeStart', 'routeChangeSuccess']);
});
}); As noted above I'm not too familiar with the $location code / design so it would be awesome if someone with more experience in this area could have another look. @caitp @petebacondarwin @lgalfaso anyone? |
I don't think there's a corner case here, if I added the router test you suggested, thanks. |
This is what I've meant - that is - we should explicitly say what is a behaviour if an event is prevented and redirection happens while processing this event. This is why I was suggesting leaving a note in the code or better yet - a test - that makes this explicit. |
@pkozlowski-opensource okay, added comment and test |
Great @shahata ! |
@caitp @petebacondarwin this one LGTM now but it would be great to have another pair of eyes looking |
|
||
expect($route.current.id).toBe('redirect'); | ||
expect($location.path()).toBe('/redirect'); | ||
expect(log).toEqual(['routeChangeStart', 'routeChangeStart', 'routeChangeSuccess']); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
i feel like it would look weird if we emit $routeChangeStart without a matching $routeChangeError or $routeChangeSuccess --- do you think this will cause issues for people?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Doing the current workaround as suggested in the original issue will result in the exact same event sequence, so I don't see a problem with this...
The only difference is that instead of doing:
$rootScope.$on("$routeChangeStart", function (event, next, current) {
if (next && next.secure) {
event.preventDefault();
$rootScope.$evalAsync(function() {
$location.path('/c');
});
}
});
You would do this:
$rootScope.$on("$routeChangeStart", function (event, next, current) {
if (next && next.secure) {
$location.path('/c');
}
});
Is there anything preventing us from merging this? |
Just adding my eyes now! |
newState, oldState).defaultPrevented; | ||
|
||
//abort location change if location changed during event broadcast | ||
//in that case, it doesn't matter if preventDefaut was invoked or not |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
These comments don't quite read right to me... should they instead be:
// only process the location change if the location Url was not changed by a
// `$locationChangeStart` event handler.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
…geStart Closes angular#9607 Closes angular#9678
Closes #9607