mirror of
https://github.com/withastro/astro.git
synced 2025-02-10 22:38:53 -05:00
Allow fallback animations on html element (#8258)
* Allow fallback animations on html element * Add fallback attr after swap * Break apart addModern and addFallback into separate functions
This commit is contained in:
parent
0a97524e38
commit
1db4e92c12
3 changed files with 25 additions and 15 deletions
5
.changeset/strong-needles-accept.md
Normal file
5
.changeset/strong-needles-accept.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'astro': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Allow fallback animations on html element
|
|
@ -220,23 +220,15 @@ const { fallback = 'animate' } = Astro.props as Props;
|
||||||
links.length && (await Promise.all(links));
|
links.length && (await Promise.all(links));
|
||||||
|
|
||||||
if (fallback === 'animate') {
|
if (fallback === 'animate') {
|
||||||
let isAnimating = false;
|
|
||||||
addEventListener('animationstart', () => (isAnimating = true), { once: true });
|
|
||||||
|
|
||||||
// Trigger the animations
|
// Trigger the animations
|
||||||
document.documentElement.dataset.astroTransitionFallback = 'old';
|
document.documentElement.dataset.astroTransitionFallback = 'old';
|
||||||
|
const finished = Promise.all(document.getAnimations().map(a => a.finished));
|
||||||
const fallbackSwap = () => {
|
const fallbackSwap = () => {
|
||||||
removeEventListener('animationend', fallbackSwap);
|
|
||||||
clearTimeout(timeout);
|
|
||||||
swap();
|
swap();
|
||||||
document.documentElement.dataset.astroTransitionFallback = 'new';
|
document.documentElement.dataset.astroTransitionFallback = 'new';
|
||||||
};
|
};
|
||||||
// If there are any animations, want for the animationend event.
|
await finished;
|
||||||
addEventListener('animationend', fallbackSwap, { once: true });
|
fallbackSwap();
|
||||||
// If there are no animations, go ahead and swap on next tick
|
|
||||||
// This is necessary because we do not know if there are animations.
|
|
||||||
// The setTimeout is a fallback in case there are none.
|
|
||||||
let timeout = setTimeout(() => !isAnimating && fallbackSwap());
|
|
||||||
} else {
|
} else {
|
||||||
swap();
|
swap();
|
||||||
}
|
}
|
||||||
|
|
|
@ -52,7 +52,8 @@ export function renderTransition(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (animationName === 'none') {
|
} else if (animationName === 'none') {
|
||||||
sheet.addAnimationRaw('old', 'animation: none; opacity: 0; mix-blend-mode: normal;');
|
sheet.addFallback('old', 'animation: none; mix-blend-mode: normal;');
|
||||||
|
sheet.addModern('old', 'animation: none; opacity: 0; mix-blend-mode: normal;');
|
||||||
sheet.addAnimationRaw('new', 'animation: none; mix-blend-mode: normal;');
|
sheet.addAnimationRaw('new', 'animation: none; mix-blend-mode: normal;');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,11 +89,22 @@ class ViewTransitionStyleSheet {
|
||||||
}
|
}
|
||||||
|
|
||||||
addAnimationRaw(image: 'old' | 'new' | 'group', animation: string) {
|
addAnimationRaw(image: 'old' | 'new' | 'group', animation: string) {
|
||||||
const { scope, name } = this;
|
this.addModern(image, animation);
|
||||||
|
this.addFallback(image, animation);
|
||||||
|
}
|
||||||
|
|
||||||
|
addModern(image: 'old' | 'new' | 'group', animation: string) {
|
||||||
|
const { name } = this;
|
||||||
this.addRule('modern', `::view-transition-${image}(${name}) { ${animation} }`);
|
this.addRule('modern', `::view-transition-${image}(${name}) { ${animation} }`);
|
||||||
|
}
|
||||||
|
|
||||||
|
addFallback(image: 'old' | 'new' | 'group', animation: string) {
|
||||||
|
const { scope } = this;
|
||||||
this.addRule(
|
this.addRule(
|
||||||
'fallback',
|
'fallback',
|
||||||
`[data-astro-transition-fallback="${image}"] [data-astro-transition-scope="${scope}"] { ${animation} }`
|
// Two selectors here, the second in case there is an animation on the root.
|
||||||
|
`[data-astro-transition-fallback="${image}"] [data-astro-transition-scope="${scope}"],
|
||||||
|
[data-astro-transition-fallback="${image}"][data-astro-transition-scope="${scope}"] { ${animation} }`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -107,7 +119,8 @@ class ViewTransitionStyleSheet {
|
||||||
this.addRule('modern', `${prefix}::view-transition-${image}(${name}) { ${animation} }`);
|
this.addRule('modern', `${prefix}::view-transition-${image}(${name}) { ${animation} }`);
|
||||||
this.addRule(
|
this.addRule(
|
||||||
'fallback',
|
'fallback',
|
||||||
`${prefix}[data-astro-transition-fallback="${image}"] [data-astro-transition-scope="${scope}"] { ${animation} }`
|
`${prefix}[data-astro-transition-fallback="${image}"] [data-astro-transition-scope="${scope}"],
|
||||||
|
${prefix}[data-astro-transition-fallback="${image}"][data-astro-transition-scope="${scope}"] { ${animation} }`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue