Skip to content

Commit d2902ca

Browse files
committed
feat: support .property shorthand syntax for v-bind.prop modifier
close #7582
1 parent 23a1459 commit d2902ca

File tree

3 files changed

+31
-4
lines changed

3 files changed

+31
-4
lines changed

src/compiler/parser/index.js

+9-4
Original file line numberDiff line numberDiff line change
@@ -21,13 +21,14 @@ import {
2121
} from '../helpers'
2222

2323
export const onRE = /^@|^v-on:/
24-
export const dirRE = /^v-|^@|^:/
24+
export const dirRE = /^v-|^@|^:|^\./
2525
export const forAliasRE = /([\s\S]*?)\s+(?:in|of)\s+([\s\S]*)/
2626
export const forIteratorRE = /,([^,\}\]]*)(?:,([^,\}\]]*))?$/
2727
const stripParensRE = /^\(|\)$/g
2828

2929
const argRE = /:(.*)$/
30-
export const bindRE = /^:|^v-bind:/
30+
export const bindRE = /^:|^\.|^v-bind:/
31+
const propBindRE = /^\./
3132
const modifierRE = /\.[^.]+/g
3233

3334
const lineBreakRE = /[\r\n]/
@@ -683,8 +684,12 @@ function processAttrs (el) {
683684
// mark element as dynamic
684685
el.hasBindings = true
685686
// modifiers
686-
modifiers = parseModifiers(name)
687-
if (modifiers) {
687+
modifiers = parseModifiers(name.replace(dirRE, ''))
688+
// support .foo shorthand syntax for the .prop modifier
689+
if (propBindRE.test(name)) {
690+
(modifiers || (modifiers = {})).prop = true
691+
name = `.` + name.slice(1).replace(modifierRE, '')
692+
} else if (modifiers) {
688693
name = name.replace(modifierRE, '')
689694
}
690695
if (bindRE.test(name)) { // v-bind

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

+12
Original file line numberDiff line numberDiff line change
@@ -133,6 +133,18 @@ describe('Directive v-bind', () => {
133133
expect(vm.$el.getAttribute('id')).toBe(null)
134134
})
135135

136+
it('.prop modifier shorthand', () => {
137+
const vm = new Vue({
138+
template: '<div><span .text-content="foo"></span><span .inner-html="bar"></span></div>',
139+
data: {
140+
foo: 'hello',
141+
bar: '<span>qux</span>'
142+
}
143+
}).$mount()
144+
expect(vm.$el.children[0].textContent).toBe('hello')
145+
expect(vm.$el.children[1].innerHTML).toBe('<span>qux</span>')
146+
})
147+
136148
it('.camel modifier', () => {
137149
const vm = new Vue({
138150
template: '<svg :view-box.camel="viewBox"></svg>',

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

+10
Original file line numberDiff line numberDiff line change
@@ -535,6 +535,16 @@ describe('parser', () => {
535535
expect('The value for a v-bind expression cannot be empty. Found in "v-bind:empty-msg"').toHaveBeenWarned()
536536
})
537537

538+
it('v-bind.prop shorthand syntax', () => {
539+
const ast = parse('<div .id="foo"></div>', baseOptions)
540+
expect(ast.props).toEqual([{ name: 'id', value: 'foo'}])
541+
})
542+
543+
it('v-bind.prop shorthand syntax w/ modifiers', () => {
544+
const ast = parse('<div .id.mod="foo"></div>', baseOptions)
545+
expect(ast.props).toEqual([{ name: 'id', value: 'foo'}])
546+
})
547+
538548
// #6887
539549
it('special case static attribute that must be props', () => {
540550
const ast = parse('<video muted></video>', baseOptions)

0 commit comments

Comments
 (0)