-
Notifications
You must be signed in to change notification settings - Fork 27.4k
Bug: Changing the location outside of $location causes infdig and/or resets the original url #11075
Comments
I noticed something : wrapping the pushState inside a $timeout call workaround the issue. However, using $scope.$evalAsync doesn't. Weird. |
What's your use case for changing the location without the $location service, but inside an angular app? |
@Narretz I need the location service to use the History API (pushState / replaceState), but also the hash URLs. For example, using Now, as rational for this, I have to use the hash routing because my application is an offline app built around an AppCache, and updating an master entry (ie. page) from the appcache requires to explicitely list it inside the cache manifest. I cannot list every possible path inside this manifest, and so I have to keep a single master entry (index.html), and use the hash routing to load the correct url. Since Angular does not allow me to do this (enabling the HTML5 mode would disable the hash routing, and leaving the HTML5 mode would prevent me from using the state objects, which I need), I was relying on a custom $location service, which was working fine until I hit this bug. |
I think I'm getting into this same issue. I have a special application that deals with a special hash code, present in the URL. We use Inside this application there is a link to replace location that conditionally replaces location to another page of the same domain. We used "$window.location.replace()" and it was working until we updated from 1.2.26 to 1.2.28. I tried changing |
Ping @Narretz ? Is there enough info ? The best for me would be to be able to use the $location service as specified (hashbang routing, set with pushstate instead of directly in window.location.hash), but at the very least, solving the bug would be great. |
A plnkr (which we can download and test) is always helpful to get an idea of the problem. Generally, Angular currently doesn't play very well with custom history / location manipulation, so I refrain from calling this a bug for now. |
I have been running into the same issue when using ui-router and $state.go with simple url parmaters |
@Narretz The code in the first post is the true minimal showcase. Here is a plunker and its run page. But what I'd really like (and the rational is detailed in my previous post) is to be able to use both a hash-based url and the history API. I get this bug because I'm trying to workaround Angular by creating a new service with the described behavior, and I'd like to avoid that. |
@Narretz We have the following usecase for using window.location directly. We have a large angular application broken up into multiple single page applications utilizing ui-router internally. As such we have (for example). Now when we want to navigate from one application to another we are forced to use Doing so will trigger this exception in production with some regularity. Now, lets look at the official angular documentation: |
I see this same problem if I navigate using the URL bar in IE directly. navigating to #/database/home which is my controller to link the current user to a home page the controller in there works only once. Every subsequent attempt using either dynamic links, manually editing the URL in the browser or a navigate call from an ActiveX control (yes I want that dead but it's stuck in there now) results in this infinite digest loop. http://stackoverflow.com/questions/30776131/angular-controller-loads-only-the-first-time-in-ie |
I'm also facing the same issue where calling It only happens when I include the P.S. Using |
This is also fixed by #15561 (updated plnkr). |
Actually, it is not exactly fixed by #15561. #15561 fixes the infinite digest error, but in certain cases it still fails to update |
Previously, when the URL was changed directly (e.g. via `location.href`) during a `$digest` (e.g. via `scope.$evalAsync()` or `promise.then()`) the change was not handled correctly, unless a `popstate` or `hashchange` event was fired synchronously. This was an issue when calling `history.pushState()/replaceState()` in all browsers, since these methods do not emit any event. This was also an issue in IE11, where (unlike other browsers) no `popstate` event is fired at all for hash-only changes ([known bug][1]) and the `hashchange` event is fired asynchronously (which is too late). This commit fixes both usecases by: 1. Keeping track of `$location` setter methods being called and only processing a URL change if it originated from such a call. If there is a URL difference but no setter method has been called, this means that the browser URL/history has been updated directly and the change hasn't yet been propagated to `$location` (e.g. due to no event being fired synchronously or at all). 2. Checking for URL/state changes at the end of the `$digest`, in order to detect changes via `history` methods (that took place during the `$digest`). [1]: https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/3740423/ Fixes angular#11075 Fixes angular#12571 Fixes angular#15556
I updated #15561 to also fix this issue (although probably not in the way you might want 😁). |
Previously, when the URL was changed directly (e.g. via `location.href`) during a `$digest` (e.g. via `scope.$evalAsync()` or `promise.then()`) the change was not handled correctly, unless a `popstate` or `hashchange` event was fired synchronously. This was an issue when calling `history.pushState()/replaceState()` in all browsers, since these methods do not emit any event. This was also an issue when setting `location.href` in IE11, where (unlike other browsers) no `popstate` event is fired at all for hash-only changes ([known bug][1]) and the `hashchange` event is fired asynchronously (which is too late). This commit fixes both usecases by: 1. Keeping track of `$location` setter methods being called and only processing a URL change if it originated from such a call. If there is a URL difference but no setter method has been called, this means that the browser URL/history has been updated directly and the change hasn't yet been propagated to `$location` (e.g. due to no event being fired synchronously or at all). 2. Checking for URL/state changes at the end of the `$digest`, in order to detect changes via `history` methods (that took place during the `$digest`). [1]: https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/3740423/ Fixes angular#11075 Fixes angular#12571 Fixes angular#15556
Previously, when the URL was changed directly (e.g. via `location.href`) during a `$digest` (e.g. via `scope.$evalAsync()` or `promise.then()`) the change was not handled correctly, unless a `popstate` or `hashchange` event was fired synchronously. This was an issue when calling `history.pushState()/replaceState()` in all browsers, since these methods do not emit any event. This was also an issue when setting `location.href` in IE11, where (unlike other browsers) no `popstate` event is fired at all for hash-only changes ([known bug][1]) and the `hashchange` event is fired asynchronously (which is too late). This commit fixes both usecases by: 1. Keeping track of `$location` setter methods being called and only processing a URL change if it originated from such a call. If there is a URL difference but no setter method has been called, this means that the browser URL/history has been updated directly and the change hasn't yet been propagated to `$location` (e.g. due to no event being fired synchronously or at all). 2. Checking for URL/state changes at the end of the `$digest`, in order to detect changes via `history` methods (that took place during the `$digest`). [1]: https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/3740423/ Fixes #11075 Fixes #12571 Fixes #15556 Closes #15561
I have app that using history(outside angular). Angular app without html5mode.. so when url changed seriously it reloading page, otherwise it return old url. It's because initialUrl, appBase, appBaseNoFile have old value. I suggest add some reloadInitPath method to LocationHashbangUrl or app or take them from $rootScope. |
Previously, when the URL was changed directly (e.g. via `location.href`) during a `$digest` (e.g. via `scope.$evalAsync()` or `promise.then()`) the change was not handled correctly, unless a `popstate` or `hashchange` event was fired synchronously. This was an issue when calling `history.pushState()/replaceState()` in all browsers, since these methods do not emit any event. This was also an issue when setting `location.href` in IE11, where (unlike other browsers) no `popstate` event is fired at all for hash-only changes ([known bug][1]) and the `hashchange` event is fired asynchronously (which is too late). This commit fixes both usecases by: 1. Keeping track of `$location` setter methods being called and only processing a URL change if it originated from such a call. If there is a URL difference but no setter method has been called, this means that the browser URL/history has been updated directly and the change hasn't yet been propagated to `$location` (e.g. due to no event being fired synchronously or at all). 2. Checking for URL/state changes at the end of the `$digest`, in order to detect changes via `history` methods (that took place during the `$digest`). [1]: https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/3740423/ Fixes angular#11075 Fixes angular#12571 Fixes angular#15556 Closes angular#15561
Just want to say thank you, we've had a bug in our system for two years that was unfixable, updates to the address via $location.search() were getting discarded on one module and we couldn't figure out why, the only thing we knew was that a POST was running in the same Digest cycle, we used a $timeout to workaround it as well. Thanks to this issue I found the exact code in angular that handles the address updating and realised that angular-block-ui was preventing the location change from happening because the POST was triggering it. Two years and many grey hairs later IT'S FIXED! thanks guys! |
@sktocha found a solution for your problem? |
Hi,
As said in the title, if the location is changed from outside the $location service (but inside a digest cycle, and especially inside a $q.then callback), then Angular will break and there is two possible outcome, apparently random (maybe some kind of race condition?) :
Here is a minimal showcase to reproduce this issue :
The text was updated successfully, but these errors were encountered: