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

feat($location): allow to change location during $locationChangeStart #9678

Closed
wants to merge 1 commit into from
Closed

feat($location): allow to change location during $locationChangeStart #9678

wants to merge 1 commit into from

Conversation

shahata
Copy link
Contributor

@shahata shahata commented Oct 18, 2014

Closes #9607

@shahata
Copy link
Contributor Author

shahata commented Oct 18, 2014

@tbosch - While I'm afraid adding custom methods to scope events is currently not a mechanism that $rootScope enables, I believe this can be fixed by simply testing if the location was changed while the scope event was processed. So basically, all the developer needs to do is call one of the $location setters in the event handler in order to redirect to somewhere else (no need to call preventDefault in that case, it has no effect)

WDYT?

@@ -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) {
Copy link
Member

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)
:(

@jeffbcross jeffbcross modified the milestones: 1.3.1, 1.3.2 Oct 31, 2014
@caitp
Copy link
Contributor

caitp commented Nov 5, 2014

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

@shahata
Copy link
Contributor Author

shahata commented Nov 6, 2014

@caitp I've added the test case you requested

@petebacondarwin
Copy link
Contributor

Assigning to @pkozlowski-opensource as he is working on #9607

@caitp caitp modified the milestones: 1.3.2, 1.3.3 Nov 7, 2014
@pkozlowski-opensource
Copy link
Member

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?

@pkozlowski-opensource
Copy link
Member

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:

  • The current implementation creates an interesting corner case where the location change start event can be both prevented and someone might change the $location's URL - we should make it clear what happens in this case
  • As noted above, a test on the router level, something along those lines:
  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?

@shahata
Copy link
Contributor Author

shahata commented Nov 8, 2014

The current implementation creates an interesting corner case where the location change start event can be both prevented and someone might change the $location's URL - we should make it clear what happens in this case

@pkozlowski-opensource

I don't think there's a corner case here, if $location is changed, we will be redirected to that new address - It doesn't matter if you prevented the initial location change or not, it will have no effect. Or did you mean something else?

I added the router test you suggested, thanks.

@pkozlowski-opensource
Copy link
Member

@shahata

I don't think there's a corner case here, if $location is changed, we will be redirected to that new address - It doesn't matter if you prevented the initial location change or not, it will have no effect.

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.

@shahata
Copy link
Contributor Author

shahata commented Nov 9, 2014

@pkozlowski-opensource okay, added comment and test

@pkozlowski-opensource
Copy link
Member

Great @shahata !

@pkozlowski-opensource
Copy link
Member

@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']);
Copy link
Contributor

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?

Copy link
Contributor Author

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');
  }
});

@shahata
Copy link
Contributor Author

shahata commented Nov 14, 2014

Is there anything preventing us from merging this?

@petebacondarwin
Copy link
Contributor

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
Copy link
Contributor

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.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

petebacondarwin pushed a commit to petebacondarwin/angular.js that referenced this pull request Nov 15, 2014
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.