最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - HTML5 History API: "popstate" not called when navigating back from another page - Stack Overflow

programmeradmin2浏览0评论

I have to maintain a site with an AJAX search form (for lets say books) and I want the search results to be a part of the browser history.

So I set the "popstate" event handler

window.addEventListener('popstate', function(e) {
    alert("popstate: " + JSON.stringify(e.state));
});

and call pushState() upon the successfull AJAX request:

var stateObj = { q: "harry potter" };
window.history.pushState(stateObj, "Result", "/search");

Then my clickflow is:

  1. Load the page with the search form - NORMAL request - path is /
  2. Enter a query string and submit the form - AJAX request - path changes to /search
  3. Click on a search result - NORMAL request - path changes to /books/harry-potter

  1. When I now go back by clicking the browser back button I would expect the "popstate" event to be triggered with the according state object. But nothing happens. - path changes to /search

  1. When I then go back one more time I get an alert with popstate: null - path changes to /

  2. And when I go forward after that, I get popstate: {"q":"harry potter"} - path changes to /search


So, the important popstate event (the one with the query string) gets only triggered when going forward, but not when going back.

Is this because im navigating back from a site, whose history entry has been created automatically instead of programmatically by pushState? But if so, the History API would only make sense for plete single page applications.

Behaviour is the same in any browser. Hope you can help me out :)

I have to maintain a site with an AJAX search form (for lets say books) and I want the search results to be a part of the browser history.

So I set the "popstate" event handler

window.addEventListener('popstate', function(e) {
    alert("popstate: " + JSON.stringify(e.state));
});

and call pushState() upon the successfull AJAX request:

var stateObj = { q: "harry potter" };
window.history.pushState(stateObj, "Result", "/search");

Then my clickflow is:

  1. Load the page with the search form - NORMAL request - path is /
  2. Enter a query string and submit the form - AJAX request - path changes to /search
  3. Click on a search result - NORMAL request - path changes to /books/harry-potter

  1. When I now go back by clicking the browser back button I would expect the "popstate" event to be triggered with the according state object. But nothing happens. - path changes to /search

  1. When I then go back one more time I get an alert with popstate: null - path changes to /

  2. And when I go forward after that, I get popstate: {"q":"harry potter"} - path changes to /search


So, the important popstate event (the one with the query string) gets only triggered when going forward, but not when going back.

Is this because im navigating back from a site, whose history entry has been created automatically instead of programmatically by pushState? But if so, the History API would only make sense for plete single page applications.

Behaviour is the same in any browser. Hope you can help me out :)

Share Improve this question edited Feb 22, 2018 at 11:44 jarvisccc asked Feb 21, 2018 at 18:12 jarviscccjarvisccc 711 silver badge3 bronze badges 6
  • Popstate only fires whenever the URL changes but you are still on the same page. Thus, when you press the back button from /books/harry-potter/, you simply load /search/ and no popstate event fires. So, a work-around is: on page load of /search/ you are going to have to read the URL and extract the search query. – Darryl Huffman Commented Feb 21, 2018 at 18:29
  • Thanks for your answer, but what do you mean with "only when the URL changes"? The URL is changing all the time (from / to /search, from /search to /books, etc.). – jarvisccc Commented Feb 21, 2018 at 18:57
  • 1 Say you're at /search/{query2} and press back... The URL will change to /search/{query1} and because the document context is still the same one as from /search/{query1}, a popstate event fires. In contrast, when you're at /books/harry-potter/ and press back, the URL will change and the entire document will be replaced with /search/{query2} - resulting in no popstate event to fire since it's a different documents context. If a user was on your search page & then went straight to google, then pressed back... that "back" action didn't happen on your webpage, so the popstate won't happen either – Darryl Huffman Commented Feb 21, 2018 at 21:19
  • Ah, its all about the document context, thank you! So the History API, as i mentioned in my question, indeed makes only sense for single page applications. I couldn't find any info on the web about that (important) point. Quite the opposite, the german mozilla docs are totally misleading: According to them there IS a popstate event fired when you navigate to a plete other page like google and go back. Even in the original english docs you can find this paragraph, but at least crossed out.... developer.mozilla/en-US/docs/Web/API/… – jarvisccc Commented Feb 22, 2018 at 11:36
  • @DarrylHuffman But your workaround (extracting the query from the URL) unfortunately wouldn't help me much, because javascript gets not called reliably across all browsers when navigating back/forth. – jarvisccc Commented Feb 22, 2018 at 11:43
 |  Show 1 more ment

2 Answers 2

Reset to default 9

For anyone wanting a little more explanation as to why the popstate event only occurs in certain scenarios, this helped me:

On the MDN popstate event page it states:

A popstate event is dispatched to the window each time the active history entry changes between two history entries for the same document.

and

The popstate event is only triggered by performing a browser action, such as clicking on the back button (or calling history.back() in JavaScript), when navigating between two history entries for the same document.

So it looks like popstate is only called when navigating to different urls within the same document. This works great for SPAs but not for anything else.

As an additional note, the html5 spec for session history doesn't seem to have much clarity as to when the popstate event is called:

The popstate event is fired in certain cases when navigating to a session history entry.

Which may indicate that different browsers will treat it differently.

I would love more insight from anyone else

There's at least one way to make it work, based on this excerpt from Mozilla's Working with the history API:

... When the page reloads, the page will receive an onload event, but no popstate event. However, if you read the history.state property, you'll get back the state object you would have gotten if a popstate had fired.

A minimal working example can be found in the snippet below.

Run the code snippet and follow these steps to see it in action:

  1. click "populate" to add some text to the display, dynamically (basically mock AJAX)
  2. click "away" to navigate to example (don't worry, this happens inside the iframe)
  3. click the browser's "back" button: the display is restored to its populated state
  4. click browser "back" again to clear the display (initial empty state)

const populate = document.getElementById('populate');
const display = document.getElementById('display');

// connect the button
populate.addEventListener('click', () => {
  const state = 'hit browser back button to clear';
  updateDisplay(state);
  // push state onto history stack
  history.pushState(state, '', '#populated');
});

// listen for popstate event, fired e.g. after using browser back button
window.addEventListener('popstate', (event) => updateDisplay(event.state));

// listen for load event, in case we navigate away and then go back
window.addEventListener('load', () => updateDisplay(history.state), false);

function updateDisplay(state) {
  if (state) {
    display.textContent = state;
  } else {
    display.textContent = '';
  }
}
/* the styling is only cosmetic */

.button {
  text-decoration: underline;
  cursor: pointer;
  color: blue;
}
<div class="button" id="populate">populate</div>
<a class="button" href="https://example">away</a>
<div id="display"></div>

As noted here:

The state object can be anything that can be serialized.

与本文相关的文章

发布评论

评论列表(0)

  1. 暂无评论