chore: make if blocks tree-shakable (#14549)

* chore: make if blocks dead code eliminable

* chore: make if blocks dead code eliminable

* chore: make if blocks dead code eliminable

* address feedback

* address feedback

* prettier

---------

Co-authored-by: Rich Harris <rich.harris@vercel.com>
pull/14557/head
Dominic Gannaway 1 month ago committed by GitHub
parent 51c9eac577
commit a220330011
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -0,0 +1,5 @@
---
'svelte': patch
---
chore: make if blocks tree-shakable

@ -9,23 +9,44 @@ import * as b from '../../../../utils/builders.js';
*/ */
export function IfBlock(node, context) { export function IfBlock(node, context) {
context.state.template.push('<!>'); context.state.template.push('<!>');
const statements = [];
const consequent = /** @type {BlockStatement} */ (context.visit(node.consequent)); const consequent = /** @type {BlockStatement} */ (context.visit(node.consequent));
const consequent_id = context.state.scope.generate('consequent');
statements.push(b.var(b.id(consequent_id), b.arrow([b.id('$$anchor')], consequent)));
let alternate_id;
if (node.alternate) {
const alternate = /** @type {BlockStatement} */ (context.visit(node.alternate));
alternate_id = context.state.scope.generate('alternate');
statements.push(b.var(b.id(alternate_id), b.arrow([b.id('$$anchor')], alternate)));
}
/** @type {Expression[]} */
const args = [ const args = [
context.state.node, context.state.node,
b.thunk(/** @type {Expression} */ (context.visit(node.test))), b.arrow(
b.arrow([b.id('$$anchor')], consequent) [b.id('$$render')],
b.block([
b.if(
/** @type {Expression} */ (context.visit(node.test)),
b.stmt(b.call(b.id('$$render'), b.id(consequent_id))),
alternate_id
? b.stmt(
b.call(
b.id('$$render'),
b.id(alternate_id),
node.alternate ? b.literal(false) : undefined
)
)
: undefined
)
])
)
]; ];
if (node.alternate || node.elseif) {
args.push(
node.alternate
? b.arrow([b.id('$$anchor')], /** @type {BlockStatement} */ (context.visit(node.alternate)))
: b.literal(null)
);
}
if (node.elseif) { if (node.elseif) {
// We treat this... // We treat this...
// //
@ -51,5 +72,7 @@ export function IfBlock(node, context) {
args.push(b.literal(true)); args.push(b.literal(true));
} }
context.state.init.push(b.stmt(b.call('$.if', ...args))); statements.push(b.stmt(b.call('$.if', ...args)));
context.state.init.push(b.block(statements));
} }

@ -13,13 +13,11 @@ import { HYDRATION_START_ELSE } from '../../../../constants.js';
/** /**
* @param {TemplateNode} node * @param {TemplateNode} node
* @param {() => boolean} get_condition * @param {(branch: (fn: (anchor: Node) => void, flag?: boolean) => void) => void} fn
* @param {(anchor: Node) => void} consequent_fn
* @param {null | ((anchor: Node) => void)} [alternate_fn]
* @param {boolean} [elseif] True if this is an `{:else if ...}` block rather than an `{#if ...}`, as that affects which transitions are considered 'local' * @param {boolean} [elseif] True if this is an `{:else if ...}` block rather than an `{#if ...}`, as that affects which transitions are considered 'local'
* @returns {void} * @returns {void}
*/ */
export function if_block(node, get_condition, consequent_fn, alternate_fn = null, elseif = false) { export function if_block(node, fn, elseif = false) {
if (hydrating) { if (hydrating) {
hydrate_next(); hydrate_next();
} }
@ -37,8 +35,18 @@ export function if_block(node, get_condition, consequent_fn, alternate_fn = null
var flags = elseif ? EFFECT_TRANSPARENT : 0; var flags = elseif ? EFFECT_TRANSPARENT : 0;
block(() => { var has_branch = false;
if (condition === (condition = !!get_condition())) return;
const set_branch = (/** @type {(anchor: Node) => void} */ fn, flag = true) => {
has_branch = true;
update_branch(flag, fn);
};
const update_branch = (
/** @type {boolean | null} */ new_condition,
/** @type {null | ((anchor: Node) => void)} */ fn
) => {
if (condition === (condition = new_condition)) return;
/** Whether or not there was a hydration mismatch. Needs to be a `let` or else it isn't treeshaken out */ /** Whether or not there was a hydration mismatch. Needs to be a `let` or else it isn't treeshaken out */
let mismatch = false; let mismatch = false;
@ -60,8 +68,8 @@ export function if_block(node, get_condition, consequent_fn, alternate_fn = null
if (condition) { if (condition) {
if (consequent_effect) { if (consequent_effect) {
resume_effect(consequent_effect); resume_effect(consequent_effect);
} else { } else if (fn) {
consequent_effect = branch(() => consequent_fn(anchor)); consequent_effect = branch(() => fn(anchor));
} }
if (alternate_effect) { if (alternate_effect) {
@ -72,8 +80,8 @@ export function if_block(node, get_condition, consequent_fn, alternate_fn = null
} else { } else {
if (alternate_effect) { if (alternate_effect) {
resume_effect(alternate_effect); resume_effect(alternate_effect);
} else if (alternate_fn) { } else if (fn) {
alternate_effect = branch(() => alternate_fn(anchor)); alternate_effect = branch(() => fn(anchor));
} }
if (consequent_effect) { if (consequent_effect) {
@ -87,6 +95,14 @@ export function if_block(node, get_condition, consequent_fn, alternate_fn = null
// continue in hydration mode // continue in hydration mode
set_hydrating(true); set_hydrating(true);
} }
};
block(() => {
has_branch = false;
fn(set_branch);
if (!has_branch) {
update_branch(null, null);
}
}, flags); }, flags);
if (hydrating) { if (hydrating) {

Loading…
Cancel
Save