Skip to content

Commit dd2bfb5

Browse files
committed
fix(compiler-dom): should bail stringification on runtime constant regardless of position
ref: vitejs/vite#157
1 parent 45c96a0 commit dd2bfb5

File tree

4 files changed

+99
-49
lines changed

4 files changed

+99
-49
lines changed

packages/compiler-core/src/transforms/hoistStatic.ts

+94-33
Original file line numberDiff line numberDiff line change
@@ -37,21 +37,44 @@ export function isSingleElementRoot(
3737
)
3838
}
3939

40+
const enum StaticType {
41+
NOT_STATIC = 0,
42+
FULL_STATIC,
43+
HAS_RUNTIME_CONSTANT
44+
}
45+
4046
function walk(
4147
children: TemplateChildNode[],
4248
context: TransformContext,
43-
resultCache: Map<TemplateChildNode, boolean>,
49+
resultCache: Map<TemplateChildNode, StaticType>,
4450
doNotHoistNode: boolean = false
4551
) {
4652
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+
4763
for (let i = 0; i < children.length; i++) {
4864
const child = children[i]
4965
// only plain elements & text calls are eligible for hoisting.
5066
if (
5167
child.type === NodeTypes.ELEMENT &&
5268
child.tagType === ElementTypes.ELEMENT
5369
) {
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+
}
5578
// whole tree is static
5679
;(child.codegenNode as VNodeCall).patchFlag =
5780
PatchFlags.HOISTED + (__DEV__ ? ` /* HOISTED */` : ``)
@@ -78,12 +101,15 @@ function walk(
78101
}
79102
}
80103
}
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+
}
87113
}
88114

89115
// walk further
@@ -101,73 +127,108 @@ function walk(
101127
}
102128
}
103129

104-
if (hasHoistedNode && context.transformHoist) {
130+
if (!hasRuntimeConstant && hasHoistedNode && context.transformHoist) {
105131
context.transformHoist(children, context)
106132
}
107133
}
108134

109-
export function isStaticNode(
135+
export function getStaticType(
110136
node: TemplateChildNode | SimpleExpressionNode,
111-
resultCache: Map<TemplateChildNode, boolean> = new Map()
112-
): boolean {
137+
resultCache: Map<TemplateChildNode, StaticType> = new Map()
138+
): StaticType {
113139
switch (node.type) {
114140
case NodeTypes.ELEMENT:
115141
if (node.tagType !== ElementTypes.ELEMENT) {
116-
return false
142+
return StaticType.NOT_STATIC
117143
}
118144
const cached = resultCache.get(node)
119145
if (cached !== undefined) {
120146
return cached
121147
}
122148
const codegenNode = node.codegenNode!
123149
if (codegenNode.type !== NodeTypes.VNODE_CALL) {
124-
return false
150+
return StaticType.NOT_STATIC
125151
}
126152
const flag = getPatchFlag(codegenNode)
127153
if (!flag && !hasDynamicKeyOrRef(node) && !hasCachedProps(node)) {
128154
// element self is static. check its children.
155+
let returnType = StaticType.FULL_STATIC
129156
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
133163
}
134164
}
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.
138185
if (codegenNode.isBlock) {
139186
codegenNode.isBlock = false
140187
}
141-
resultCache.set(node, true)
142-
return true
188+
189+
resultCache.set(node, returnType)
190+
return returnType
143191
} else {
144-
resultCache.set(node, false)
145-
return false
192+
resultCache.set(node, StaticType.NOT_STATIC)
193+
return StaticType.NOT_STATIC
146194
}
147195
case NodeTypes.TEXT:
148196
case NodeTypes.COMMENT:
149-
return true
197+
return StaticType.FULL_STATIC
150198
case NodeTypes.IF:
151199
case NodeTypes.FOR:
152200
case NodeTypes.IF_BRANCH:
153-
return false
201+
return StaticType.NOT_STATIC
154202
case NodeTypes.INTERPOLATION:
155203
case NodeTypes.TEXT_CALL:
156-
return isStaticNode(node.content, resultCache)
204+
return getStaticType(node.content, resultCache)
157205
case NodeTypes.SIMPLE_EXPRESSION:
158206
return node.isConstant
207+
? node.isRuntimeConstant
208+
? StaticType.HAS_RUNTIME_CONSTANT
209+
: StaticType.FULL_STATIC
210+
: StaticType.NOT_STATIC
159211
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
165226
default:
166227
if (__DEV__) {
167228
const exhaustiveCheck: never = node
168229
exhaustiveCheck
169230
}
170-
return false
231+
return StaticType.NOT_STATIC
171232
}
172233
}
173234

packages/compiler-core/src/transforms/transformElement.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@ import {
4646
findDir
4747
} from '../utils'
4848
import { buildSlots } from './vSlot'
49-
import { isStaticNode } from './hoistStatic'
49+
import { getStaticType } from './hoistStatic'
5050

5151
// some directive transforms (e.g. v-model) may return a symbol for runtime
5252
// import, which should be used instead of a resolveDirective call.
@@ -156,7 +156,7 @@ export const transformElement: NodeTransform = (node, context) => {
156156
const hasDynamicTextChild =
157157
type === NodeTypes.INTERPOLATION ||
158158
type === NodeTypes.COMPOUND_EXPRESSION
159-
if (hasDynamicTextChild && !isStaticNode(child)) {
159+
if (hasDynamicTextChild && !getStaticType(child)) {
160160
patchFlag |= PatchFlags.TEXT
161161
}
162162
// pass directly if the only child is a text node
@@ -292,7 +292,7 @@ export function buildProps(
292292
value.type === NodeTypes.JS_CACHE_EXPRESSION ||
293293
((value.type === NodeTypes.SIMPLE_EXPRESSION ||
294294
value.type === NodeTypes.COMPOUND_EXPRESSION) &&
295-
isStaticNode(value))
295+
getStaticType(value) > 0)
296296
) {
297297
// skip if the prop is a cached handler or has constant value
298298
return

packages/compiler-dom/__tests__/transforms/stringifyStatic.spec.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -161,10 +161,10 @@ describe('stringify static html', () => {
161161

162162
test('should bail on runtime constant v-bind bindings', () => {
163163
const { ast } = compile(
164-
`<div><div><img src="./foo" />${repeat(
164+
`<div><div>${repeat(
165165
`<span class="foo">foo</span>`,
166166
StringifyThresholds.ELEMENT_WITH_BINDING_COUNT
167-
)}</div></div>`,
167+
)}<img src="./foo" /></div></div>`,
168168
{
169169
hoistStatic: true,
170170
prefixIdentifiers: true,

packages/compiler-dom/src/transforms/stringifyStatic.ts

-11
Original file line numberDiff line numberDiff line change
@@ -185,17 +185,6 @@ function analyzeNode(node: StringiableNode): [number, number] | false {
185185
) {
186186
return bail()
187187
}
188-
// some transforms, e.g. `transformAssetUrls` in `@vue/compiler-sfc` may
189-
// convert static attributes into a v-bind with a constnat expresion.
190-
// Such constant bindings are eligible for hoisting but not for static
191-
// stringification because they cannot be pre-evaluated.
192-
if (
193-
p.exp &&
194-
(p.exp.type === NodeTypes.COMPOUND_EXPRESSION ||
195-
p.exp.isRuntimeConstant)
196-
) {
197-
return bail()
198-
}
199188
}
200189
}
201190
for (let i = 0; i < node.children.length; i++) {

0 commit comments

Comments
 (0)