Skip to content

Commit aa82625

Browse files
committed
fix: fix v-for iterator parsing destructuring + parens without index
1 parent 14e9908 commit aa82625

File tree

3 files changed

+48
-8
lines changed

3 files changed

+48
-8
lines changed

src/compiler/parser/index.js

+3-2
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,8 @@ import {
2222
export const onRE = /^@|^v-on:/
2323
export const dirRE = /^v-|^@|^:/
2424
export const forAliasRE = /(.*?)\s+(?:in|of)\s+(.*)/
25-
export const forIteratorRE = /\((\{[^}]*\}|[^,]*),([^,]*)(?:,([^,]*))?\)/
25+
export const forIteratorRE = /\((\{[^}]*\}|[^,{]*),([^,]*)(?:,([^,]*))?\)/
26+
const stripParensRE = /^\(|\)$/g
2627

2728
const argRE = /:(.*)$/
2829
const bindRE = /^:|^v-bind:/
@@ -364,7 +365,7 @@ export function processFor (el: ASTElement) {
364365
el.iterator2 = iteratorMatch[3].trim()
365366
}
366367
} else {
367-
el.alias = alias
368+
el.alias = alias.replace(stripParensRE, '')
368369
}
369370
}
370371
}

test/unit/features/directives/for.spec.js

+6-6
Original file line numberDiff line numberDiff line change
@@ -464,20 +464,20 @@ describe('Directive v-for', () => {
464464
}).then(done)
465465
})
466466

467-
const supportsDeconstruct = (() => {
467+
const supportsDestructuring = (() => {
468468
try {
469469
new Function('var { foo } = bar')
470470
return true
471471
} catch (e) {}
472472
})()
473473

474-
if (supportsDeconstruct) {
475-
it('should support deconstruct syntax in alias position', () => {
474+
if (supportsDestructuring) {
475+
it('should support destructuring syntax in alias position', () => {
476476
const vm = new Vue({
477-
data: { list: [{ foo: 'hi' }] },
478-
template: '<div><div v-for="({ foo }, i) in list">{{ foo }}{{ i }}</div></div>'
477+
data: { list: [{ foo: 'hi', bar: 'ho' }] },
478+
template: '<div><div v-for="({ foo, bar }, i) in list">{{ foo }} {{ bar }} {{ i }}</div></div>'
479479
}).$mount()
480-
expect(vm.$el.textContent).toBe('hi0')
480+
expect(vm.$el.textContent).toBe('hi ho 0')
481481
})
482482
}
483483
})

test/unit/modules/compiler/parser.spec.js

+39
Original file line numberDiff line numberDiff line change
@@ -283,6 +283,45 @@ describe('parser', () => {
283283
expect(liAst.key).toBe('item.uid')
284284
})
285285

286+
it('v-for directive destructuring', () => {
287+
let ast = parse('<ul><li v-for="{ foo } in items"></li></ul>', baseOptions)
288+
let liAst = ast.children[0]
289+
expect(liAst.for).toBe('items')
290+
expect(liAst.alias).toBe('{ foo }')
291+
292+
// with paren
293+
ast = parse('<ul><li v-for="({ foo }) in items"></li></ul>', baseOptions)
294+
liAst = ast.children[0]
295+
expect(liAst.for).toBe('items')
296+
expect(liAst.alias).toBe('{ foo }')
297+
298+
// multi-var destructuring
299+
ast = parse('<ul><li v-for="{ foo, bar, baz } in items"></li></ul>', baseOptions)
300+
liAst = ast.children[0]
301+
expect(liAst.for).toBe('items')
302+
expect(liAst.alias).toBe('{ foo, bar, baz }')
303+
304+
// multi-var destructuring with paren
305+
ast = parse('<ul><li v-for="({ foo, bar, baz }) in items"></li></ul>', baseOptions)
306+
liAst = ast.children[0]
307+
expect(liAst.for).toBe('items')
308+
expect(liAst.alias).toBe('{ foo, bar, baz }')
309+
310+
// with index
311+
ast = parse('<ul><li v-for="({ foo }, i) in items"></li></ul>', baseOptions)
312+
liAst = ast.children[0]
313+
expect(liAst.for).toBe('items')
314+
expect(liAst.alias).toBe('{ foo }')
315+
expect(liAst.iterator1).toBe('i')
316+
317+
// multi-var destructuring with index
318+
ast = parse('<ul><li v-for="({ foo, bar, baz }, i) in items"></li></ul>', baseOptions)
319+
liAst = ast.children[0]
320+
expect(liAst.for).toBe('items')
321+
expect(liAst.alias).toBe('{ foo, bar, baz }')
322+
expect(liAst.iterator1).toBe('i')
323+
})
324+
286325
it('v-for directive invalid syntax', () => {
287326
parse('<ul><li v-for="item into items"></li></ul>', baseOptions)
288327
expect('Invalid v-for expression').toHaveBeenWarned()

0 commit comments

Comments
 (0)