mirror of
https://github.com/withastro/astro.git
synced 2025-01-20 22:12:38 -05:00
Increase consistency between navigation with and without ViewTransitions (#9279)
* Greater consistency between navigations with and without ViewTransitions * hmpf * formatting and test fixing * An attempt to explain * fixed tests * explain the event listener * Update .changeset/nervous-beans-peel.md Co-authored-by: Nate Moore <natemoo-re@users.noreply.github.com> --------- Co-authored-by: Matthew Phillips <matthew@skypack.dev> Co-authored-by: Nate Moore <natemoo-re@users.noreply.github.com>
This commit is contained in:
parent
8a228fce01
commit
6a9669b810
4 changed files with 65 additions and 65 deletions
6
.changeset/nervous-beans-peel.md
Normal file
6
.changeset/nervous-beans-peel.md
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
---
|
||||||
|
'astro': minor
|
||||||
|
---
|
||||||
|
|
||||||
|
Improves consistency between navigations with and without `<ViewTransitions>`. See [#9279](https://github.com/withastro/astro/pull/9279) for more details.
|
||||||
|
|
|
@ -51,3 +51,9 @@ import Layout from '../components/Layout.astro';
|
||||||
Libero id faucibus nisl tincidunt eget nullam non. Faucibus a pellentesque sit amet porttitor eget dolor. Posuere urna nec tincidunt praesent semper feugiat nibh sed. Suspendisse ultrices gravida dictum fusce. Porttitor eget dolor morbi non arcu. Neque egestas congue quisque egestas diam in. Suscipit tellus mauris a diam maecenas sed enim ut sem. Luctus accumsan tortor posuere ac. Tortor posuere ac ut consequat semper viverra. Egestas tellus rutrum tellus pellentesque eu tincidunt tortor aliquam nulla. Senectus et netus et malesuada fames ac turpis egestas. Sed libero enim sed faucibus turpis in eu mi bibendum. Sollicitudin tempor id eu nisl nunc mi.
|
Libero id faucibus nisl tincidunt eget nullam non. Faucibus a pellentesque sit amet porttitor eget dolor. Posuere urna nec tincidunt praesent semper feugiat nibh sed. Suspendisse ultrices gravida dictum fusce. Porttitor eget dolor morbi non arcu. Neque egestas congue quisque egestas diam in. Suscipit tellus mauris a diam maecenas sed enim ut sem. Luctus accumsan tortor posuere ac. Tortor posuere ac ut consequat semper viverra. Egestas tellus rutrum tellus pellentesque eu tincidunt tortor aliquam nulla. Senectus et netus et malesuada fames ac turpis egestas. Sed libero enim sed faucibus turpis in eu mi bibendum. Sollicitudin tempor id eu nisl nunc mi.
|
||||||
</article>
|
</article>
|
||||||
</Layout>
|
</Layout>
|
||||||
|
<script is:inline>
|
||||||
|
// let playwright know when navigate() is done
|
||||||
|
document.addEventListener('astro:before-swap', (e) => {
|
||||||
|
e.viewTransition.ready.then(()=>console.log("ready"))
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
|
@ -394,7 +394,10 @@ test.describe('View Transitions', () => {
|
||||||
await expect(locator).toBeInViewport();
|
await expect(locator).toBeInViewport();
|
||||||
|
|
||||||
// Scroll back to top
|
// Scroll back to top
|
||||||
|
// back returns immediately, but we need to wait for navigate() to complete
|
||||||
|
const waitForReady = page.waitForEvent('console');
|
||||||
await page.goBack();
|
await page.goBack();
|
||||||
|
await waitForReady;
|
||||||
locator = page.locator('#longpage');
|
locator = page.locator('#longpage');
|
||||||
await expect(locator).toBeInViewport();
|
await expect(locator).toBeInViewport();
|
||||||
|
|
||||||
|
|
|
@ -25,9 +25,7 @@ export const transitionEnabledOnThisPage = () =>
|
||||||
inBrowser && !!document.querySelector('[name="astro-view-transitions-enabled"]');
|
inBrowser && !!document.querySelector('[name="astro-view-transitions-enabled"]');
|
||||||
|
|
||||||
const samePage = (thisLocation: URL, otherLocation: URL) =>
|
const samePage = (thisLocation: URL, otherLocation: URL) =>
|
||||||
thisLocation.origin === otherLocation.origin &&
|
thisLocation.pathname === otherLocation.pathname && thisLocation.search === otherLocation.search;
|
||||||
thisLocation.pathname === otherLocation.pathname &&
|
|
||||||
thisLocation.search === otherLocation.search;
|
|
||||||
|
|
||||||
// When we traverse the history, the window.location is already set to the new location.
|
// When we traverse the history, the window.location is already set to the new location.
|
||||||
// This variable tells us where we came from
|
// This variable tells us where we came from
|
||||||
|
@ -418,16 +416,22 @@ async function transition(
|
||||||
options: Options,
|
options: Options,
|
||||||
historyState?: State
|
historyState?: State
|
||||||
) {
|
) {
|
||||||
|
// not ours
|
||||||
|
if (!transitionEnabledOnThisPage() || location.origin !== to.origin) {
|
||||||
|
location.href = to.href;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const navigationType = historyState
|
const navigationType = historyState
|
||||||
? 'traverse'
|
? 'traverse'
|
||||||
: options.history === 'replace'
|
: options.history === 'replace'
|
||||||
? 'replace'
|
? 'replace'
|
||||||
: 'push';
|
: 'push';
|
||||||
|
|
||||||
if (samePage(from, to) && !options.formData /* not yet: && to.hash*/) {
|
if (navigationType !== 'traverse') {
|
||||||
if (navigationType !== 'traverse') {
|
updateScrollPosition({ scrollX, scrollY });
|
||||||
updateScrollPosition({ scrollX, scrollY });
|
}
|
||||||
}
|
if (samePage(from, to) && !!to.hash) {
|
||||||
moveToLocation(to, from, options, historyState);
|
moveToLocation(to, from, options, historyState);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -447,61 +451,48 @@ async function transition(
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
function pageMustReload(preparationEvent: TransitionBeforePreparationEvent) {
|
|
||||||
return (
|
|
||||||
preparationEvent.to.hash === '' ||
|
|
||||||
!samePage(preparationEvent.from, preparationEvent.to) ||
|
|
||||||
preparationEvent.sourceElement instanceof HTMLFormElement
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
async function defaultLoader(preparationEvent: TransitionBeforePreparationEvent) {
|
async function defaultLoader(preparationEvent: TransitionBeforePreparationEvent) {
|
||||||
if (pageMustReload(preparationEvent)) {
|
const href = preparationEvent.to.href;
|
||||||
const href = preparationEvent.to.href;
|
const init: RequestInit = {};
|
||||||
const init: RequestInit = {};
|
if (preparationEvent.formData) {
|
||||||
if (preparationEvent.formData) {
|
init.method = 'POST';
|
||||||
init.method = 'POST';
|
init.body = preparationEvent.formData;
|
||||||
init.body = preparationEvent.formData;
|
}
|
||||||
}
|
const response = await fetchHTML(href, init);
|
||||||
const response = await fetchHTML(href, init);
|
// If there is a problem fetching the new page, just do an MPA navigation to it.
|
||||||
// If there is a problem fetching the new page, just do an MPA navigation to it.
|
if (response === null) {
|
||||||
if (response === null) {
|
preparationEvent.preventDefault();
|
||||||
preparationEvent.preventDefault();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// if there was a redirection, show the final URL in the browser's address bar
|
|
||||||
if (response.redirected) {
|
|
||||||
preparationEvent.to = new URL(response.redirected);
|
|
||||||
}
|
|
||||||
|
|
||||||
parser ??= new DOMParser();
|
|
||||||
|
|
||||||
preparationEvent.newDocument = parser.parseFromString(response.html, response.mediaType);
|
|
||||||
// The next line might look like a hack,
|
|
||||||
// but it is actually necessary as noscript elements
|
|
||||||
// and their contents are returned as markup by the parser,
|
|
||||||
// see https://developer.mozilla.org/en-US/docs/Web/API/DOMParser/parseFromString
|
|
||||||
preparationEvent.newDocument.querySelectorAll('noscript').forEach((el) => el.remove());
|
|
||||||
|
|
||||||
// If ViewTransitions is not enabled on the incoming page, do a full page load to it.
|
|
||||||
// Unless this was a form submission, in which case we do not want to trigger another mutation.
|
|
||||||
if (
|
|
||||||
!preparationEvent.newDocument.querySelector('[name="astro-view-transitions-enabled"]') &&
|
|
||||||
!preparationEvent.formData
|
|
||||||
) {
|
|
||||||
preparationEvent.preventDefault();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const links = preloadStyleLinks(preparationEvent.newDocument);
|
|
||||||
links.length && (await Promise.all(links));
|
|
||||||
|
|
||||||
if (import.meta.env.DEV)
|
|
||||||
await prepareForClientOnlyComponents(preparationEvent.newDocument, preparationEvent.to);
|
|
||||||
} else {
|
|
||||||
preparationEvent.newDocument = document;
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
// if there was a redirection, show the final URL in the browser's address bar
|
||||||
|
if (response.redirected) {
|
||||||
|
preparationEvent.to = new URL(response.redirected);
|
||||||
|
}
|
||||||
|
|
||||||
|
parser ??= new DOMParser();
|
||||||
|
|
||||||
|
preparationEvent.newDocument = parser.parseFromString(response.html, response.mediaType);
|
||||||
|
// The next line might look like a hack,
|
||||||
|
// but it is actually necessary as noscript elements
|
||||||
|
// and their contents are returned as markup by the parser,
|
||||||
|
// see https://developer.mozilla.org/en-US/docs/Web/API/DOMParser/parseFromString
|
||||||
|
preparationEvent.newDocument.querySelectorAll('noscript').forEach((el) => el.remove());
|
||||||
|
|
||||||
|
// If ViewTransitions is not enabled on the incoming page, do a full page load to it.
|
||||||
|
// Unless this was a form submission, in which case we do not want to trigger another mutation.
|
||||||
|
if (
|
||||||
|
!preparationEvent.newDocument.querySelector('[name="astro-view-transitions-enabled"]') &&
|
||||||
|
!preparationEvent.formData
|
||||||
|
) {
|
||||||
|
preparationEvent.preventDefault();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const links = preloadStyleLinks(preparationEvent.newDocument);
|
||||||
|
links.length && (await Promise.all(links));
|
||||||
|
|
||||||
|
if (import.meta.env.DEV)
|
||||||
|
await prepareForClientOnlyComponents(preparationEvent.newDocument, preparationEvent.to);
|
||||||
}
|
}
|
||||||
|
|
||||||
skipTransition = false;
|
skipTransition = false;
|
||||||
|
@ -566,12 +557,6 @@ export async function navigate(href: string, options?: Options) {
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// not ours
|
|
||||||
if (!transitionEnabledOnThisPage()) {
|
|
||||||
location.href = href;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await transition('forward', originalLocation, new URL(href, location.href), options ?? {});
|
await transition('forward', originalLocation, new URL(href, location.href), options ?? {});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue