Skip to content

Commit 958b6c8

Browse files
authored
fix(compiler-core): allow multiline expression on v-model and v-on (#1234)
1 parent c97d1ba commit 958b6c8

File tree

5 files changed

+79
-2
lines changed

5 files changed

+79
-2
lines changed

packages/compiler-core/__tests__/transforms/__snapshots__/vModel.spec.ts.snap

+19
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,25 @@ return function render(_ctx, _cache) {
2626
}"
2727
`;
2828

29+
exports[`compiler: transform v-model simple expression (with multilines) 1`] = `
30+
"const _Vue = Vue
31+
32+
return function render(_ctx, _cache) {
33+
with (_ctx) {
34+
const { createVNode: _createVNode, openBlock: _openBlock, createBlock: _createBlock } = _Vue
35+
36+
return (_openBlock(), _createBlock(\\"input\\", {
37+
modelValue:
38+
model
39+
,
40+
\\"onUpdate:modelValue\\": $event => (
41+
model
42+
= $event)
43+
}, null, 8 /* PROPS */, [\\"modelValue\\", \\"onUpdate:modelValue\\"]))
44+
}
45+
}"
46+
`;
47+
2948
exports[`compiler: transform v-model simple expression (with prefixIdentifiers) 1`] = `
3049
"import { createVNode as _createVNode, openBlock as _openBlock, createBlock as _createBlock } from \\"vue\\"
3150

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

+37
Original file line numberDiff line numberDiff line change
@@ -115,6 +115,43 @@ describe('compiler: transform v-model', () => {
115115
expect(generate(root, { mode: 'module' }).code).toMatchSnapshot()
116116
})
117117

118+
test('simple expression (with multilines)', () => {
119+
const root = parseWithVModel('<input v-model="\n model \n" />')
120+
const node = root.children[0] as ElementNode
121+
const props = ((node.codegenNode as VNodeCall).props as ObjectExpression)
122+
.properties
123+
124+
expect(props[0]).toMatchObject({
125+
key: {
126+
content: 'modelValue',
127+
isStatic: true
128+
},
129+
value: {
130+
content: '\n model \n',
131+
isStatic: false
132+
}
133+
})
134+
135+
expect(props[1]).toMatchObject({
136+
key: {
137+
content: 'onUpdate:modelValue',
138+
isStatic: true
139+
},
140+
value: {
141+
children: [
142+
'$event => (',
143+
{
144+
content: '\n model \n',
145+
isStatic: false
146+
},
147+
' = $event)'
148+
]
149+
}
150+
})
151+
152+
expect(generate(root).code).toMatchSnapshot()
153+
})
154+
118155
test('compound expression', () => {
119156
const root = parseWithVModel('<input v-model="model[index]" />')
120157
const node = root.children[0] as ElementNode

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

+18
Original file line numberDiff line numberDiff line change
@@ -167,6 +167,24 @@ describe('compiler: transform v-on', () => {
167167
})
168168
})
169169

170+
test('should handle multiple line statement', () => {
171+
const { node } = parseWithVOn(`<div @click="\nfoo();\nbar()\n"/>`)
172+
expect((node.codegenNode as VNodeCall).props).toMatchObject({
173+
properties: [
174+
{
175+
key: { content: `onClick` },
176+
value: {
177+
type: NodeTypes.COMPOUND_EXPRESSION,
178+
// should wrap with `{` for multiple statements
179+
// in this case the return value is discarded and the behavior is
180+
// consistent with 2.x
181+
children: [`$event => {`, { content: `\nfoo();\nbar()\n` }, `}`]
182+
}
183+
}
184+
]
185+
})
186+
})
187+
170188
test('inline statement w/ prefixIdentifiers: true', () => {
171189
const { node } = parseWithVOn(`<div @click="foo($event)"/>`, {
172190
prefixIdentifiers: true

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

+1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ export const transformModel: DirectiveTransform = (dir, node, context) => {
2121

2222
const expString =
2323
exp.type === NodeTypes.SIMPLE_EXPRESSION ? exp.content : exp.loc.source
24+
2425
if (!isMemberExpression(expString)) {
2526
context.onError(
2627
createCompilerError(ErrorCodes.X_V_MODEL_MALFORMED_EXPRESSION, exp.loc)

packages/compiler-core/src/utils.ts

+4-2
Original file line numberDiff line numberDiff line change
@@ -84,8 +84,10 @@ export const isSimpleIdentifier = (name: string): boolean =>
8484
!nonIdentifierRE.test(name)
8585

8686
const memberExpRE = /^[A-Za-z_$][\w$]*(?:\.[A-Za-z_$][\w$]*|\[[^\]]+\])*$/
87-
export const isMemberExpression = (path: string): boolean =>
88-
memberExpRE.test(path)
87+
export const isMemberExpression = (path: string): boolean => {
88+
if (!path) return false
89+
return memberExpRE.test(path.trim())
90+
}
8991

9092
export function getInnerRange(
9193
loc: SourceLocation,

0 commit comments

Comments
 (0)