@@ -101,11 +101,15 @@ export function build_template_chunk(
101
101
102
102
if ( node . type === 'Text' ) {
103
103
quasi . value . cooked += node . data ;
104
- } else if ( node . type === 'ExpressionTag' && node . expression . type === 'Literal' ) {
104
+ } else if ( node . expression . type === 'Literal' ) {
105
105
if ( node . expression . value != null ) {
106
106
quasi . value . cooked += node . expression . value + '' ;
107
107
}
108
- } else {
108
+ } else if (
109
+ node . expression . type !== 'Identifier' ||
110
+ node . expression . name !== 'undefined' ||
111
+ state . scope . get ( 'undefined' )
112
+ ) {
109
113
let value = memoize (
110
114
/** @type {Expression } */ ( visit ( node . expression , state ) ) ,
111
115
node . metadata . expression
@@ -117,31 +121,33 @@ export function build_template_chunk(
117
121
// If we have a single expression, then pass that in directly to possibly avoid doing
118
122
// extra work in the template_effect (instead we do the work in set_text).
119
123
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 ( '' ) } ;
140
135
}
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 ) ;
141
143
142
- expressions . push ( value ) ;
144
+ if ( ! is_defined ) {
145
+ // add `?? ''` where necessary (TODO optimise more cases)
146
+ value = b . logical ( '??' , value , b . literal ( '' ) ) ;
143
147
}
144
148
149
+ expressions . push ( value ) ;
150
+
145
151
quasi = b . quasi ( '' , i + 1 === values . length ) ;
146
152
quasis . push ( quasi ) ;
147
153
}
@@ -151,7 +157,10 @@ export function build_template_chunk(
151
157
quasi . value . raw = sanitize_template_string ( /** @type {string } */ ( quasi . value . cooked ) ) ;
152
158
}
153
159
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 ) ) ;
155
164
156
165
return { value, has_state } ;
157
166
}
0 commit comments