fix: further animation transition improvements (#10138)

* fix: further animation transition improvements

* clever hack
pull/10137/head
Dominic Gannaway 1 year ago committed by GitHub
parent 960fe308a5
commit 4ad5b73341
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -0,0 +1,5 @@
---
'svelte': patch
---
fix: further animation transition improvements

@ -369,6 +369,8 @@ function create_transition(dom, init, direction, effect) {
}, },
// out // out
o() { o() {
// @ts-ignore
const has_keyed_transition = dom.__animate;
const needs_reverse = direction === 'both' && curr_direction !== 'out'; const needs_reverse = direction === 'both' && curr_direction !== 'out';
curr_direction = 'out'; curr_direction = 'out';
if (animation === null || cancelled) { if (animation === null || cancelled) {
@ -385,6 +387,35 @@ function create_transition(dom, init, direction, effect) {
/** @type {Animation | TickAnimation} */ (animation).play(); /** @type {Animation | TickAnimation} */ (animation).play();
} }
} }
// If we're outroing an element that has an animation, then we need to fix
// its position to ensure it behaves nicely without causing layout shift.
if (has_keyed_transition) {
const style = getComputedStyle(dom);
const position = style.position;
if (position !== 'absolute' && position !== 'fixed') {
const { width, height } = style;
const a = dom.getBoundingClientRect();
dom.style.position = 'absolute';
dom.style.width = width;
dom.style.height = height;
const b = dom.getBoundingClientRect();
if (a.left !== b.left || a.top !== b.top) {
// Previously, in the Svelte 4, we'd just apply the transform the the DOM element. However,
// because we're now using Web Animations, we can't do that as it won't work properly if the
// animation is also making use of the same transformations. So instead, we apply an instantaneous
// animation and pause it on the first frame, just applying the same behavior.
const style = getComputedStyle(dom);
const transform = style.transform === 'none' ? '' : style.transform;
const frame = {
transform: `${transform} translate(${a.left - b.left}px, ${a.top - b.top}px)`
};
const animation = dom.animate([frame, frame], { duration: 1 });
animation.pause();
}
}
}
}, },
// cancel // cancel
c() { c() {
@ -432,10 +463,16 @@ function is_transition_block(block) {
export function bind_transition(dom, get_transition_fn, props_fn, direction, global) { export function bind_transition(dom, get_transition_fn, props_fn, direction, global) {
const transition_effect = /** @type {import('./types.js').EffectSignal} */ (current_effect); const transition_effect = /** @type {import('./types.js').EffectSignal} */ (current_effect);
const block = current_block; const block = current_block;
const is_keyed_transition = direction === 'key';
let can_show_intro_on_mount = true; let can_show_intro_on_mount = true;
let can_apply_lazy_transitions = false; let can_apply_lazy_transitions = false;
if (is_keyed_transition) {
// @ts-ignore
dom.__animate = true;
}
/** @type {import('./types.js').Block | null} */ /** @type {import('./types.js').Block | null} */
let transition_block = block; let transition_block = block;
while (transition_block !== null) { while (transition_block !== null) {
@ -479,7 +516,7 @@ export function bind_transition(dom, get_transition_fn, props_fn, direction, glo
const init = (from) => const init = (from) =>
untrack(() => { untrack(() => {
const props = props_fn === null ? {} : props_fn(); const props = props_fn === null ? {} : props_fn();
return direction === 'key' return is_keyed_transition
? /** @type {import('./types.js').AnimateFn<any>} */ (transition_fn)( ? /** @type {import('./types.js').AnimateFn<any>} */ (transition_fn)(
dom, dom,
{ from: /** @type {DOMRect} */ (from), to: dom.getBoundingClientRect() }, { from: /** @type {DOMRect} */ (from), to: dom.getBoundingClientRect() },

Loading…
Cancel
Save