fix: improve how transitions are handled on mount (#10157)

pull/10159/head
Dominic Gannaway 1 year ago committed by GitHub
parent a1a3e819c9
commit 8c95777644
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -0,0 +1,5 @@
---
'svelte': patch
---
fix: improve how transitions are handled on mount

@ -1012,11 +1012,10 @@ export function mutate_store(store, expression, new_value) {
/** /**
* @param {import('./types.js').ComputationSignal} signal * @param {import('./types.js').ComputationSignal} signal
* @param {import('./types.js').ComputationSignal} root
* @param {boolean} inert * @param {boolean} inert
* @returns {void} * @returns {void}
*/ */
export function mark_subtree_inert(signal, root, inert, visited_blocks = new Set()) { export function mark_subtree_inert(signal, inert, visited_blocks = new Set()) {
const flags = signal.f; const flags = signal.f;
const is_already_inert = (flags & INERT) !== 0; const is_already_inert = (flags & INERT) !== 0;
if (is_already_inert !== inert) { if (is_already_inert !== inert) {
@ -1031,23 +1030,22 @@ export function mark_subtree_inert(signal, root, inert, visited_blocks = new Set
const type = block.t; const type = block.t;
if (type === IF_BLOCK) { if (type === IF_BLOCK) {
const condition_effect = block.e; const condition_effect = block.e;
const root_block = root.b?.p; if (condition_effect !== null && block !== current_block) {
if (condition_effect !== null && root_block?.t === IF_BLOCK) { mark_subtree_inert(condition_effect, inert);
mark_subtree_inert(condition_effect, root, inert);
} }
const consequent_effect = block.ce; const consequent_effect = block.ce;
if (consequent_effect !== null) { if (consequent_effect !== null) {
mark_subtree_inert(consequent_effect, root, inert, visited_blocks); mark_subtree_inert(consequent_effect, inert, visited_blocks);
} }
const alternate_effect = block.ae; const alternate_effect = block.ae;
if (alternate_effect !== null) { if (alternate_effect !== null) {
mark_subtree_inert(alternate_effect, root, inert, visited_blocks); mark_subtree_inert(alternate_effect, inert, visited_blocks);
} }
} else if (type === EACH_BLOCK) { } else if (type === EACH_BLOCK) {
const items = block.v; const items = block.v;
for (let { e: each_item_effect } of items) { for (let { e: each_item_effect } of items) {
if (each_item_effect !== null) { if (each_item_effect !== null) {
mark_subtree_inert(each_item_effect, root, inert, visited_blocks); mark_subtree_inert(each_item_effect, inert, visited_blocks);
} }
} }
} }
@ -1057,7 +1055,7 @@ export function mark_subtree_inert(signal, root, inert, visited_blocks = new Set
if (references !== null) { if (references !== null) {
let i; let i;
for (i = 0; i < references.length; i++) { for (i = 0; i < references.length; i++) {
mark_subtree_inert(references[i], root, inert, visited_blocks); mark_subtree_inert(references[i], inert, visited_blocks);
} }
} }
} }

@ -482,10 +482,10 @@ export function bind_transition(dom, get_transition_fn, props_fn, direction, glo
// @ts-ignore // @ts-ignore
dom.__animate = true; dom.__animate = true;
} }
let foo = false;
/** @type {import('./types.js').Block | null} */ /** @type {import('./types.js').Block | null} */
let transition_block = block; let transition_block = block;
while (transition_block !== null) { main: while (transition_block !== null) {
if (is_transition_block(transition_block)) { if (is_transition_block(transition_block)) {
if (transition_block.t === EACH_ITEM_BLOCK) { if (transition_block.t === EACH_ITEM_BLOCK) {
// Lazily apply the each block transition // Lazily apply the each block transition
@ -493,22 +493,32 @@ export function bind_transition(dom, get_transition_fn, props_fn, direction, glo
transition_block.a = each_item_animate; transition_block.a = each_item_animate;
transition_block = transition_block.p; transition_block = transition_block.p;
} else if (transition_block.t === AWAIT_BLOCK && transition_block.n /* pending */) { } else if (transition_block.t === AWAIT_BLOCK && transition_block.n /* pending */) {
can_show_intro_on_mount = false; can_show_intro_on_mount = true;
} else if (transition_block.t === IF_BLOCK) { } else if (transition_block.t === IF_BLOCK) {
transition_block.r = if_block_transition; transition_block.r = if_block_transition;
if (can_show_intro_on_mount) {
/** @type {import('./types.js').Block | null} */
let if_block = transition_block;
while (if_block.t === IF_BLOCK) {
// If we have an if block parent that is currently falsy then
// we can show the intro on mount as long as that block is mounted
if (if_block.e !== null && !if_block.v) {
can_show_intro_on_mount = true;
break main;
}
if_block = if_block.p;
}
}
} }
if (!can_apply_lazy_transitions && can_show_intro_on_mount) { if (!can_apply_lazy_transitions && can_show_intro_on_mount) {
can_show_intro_on_mount = transition_block.e === null; can_show_intro_on_mount = transition_block.e !== null;
foo = true;
} }
if (!can_show_intro_on_mount || !global) { if (can_show_intro_on_mount || !global) {
can_apply_lazy_transitions = true; can_apply_lazy_transitions = true;
} }
} else if ( } else if (transition_block.t === ROOT_BLOCK && !can_apply_lazy_transitions) {
!can_apply_lazy_transitions && can_show_intro_on_mount = transition_block.e !== null || transition_block.i;
transition_block.t === ROOT_BLOCK &&
(transition_block.e !== null || transition_block.i)
) {
can_show_intro_on_mount = false;
} }
transition_block = transition_block.p; transition_block = transition_block.p;
} }
@ -540,7 +550,7 @@ export function bind_transition(dom, get_transition_fn, props_fn, direction, glo
transition = create_transition(dom, init, direction, transition_effect); transition = create_transition(dom, init, direction, transition_effect);
const is_intro = direction === 'in'; const is_intro = direction === 'in';
const show_intro = !can_show_intro_on_mount && (is_intro || direction === 'both'); const show_intro = can_show_intro_on_mount && (is_intro || direction === 'both');
if (show_intro) { if (show_intro) {
transition.p = transition.i(); transition.p = transition.i();
@ -602,7 +612,7 @@ export function trigger_transitions(transitions, target_direction, from) {
transition.c(); transition.c();
} }
transition.d.inert = false; transition.d.inert = false;
mark_subtree_inert(effect, effect, false); mark_subtree_inert(effect, false);
} else if (target_direction === 'key') { } else if (target_direction === 'key') {
if (direction === 'key') { if (direction === 'key') {
transition.p = transition.i(/** @type {DOMRect} */ (from)); transition.p = transition.i(/** @type {DOMRect} */ (from));
@ -614,7 +624,7 @@ export function trigger_transitions(transitions, target_direction, from) {
outros.push(transition.o); outros.push(transition.o);
} }
transition.d.inert = true; transition.d.inert = true;
mark_subtree_inert(effect, effect, true); mark_subtree_inert(effect, true);
} }
} }
if (outros.length > 0) { if (outros.length > 0) {

Loading…
Cancel
Save