files/en-us/web/api/navigateevent/intercept/index.md
{{APIRef("Navigation API")}}
The intercept() method of the
{{domxref("NavigateEvent")}} interface intercepts this navigation, turning it into a same-document navigation to the {{domxref("NavigationDestination.url", "destination")}} URL.
intercept()
intercept(options)
options {{optional_inline}}
handler {{optional_inline}}
precommitHandler {{optional_inline}}
focusReset {{optional_inline}}
after-transition
autofocus attribute, or the {{htmlelement("body")}} element if no element has autofocus set. This is the default value.manual
scroll {{optional_inline}}
after-transition
manual
None (undefined).
InvalidStateError {{domxref("DOMException")}}
SecurityError {{domxref("DOMException")}}
false).precommitHandler() callback is provided on a non-cancelable event ({{domxref("Event.cancelable")}} is false).The intercept() method is used to implement same-document (SPA) navigation behavior when a navigation occurs; for example, when a link is clicked, a form is submitted, or a programmatic navigation is initiated (using {{domxref("History.pushState()")}}, {{domxref("Window.location")}}, etc.).
It does this via a couple of different callbacks, handler() and precommitHandler().
handler()The handler() callback is run in response to a committed navigation. It will run after the {{domxref("Navigation.currentEntry", "currentEntry")}} property has been updated, meaning that a new URL is shown in the browser UI and the history is updated with a new entry.
A typical example looks like this, enabling specific content to be rendered and loaded in response to a certain navigation:
navigation.addEventListener("navigate", (event) => {
const url = new URL(event.destination.url);
if (url.pathname.startsWith("/articles/")) {
event.intercept({
async handler() {
// Fetch the new content and display when ready
const articleContent = await getArticleContent(url.pathname);
renderArticlePage(articleContent);
},
});
}
// Include multiple conditions for different page types here, as needed
});
handler() should be used to implement navigation behavior where the navigation is committed to: the user should be shown something new.
precommitHandler()However, you might also wish to modify or cancel in-flight navigation, or to perform work while the navigation is ongoing and before it is committed. This kind of scenario can be dealt with using the precommitHandler() callback, which runs before the {{domxref("Navigation.currentEntry", "currentEntry")}} property has been updated and the browser UI shows the new location.
For example, if the user navigates to a restricted page and is not signed in, you may want to redirect the browser to a sign-in page. This might be handled like so:
navigation.addEventListener("navigate", (event) => {
const url = new URL(event.destination.url);
if (url.pathname.startsWith("/restricted/") && !userSignedIn) {
event.intercept({
async precommitHandler(controller) {
controller.redirect("/signin/", {
state: "signin-redirect",
history: "push",
});
},
});
}
});
This pattern is simpler than the alternative of canceling the original navigation and starting a new one to the redirect location, because it avoids exposing the intermediate state. For example, only one {{domxref("Navigation.navigatesuccess_event", "navigatesuccess")}} or {{domxref("Navigation.navigateerror_event", "navigateerror")}} event fires, and if the navigation was triggered by a call to {{domxref("Navigation.navigate()")}}, the promise only fulfills once the redirect destination is reached.
The precommitHandler() callback takes a {{domxref("NavigationPrecommitController")}} object as an argument, which contains a {{domxref("NavigationPrecommitController.redirect", "redirect()")}} method. The redirect() method takes two parameters — a string representing the URL to redirect to, and an optional options object than can specify state and history behavior.
precommitHandler() generally handles any modifications to the navigation behavior that are required before the destination URL is actually displayed in the browser, cancelling or redirecting it somewhere else as required.
[!NOTE] Because
precommitHandler()can be used to cancel navigations, it will only work as expected when the event's {{domxref("Event.cancelable")}} property istrue. Callingintercept()with aprecommitHandler()on a non-cancelable event results in aSecurityErrorbeing thrown.
precommitHandler()As we saw above, you can specify a handler() callback in the object passed to the intercept() method in order to preform actions after a navigation is is committed.
This approach works well if the actions required after commit do not depend on any actions run in the pre-commit phase.
If they do, then you can use {{domxref("NavigationPrecommitController.addHandler()")}} in precommitHandler() to dynamically add a handler that will run after the navigation commits.
For example, consider this code that extends the previous example for redirecting a logged-out user to a sign-in page.
The code uses addHandler() to add a post-commit handler callback that shows a message explaining the redirect reason.
Note that the handler only runs for the specific case of a redirect to the sign-in page.
navigation.addEventListener("navigate", (event) => {
const url = new URL(event.destination.url);
if (url.pathname.startsWith("/restricted/") && !userSignedIn) {
event.intercept({
async precommitHandler(controller) {
controller.redirect("/signin/", {
state: "signin-redirect",
history: "push",
});
// Use addHandler to trigger logic once the /signin/ page commits
controller.addHandler(() => {
showMessage("Please sign in to view that content.");
});
},
});
}
});
When the promises returned by the intercept() handler functions fulfill, the Navigation object's {{domxref("Navigation/navigatesuccess_event", "navigatesuccess")}} event fires, allowing you to run cleanup code after a successful navigation has completed. If those promises reject, meaning the navigation has failed, {{domxref("Navigation/navigateerror_event", "navigateerror")}} fires instead, allowing you to gracefully handle the failure case.
There is also a finished property on the return value of navigation methods (such as {{domxref("Navigation.navigate()")}}), which fulfills or rejects at the same time as the aforementioned events are fired, providing another path for handling the success and failure cases.
precommitHandler() and handler()Both precommitHandler() and handler() callbacks can be included inside the same intercept() call. In such cases, the order of operations is as follows:
First, the precommitHandler() handler runs.
precommitHandler() promise fulfills, the navigation commits.precommitHandler() rejects, the navigateerror event fires, the committed and finished promises reject, and the navigation is cancelled.When the navigation commits, a new {{domxref("NavigationHistoryEntry")}} is created for the navigation, and its committed promise fulfills.
Next, the handler() promise runs.
handler() promise fulfills and the navigatesuccess event fires, the navigation finished promise fulfills as well, to indicate the navigation is finished.handler() rejects, the navigateerror event fires, the finished promise rejects, and the navigation is canceled.Note that the above process is upheld even across multiple intercept() calls on the same NavigateEvent, and for handler() callbacks added in the precommitHandler().
All precommitHandler() callbacks are called first, and when all of them resolve, the navigation commits, and all the handler() callbacks are called.
By default, after a navigation handled using intercept() has occurred, the document focus will reset to the first element in the DOM with an autofocus attribute set, or otherwise to the {{htmlelement("body")}} element, if no autofocus attribute is set. If you want to override this behavior, to manually implement a more accessible focus position on navigation (for example, the new top-level heading), you can do so by setting the focusReset option to manual.
navigation.addEventListener("navigate", (event) => {
const url = new URL(event.destination.url);
if (url.pathname.startsWith("/articles/")) {
event.intercept({
focusReset: manual,
async handler() {
// Fetch the new content and display when ready
const articleContent = await getArticleContent(url.pathname);
renderArticlePage(articleContent);
// Handle page focus with a custom function
setPageFocus();
},
});
}
});
After an intercept() navigation has completed, the following scrolling behavior occurs:
push and replace navigations (see {{domxref("Navigation.navigate()")}}), the browser will attempt to scroll to the fragment given by event.destination.url. If there is no fragment available, it will reset the scroll position to the top of the page.push and replace navigations, but the browser delays its scroll restoration logic until the intercept() promise fulfills. It will perform no scroll restoration if the promise rejects. If the user has scrolled during the transition then no scroll restoration will be performed.If you want to turn this behavior off, you can do so by setting the scroll option to manual.
navigation.addEventListener("navigate", (event) => {
const url = new URL(event.destination.url);
if (url.pathname.startsWith("/articles/")) {
event.intercept({
scroll: manual,
async handler() {
// Fetch the new content and display when ready
const articleContent = await getArticleContent(url.pathname);
renderArticlePage(articleContent);
// Handle scroll behavior with a custom function
setScroll();
},
});
}
});
If you want to manually trigger the default scrolling behavior described earlier (maybe you want to reset the scroll position to the top of the page early, before the full navigation has finished), you can do so by calling {{domxref("NavigateEvent.scroll()")}}.
intercept()navigation.addEventListener("navigate", (event) => {
// Exit early if this navigation shouldn't be intercepted,
// e.g. if the navigation is cross-origin, or a download request
if (shouldNotIntercept(event)) return;
const url = new URL(event.destination.url);
if (url.pathname.startsWith("/articles/")) {
event.intercept({
async handler() {
// The URL has already changed, so show a placeholder while
// fetching the new content, such as a spinner or loading page
renderArticlePagePlaceholder();
// Fetch the new content and display when ready
const articleContent = await getArticleContent(url.pathname);
renderArticlePage(articleContent);
},
});
}
});
focusReset and scrollForm submission can be detected by querying for the {{domxref("NavigateEvent.formData")}} property. The following example turns any form submission into one which stays on the current page. In this case, you don't update the DOM, so you can cancel any default reset and scroll behavior using focusReset and scroll.
navigation.addEventListener("navigate", (event) => {
if (event.formData && event.canIntercept) {
// User submitted a POST form to a same-domain URL
// (If canIntercept is false, the event is just informative:
// you can't intercept this request, although you could
// likely still call .preventDefault() to stop it completely).
event.intercept({
// Since we don't update the DOM in this navigation,
// don't allow focus or scrolling to reset:
focusReset: "manual",
scroll: "manual",
async handler() {
await fetch(event.destination.url, {
method: "POST",
body: event.formData,
});
// You could navigate again with {history: 'replace'} to
// change the URL here, which might indicate "done"
},
});
}
});
{{Specifications}}
{{Compat}}