Skip to content

Commit 1e1aea4

Browse files
adigubaRich-Harris
andauthored
chore: remove unnecessary ?? '' on some expressions (#15287)
* no `?? ''` on some expressions * changeset * delete operator returns boolean * inverted conditions are a lil confusing * no need for else after return * simplify condition * may as well add special handling for undefined while we're here * use normal string literal when there are no values * omit assignment when there is no text content --------- Co-authored-by: Rich Harris <[email protected]>
1 parent 60b22c0 commit 1e1aea4

File tree

3 files changed

+46
-33
lines changed

3 files changed

+46
-33
lines changed

.changeset/serious-glasses-kiss.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'svelte': patch
3+
---
4+
5+
chore: remove unnecessary `?? ''` on some expressions

packages/svelte/src/compiler/phases/3-transform/client/visitors/RegularElement.js

+8-9
Original file line numberDiff line numberDiff line change
@@ -364,15 +364,14 @@ export function RegularElement(node, context) {
364364
trimmed.some((node) => node.type === 'ExpressionTag');
365365

366366
if (use_text_content) {
367-
child_state.init.push(
368-
b.stmt(
369-
b.assignment(
370-
'=',
371-
b.member(context.state.node, 'textContent'),
372-
build_template_chunk(trimmed, context.visit, child_state).value
373-
)
374-
)
375-
);
367+
const { value } = build_template_chunk(trimmed, context.visit, child_state);
368+
const empty_string = value.type === 'Literal' && value.value === '';
369+
370+
if (!empty_string) {
371+
child_state.init.push(
372+
b.stmt(b.assignment('=', b.member(context.state.node, 'textContent'), value))
373+
);
374+
}
376375
} else {
377376
/** @type {Expression} */
378377
let arg = context.state.node;

packages/svelte/src/compiler/phases/3-transform/client/visitors/shared/utils.js

+33-24
Original file line numberDiff line numberDiff line change
@@ -101,11 +101,15 @@ export function build_template_chunk(
101101

102102
if (node.type === 'Text') {
103103
quasi.value.cooked += node.data;
104-
} else if (node.type === 'ExpressionTag' && node.expression.type === 'Literal') {
104+
} else if (node.expression.type === 'Literal') {
105105
if (node.expression.value != null) {
106106
quasi.value.cooked += node.expression.value + '';
107107
}
108-
} else {
108+
} else if (
109+
node.expression.type !== 'Identifier' ||
110+
node.expression.name !== 'undefined' ||
111+
state.scope.get('undefined')
112+
) {
109113
let value = memoize(
110114
/** @type {Expression} */ (visit(node.expression, state)),
111115
node.metadata.expression
@@ -117,31 +121,33 @@ export function build_template_chunk(
117121
// If we have a single expression, then pass that in directly to possibly avoid doing
118122
// extra work in the template_effect (instead we do the work in set_text).
119123
return { value, has_state };
120-
} else {
121-
// add `?? ''` where necessary (TODO optimise more cases)
122-
if (
123-
value.type === 'LogicalExpression' &&
124-
value.right.type === 'Literal' &&
125-
(value.operator === '??' || value.operator === '||')
126-
) {
127-
// `foo ?? null` -=> `foo ?? ''`
128-
// otherwise leave the expression untouched
129-
if (value.right.value === null) {
130-
value = { ...value, right: b.literal('') };
131-
}
132-
} else if (
133-
state.analysis.props_id &&
134-
value.type === 'Identifier' &&
135-
value.name === state.analysis.props_id.name
136-
) {
137-
// do nothing ($props.id() is never null/undefined)
138-
} else {
139-
value = b.logical('??', value, b.literal(''));
124+
}
125+
126+
if (
127+
value.type === 'LogicalExpression' &&
128+
value.right.type === 'Literal' &&
129+
(value.operator === '??' || value.operator === '||')
130+
) {
131+
// `foo ?? null` -=> `foo ?? ''`
132+
// otherwise leave the expression untouched
133+
if (value.right.value === null) {
134+
value = { ...value, right: b.literal('') };
140135
}
136+
}
137+
138+
const is_defined =
139+
value.type === 'BinaryExpression' ||
140+
(value.type === 'UnaryExpression' && value.operator !== 'void') ||
141+
(value.type === 'LogicalExpression' && value.right.type === 'Literal') ||
142+
(value.type === 'Identifier' && value.name === state.analysis.props_id?.name);
141143

142-
expressions.push(value);
144+
if (!is_defined) {
145+
// add `?? ''` where necessary (TODO optimise more cases)
146+
value = b.logical('??', value, b.literal(''));
143147
}
144148

149+
expressions.push(value);
150+
145151
quasi = b.quasi('', i + 1 === values.length);
146152
quasis.push(quasi);
147153
}
@@ -151,7 +157,10 @@ export function build_template_chunk(
151157
quasi.value.raw = sanitize_template_string(/** @type {string} */ (quasi.value.cooked));
152158
}
153159

154-
const value = b.template(quasis, expressions);
160+
const value =
161+
expressions.length > 0
162+
? b.template(quasis, expressions)
163+
: b.literal(/** @type {string} */ (quasi.value.cooked));
155164

156165
return { value, has_state };
157166
}

0 commit comments

Comments
 (0)