Skip to content

Commit 1e14603

Browse files
Robert Pembertonyyx990803
Robert Pemberton
authored andcommitted
feat: warn if both v-model and v-bind:value used on same element (#7056)
* test($compile): warn if v-model and :value used on same text input #7048 * test($compile): make v-model and v-bind:value warning apply to all but exceptions #7048 * test($compile): move v-model/:value conflict warner to model.js #7048 * style: split long warning messages onto new lines
1 parent a2cd14c commit 1e14603

File tree

2 files changed

+41
-0
lines changed

2 files changed

+41
-0
lines changed

src/platforms/web/compiler/directives/model.js

+15
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ export default function model (
2121
const modifiers = dir.modifiers
2222
const tag = el.tag
2323
const type = el.attrsMap.type
24+
const attrsMap = el.attrsMap
2425

2526
if (process.env.NODE_ENV !== 'production') {
2627
// inputs with type="file" are read only and setting the input's
@@ -31,6 +32,20 @@ export default function model (
3132
`File inputs are read only. Use a v-on:change listener instead.`
3233
)
3334
}
35+
36+
// warn if v-bind:value conflicts with v-model
37+
if (
38+
(attrsMap['v-bind:value'] || attrsMap[':value']) &&
39+
type !== 'checkbox' &&
40+
type !== 'radio' &&
41+
tag !== 'select'
42+
) {
43+
const vBindValue = attrsMap['v-bind:value'] ? 'v-bind:value' : ':value'
44+
warn(
45+
`${vBindValue} conflicts with v-model on the same element ` +
46+
'because the latter already expands to a value binding internally'
47+
)
48+
}
3449
}
3550

3651
if (el.component) {

test/unit/features/directives/model-text.spec.js

+26
Original file line numberDiff line numberDiff line change
@@ -250,6 +250,32 @@ describe('Directive v-model text', () => {
250250
expect('You are binding v-model directly to a v-for iteration alias').toHaveBeenWarned()
251251
})
252252

253+
it('warn if v-model and v-bind:value conflict', () => {
254+
new Vue({
255+
data: {
256+
test: 'foo'
257+
},
258+
template: '<input type="text" v-model="test" v-bind:value="test"/>'
259+
}).$mount()
260+
expect(
261+
'v-bind:value conflicts with v-model on the same element because the latter already ' +
262+
'expands to a value binding internally'
263+
).toHaveBeenWarned()
264+
})
265+
266+
it('warn if v-model and :value conflict', () => {
267+
new Vue({
268+
data: {
269+
test: 'foo'
270+
},
271+
template: '<input type="text" v-model="test" :value="test"/>'
272+
}).$mount()
273+
expect(
274+
':value conflicts with v-model on the same element because the latter already ' +
275+
'expands to a value binding internally'
276+
).toHaveBeenWarned()
277+
})
278+
253279
if (!isAndroid) {
254280
it('does not trigger extra input events with single compositionend', () => {
255281
const spy = jasmine.createSpy()

0 commit comments

Comments
 (0)