@@ -37,21 +37,44 @@ export function isSingleElementRoot(
37
37
)
38
38
}
39
39
40
+ const enum StaticType {
41
+ NOT_STATIC = 0 ,
42
+ FULL_STATIC ,
43
+ HAS_RUNTIME_CONSTANT
44
+ }
45
+
40
46
function walk (
41
47
children : TemplateChildNode [ ] ,
42
48
context : TransformContext ,
43
- resultCache : Map < TemplateChildNode , boolean > ,
49
+ resultCache : Map < TemplateChildNode , StaticType > ,
44
50
doNotHoistNode : boolean = false
45
51
) {
46
52
let hasHoistedNode = false
53
+ // Some transforms, e.g. trasnformAssetUrls from @vue/compiler-sfc, replaces
54
+ // static bindings with expressions. These expressions are guaranteed to be
55
+ // constant so they are still eligible for hoisting, but they are only
56
+ // available at runtime and therefore cannot be evaluated ahead of time.
57
+ // This is only a concern for pre-stringification (via transformHoist by
58
+ // @vue /compiler-dom), but doing it here allows us to perform only one full
59
+ // walk of the AST and allow `stringifyStatic` to stop walking as soon as its
60
+ // stringficiation threshold is met.
61
+ let hasRuntimeConstant = false
62
+
47
63
for ( let i = 0 ; i < children . length ; i ++ ) {
48
64
const child = children [ i ]
49
65
// only plain elements & text calls are eligible for hoisting.
50
66
if (
51
67
child . type === NodeTypes . ELEMENT &&
52
68
child . tagType === ElementTypes . ELEMENT
53
69
) {
54
- if ( ! doNotHoistNode && isStaticNode ( child , resultCache ) ) {
70
+ let staticType
71
+ if (
72
+ ! doNotHoistNode &&
73
+ ( staticType = getStaticType ( child , resultCache ) ) > 0
74
+ ) {
75
+ if ( staticType === StaticType . HAS_RUNTIME_CONSTANT ) {
76
+ hasRuntimeConstant = true
77
+ }
55
78
// whole tree is static
56
79
; ( child . codegenNode as VNodeCall ) . patchFlag =
57
80
PatchFlags . HOISTED + ( __DEV__ ? ` /* HOISTED */` : `` )
@@ -78,12 +101,15 @@ function walk(
78
101
}
79
102
}
80
103
}
81
- } else if (
82
- child . type === NodeTypes . TEXT_CALL &&
83
- isStaticNode ( child . content , resultCache )
84
- ) {
85
- child . codegenNode = context . hoist ( child . codegenNode )
86
- hasHoistedNode = true
104
+ } else if ( child . type === NodeTypes . TEXT_CALL ) {
105
+ const staticType = getStaticType ( child . content , resultCache )
106
+ if ( staticType > 0 ) {
107
+ if ( staticType === StaticType . HAS_RUNTIME_CONSTANT ) {
108
+ hasRuntimeConstant = true
109
+ }
110
+ child . codegenNode = context . hoist ( child . codegenNode )
111
+ hasHoistedNode = true
112
+ }
87
113
}
88
114
89
115
// walk further
@@ -101,73 +127,108 @@ function walk(
101
127
}
102
128
}
103
129
104
- if ( hasHoistedNode && context . transformHoist ) {
130
+ if ( ! hasRuntimeConstant && hasHoistedNode && context . transformHoist ) {
105
131
context . transformHoist ( children , context )
106
132
}
107
133
}
108
134
109
- export function isStaticNode (
135
+ export function getStaticType (
110
136
node : TemplateChildNode | SimpleExpressionNode ,
111
- resultCache : Map < TemplateChildNode , boolean > = new Map ( )
112
- ) : boolean {
137
+ resultCache : Map < TemplateChildNode , StaticType > = new Map ( )
138
+ ) : StaticType {
113
139
switch ( node . type ) {
114
140
case NodeTypes . ELEMENT :
115
141
if ( node . tagType !== ElementTypes . ELEMENT ) {
116
- return false
142
+ return StaticType . NOT_STATIC
117
143
}
118
144
const cached = resultCache . get ( node )
119
145
if ( cached !== undefined ) {
120
146
return cached
121
147
}
122
148
const codegenNode = node . codegenNode !
123
149
if ( codegenNode . type !== NodeTypes . VNODE_CALL ) {
124
- return false
150
+ return StaticType . NOT_STATIC
125
151
}
126
152
const flag = getPatchFlag ( codegenNode )
127
153
if ( ! flag && ! hasDynamicKeyOrRef ( node ) && ! hasCachedProps ( node ) ) {
128
154
// element self is static. check its children.
155
+ let returnType = StaticType . FULL_STATIC
129
156
for ( let i = 0 ; i < node . children . length ; i ++ ) {
130
- if ( ! isStaticNode ( node . children [ i ] , resultCache ) ) {
131
- resultCache . set ( node , false )
132
- return false
157
+ const childType = getStaticType ( node . children [ i ] , resultCache )
158
+ if ( childType === StaticType . NOT_STATIC ) {
159
+ resultCache . set ( node , StaticType . NOT_STATIC )
160
+ return StaticType . NOT_STATIC
161
+ } else if ( childType === StaticType . HAS_RUNTIME_CONSTANT ) {
162
+ returnType = StaticType . HAS_RUNTIME_CONSTANT
133
163
}
134
164
}
135
- // only svg/foreignObject could be block here, however if they are static
136
- // then they don't need to be blocks since there will be no nested
137
- // updates.
165
+
166
+ // check if any of the props contain runtime constants
167
+ if ( returnType !== StaticType . HAS_RUNTIME_CONSTANT ) {
168
+ for ( let i = 0 ; i < node . props . length ; i ++ ) {
169
+ const p = node . props [ i ]
170
+ if (
171
+ p . type === NodeTypes . DIRECTIVE &&
172
+ p . name === 'bind' &&
173
+ p . exp &&
174
+ ( p . exp . type === NodeTypes . COMPOUND_EXPRESSION ||
175
+ p . exp . isRuntimeConstant )
176
+ ) {
177
+ returnType = StaticType . HAS_RUNTIME_CONSTANT
178
+ }
179
+ }
180
+ }
181
+
182
+ // only svg/foreignObject could be block here, however if they are
183
+ // stati then they don't need to be blocks since there will be no
184
+ // nested updates.
138
185
if ( codegenNode . isBlock ) {
139
186
codegenNode . isBlock = false
140
187
}
141
- resultCache . set ( node , true )
142
- return true
188
+
189
+ resultCache . set ( node , returnType )
190
+ return returnType
143
191
} else {
144
- resultCache . set ( node , false )
145
- return false
192
+ resultCache . set ( node , StaticType . NOT_STATIC )
193
+ return StaticType . NOT_STATIC
146
194
}
147
195
case NodeTypes . TEXT :
148
196
case NodeTypes . COMMENT :
149
- return true
197
+ return StaticType . FULL_STATIC
150
198
case NodeTypes . IF :
151
199
case NodeTypes . FOR :
152
200
case NodeTypes . IF_BRANCH :
153
- return false
201
+ return StaticType . NOT_STATIC
154
202
case NodeTypes . INTERPOLATION :
155
203
case NodeTypes . TEXT_CALL :
156
- return isStaticNode ( node . content , resultCache )
204
+ return getStaticType ( node . content , resultCache )
157
205
case NodeTypes . SIMPLE_EXPRESSION :
158
206
return node . isConstant
207
+ ? node . isRuntimeConstant
208
+ ? StaticType . HAS_RUNTIME_CONSTANT
209
+ : StaticType . FULL_STATIC
210
+ : StaticType . NOT_STATIC
159
211
case NodeTypes . COMPOUND_EXPRESSION :
160
- return node . children . every ( child => {
161
- return (
162
- isString ( child ) || isSymbol ( child ) || isStaticNode ( child , resultCache )
163
- )
164
- } )
212
+ let returnType = StaticType . FULL_STATIC
213
+ for ( let i = 0 ; i < node . children . length ; i ++ ) {
214
+ const child = node . children [ i ]
215
+ if ( isString ( child ) || isSymbol ( child ) ) {
216
+ continue
217
+ }
218
+ const childType = getStaticType ( child , resultCache )
219
+ if ( childType === StaticType . NOT_STATIC ) {
220
+ return StaticType . NOT_STATIC
221
+ } else if ( childType === StaticType . HAS_RUNTIME_CONSTANT ) {
222
+ returnType = StaticType . HAS_RUNTIME_CONSTANT
223
+ }
224
+ }
225
+ return returnType
165
226
default :
166
227
if ( __DEV__ ) {
167
228
const exhaustiveCheck : never = node
168
229
exhaustiveCheck
169
230
}
170
- return false
231
+ return StaticType . NOT_STATIC
171
232
}
172
233
}
173
234
0 commit comments