Skip to content

Commit 137893a

Browse files
committed
fix(compiler/v-on): handle multiple statements in v-on handler (close #572)
1 parent 46a7937 commit 137893a

File tree

3 files changed

+56
-7
lines changed

3 files changed

+56
-7
lines changed

packages/compiler-core/__tests__/transforms/vOn.spec.ts

+41
Original file line numberDiff line numberDiff line change
@@ -140,6 +140,22 @@ describe('compiler: transform v-on', () => {
140140
})
141141
})
142142

143+
test('should handle multiple inline statement', () => {
144+
const { node } = parseWithVOn(`<div @click="foo();bar()"/>`)
145+
const props = (node.codegenNode as CallExpression)
146+
.arguments[1] as ObjectExpression
147+
expect(props.properties[0]).toMatchObject({
148+
key: { content: `onClick` },
149+
value: {
150+
type: NodeTypes.COMPOUND_EXPRESSION,
151+
// should wrap with `{` for multiple statements
152+
// in this case the return value is discarded and the behavior is
153+
// consistent with 2.x
154+
children: [`$event => {`, { content: `foo();bar()` }, `}`]
155+
}
156+
})
157+
})
158+
143159
test('inline statement w/ prefixIdentifiers: true', () => {
144160
const { node } = parseWithVOn(`<div @click="foo($event)"/>`, {
145161
prefixIdentifiers: true
@@ -163,6 +179,31 @@ describe('compiler: transform v-on', () => {
163179
})
164180
})
165181

182+
test('multiple inline statements w/ prefixIdentifiers: true', () => {
183+
const { node } = parseWithVOn(`<div @click="foo($event);bar()"/>`, {
184+
prefixIdentifiers: true
185+
})
186+
const props = (node.codegenNode as CallExpression)
187+
.arguments[1] as ObjectExpression
188+
expect(props.properties[0]).toMatchObject({
189+
key: { content: `onClick` },
190+
value: {
191+
type: NodeTypes.COMPOUND_EXPRESSION,
192+
children: [
193+
`$event => {`,
194+
{ content: `_ctx.foo` },
195+
`(`,
196+
// should NOT prefix $event
197+
{ content: `$event` },
198+
`);`,
199+
{ content: `_ctx.bar` },
200+
`()`,
201+
`}`
202+
]
203+
}
204+
})
205+
})
206+
166207
test('should NOT wrap as function if expression is already function expression', () => {
167208
const { node } = parseWithVOn(`<div @click="$event => foo($event)"/>`)
168209
const props = (node.codegenNode as CallExpression)

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

+11-4
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,9 @@ export function processExpression(
7676
context: TransformContext,
7777
// some expressions like v-slot props & v-for aliases should be parsed as
7878
// function params
79-
asParams: boolean = false
79+
asParams = false,
80+
// v-on handler values may contain multiple statements
81+
asRawStatements = false
8082
): ExpressionNode {
8183
if (!context.prefixIdentifiers || !node.content.trim()) {
8284
return node
@@ -100,9 +102,14 @@ export function processExpression(
100102
}
101103

102104
let ast: any
103-
// if the expression is supposed to be used in a function params position
104-
// we need to parse it differently.
105-
const source = `(${rawExp})${asParams ? `=>{}` : ``}`
105+
// exp needs to be parsed differently:
106+
// 1. Multiple inline statements (v-on, with presence of `;`): parse as raw
107+
// exp, but make sure to pad with spaces for consistent ranges
108+
// 2. Expressions: wrap with parens (for e.g. object expressions)
109+
// 3. Function arguments (v-for, v-slot): place in a function argument position
110+
const source = asRawStatements
111+
? ` ${rawExp} `
112+
: `(${rawExp})${asParams ? `=>{}` : ``}`
106113
try {
107114
ast = parseJS(source, { ranges: true })
108115
} catch (e) {

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

+4-3
Original file line numberDiff line numberDiff line change
@@ -59,11 +59,12 @@ export const transformOn: DirectiveTransform = (
5959
if (exp) {
6060
const isMemberExp = isMemberExpression(exp.content)
6161
const isInlineStatement = !(isMemberExp || fnExpRE.test(exp.content))
62+
const hasMultipleStatements = exp.content.includes(`;`)
6263

6364
// process the expression since it's been skipped
6465
if (!__BROWSER__ && context.prefixIdentifiers) {
6566
context.addIdentifiers(`$event`)
66-
exp = processExpression(exp, context)
67+
exp = processExpression(exp, context, false, hasMultipleStatements)
6768
context.removeIdentifiers(`$event`)
6869
// with scope analysis, the function is hoistable if it has no reference
6970
// to scope variables.
@@ -85,9 +86,9 @@ export const transformOn: DirectiveTransform = (
8586
if (isInlineStatement || (isCacheable && isMemberExp)) {
8687
// wrap inline statement in a function expression
8788
exp = createCompoundExpression([
88-
`$event => (`,
89+
`$event => ${hasMultipleStatements ? `{` : `(`}`,
8990
...(exp.type === NodeTypes.SIMPLE_EXPRESSION ? [exp] : exp.children),
90-
`)`
91+
hasMultipleStatements ? `}` : `)`
9192
])
9293
}
9394
}

0 commit comments

Comments
 (0)