Skip to content

Commit 3965e50

Browse files
Kingwlyyx990803
authored andcommitted
feat: support sync modifier for v-bind="object" (#5943)
close #5937
1 parent bd79d7e commit 3965e50

File tree

5 files changed

+59
-5
lines changed

5 files changed

+59
-5
lines changed

flow/component.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -120,7 +120,7 @@ declare interface Component {
120120
// renderSlot
121121
_t: (name: string, fallback: ?Array<VNode>, props: ?Object) => ?Array<VNode>;
122122
// apply v-bind object
123-
_b: (data: any, value: any, asProp?: boolean) => VNodeData;
123+
_b: (data: any, tag: string, value: any, asProp: boolean, isSync?: boolean) => VNodeData;
124124
// check custom keyCode
125125
_k: (eventKeyCode: number, key: string, builtInAlias: number | Array<number> | void) => boolean;
126126
// resolve scoped slots

src/compiler/directives/bind.js

+4-2
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,10 @@
22

33
export default function bind (el: ASTElement, dir: ASTDirective) {
44
el.wrapData = (code: string) => {
5-
return `_b(${code},'${el.tag}',${dir.value}${
6-
dir.modifiers && dir.modifiers.prop ? ',true' : ''
5+
return `_b(${code},'${el.tag}',${dir.value},${
6+
dir.modifiers && dir.modifiers.prop ? 'true' : 'false'
7+
}${
8+
dir.modifiers && dir.modifiers.sync ? ',true' : ''
79
})`
810
}
911
}

src/core/instance/render-helpers/bind-object-props.js

+9-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,8 @@ export function bindObjectProps (
1616
data: any,
1717
tag: string,
1818
value: any,
19-
asProp?: boolean
19+
asProp: boolean,
20+
isSync?: boolean
2021
): VNodeData {
2122
if (value) {
2223
if (!isObject(value)) {
@@ -44,6 +45,13 @@ export function bindObjectProps (
4445
}
4546
if (!(key in hash)) {
4647
hash[key] = value[key]
48+
49+
if (isSync) {
50+
const on = data.on || (data.on = {})
51+
on[`update:${key}`] = function ($event) {
52+
value[key] = $event
53+
}
54+
}
4755
}
4856
}
4957
}

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

+30
Original file line numberDiff line numberDiff line change
@@ -187,6 +187,36 @@ describe('Directive v-bind', () => {
187187
}).then(done)
188188
})
189189

190+
it('.sync modifier with bind object', done => {
191+
const vm = new Vue({
192+
template: `<test v-bind.sync="test"/>`,
193+
data: {
194+
test: {
195+
fooBar: 1
196+
}
197+
},
198+
components: {
199+
test: {
200+
props: ['fooBar'],
201+
template: `<div @click="handleUpdate">{{ fooBar }}</div>`,
202+
methods: {
203+
handleUpdate () {
204+
this.$emit('update:fooBar', 2)
205+
}
206+
}
207+
}
208+
}
209+
}).$mount()
210+
expect(vm.$el.textContent).toBe('1')
211+
triggerEvent(vm.$el, 'click')
212+
waitForUpdate(() => {
213+
expect(vm.$el.textContent).toBe('2')
214+
vm.test.fooBar = 3
215+
}).then(() => {
216+
expect(vm.$el.textContent).toBe('3')
217+
}).then(done)
218+
})
219+
190220
it('bind object with overwrite', done => {
191221
const vm = new Vue({
192222
template: '<input v-bind="test" id="foo" :class="test.value">',

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

+15-1
Original file line numberDiff line numberDiff line change
@@ -126,7 +126,21 @@ describe('codegen', () => {
126126
it('generate v-bind directive', () => {
127127
assertCodegen(
128128
'<p v-bind="test"></p>',
129-
`with(this){return _c('p',_b({},'p',test))}`
129+
`with(this){return _c('p',_b({},'p',test,false))}`
130+
)
131+
})
132+
133+
it('generate v-bind with prop directive', () => {
134+
assertCodegen(
135+
'<p v-bind.prop="test"></p>',
136+
`with(this){return _c('p',_b({},'p',test,true))}`
137+
)
138+
})
139+
140+
it('generate v-bind directive with sync modifier', () => {
141+
assertCodegen(
142+
'<p v-bind.sync="test"></p>',
143+
`with(this){return _c('p',_b({},'p',test,false,true))}`
130144
)
131145
})
132146

0 commit comments

Comments
 (0)