Skip to content

Commit d4d553c

Browse files
jkzingyyx990803
authored andcommitted
fix(v-model): avoid unnecessary change event on select options change
fix #6193 via #6194
1 parent 41d774d commit d4d553c

File tree

2 files changed

+41
-1
lines changed

2 files changed

+41
-1
lines changed

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

+12-1
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,14 @@ export default {
5353
const prevOptions = el._vOptions
5454
const curOptions = el._vOptions = [].map.call(el.options, getValue)
5555
if (curOptions.some((o, i) => !looseEqual(o, prevOptions[i]))) {
56-
trigger(el, 'change')
56+
// trigger change event if
57+
// no matching option found for at least one value
58+
const needReset = el.multiple
59+
? binding.value.some(v => hasNoMatchingOption(v, curOptions))
60+
: binding.value !== binding.oldValue && hasNoMatchingOption(binding.value, curOptions)
61+
if (needReset) {
62+
trigger(el, 'change')
63+
}
5764
}
5865
}
5966
}
@@ -104,6 +111,10 @@ function actuallySetSelected (el, binding, vm) {
104111
}
105112
}
106113

114+
function hasNoMatchingOption (value, options) {
115+
return options.every(o => !looseEqual(o, value))
116+
}
117+
107118
function getValue (option) {
108119
return '_value' in option
109120
? option._value

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

+29
Original file line numberDiff line numberDiff line change
@@ -517,4 +517,33 @@ describe('Directive v-model select', () => {
517517
expect(vm.test).toBe('2')
518518
}).then(done)
519519
})
520+
521+
// #6193
522+
it('should not trigger change event when matching option can be found for each value', done => {
523+
const spy = jasmine.createSpy()
524+
const vm = new Vue({
525+
data: {
526+
options: ['1']
527+
},
528+
computed: {
529+
test: {
530+
get () {
531+
return '1'
532+
},
533+
set () {
534+
spy()
535+
}
536+
}
537+
},
538+
template:
539+
'<select v-model="test">' +
540+
'<option :key="opt" v-for="opt in options" :value="opt">{{ opt }}</option>' +
541+
'</select>'
542+
}).$mount()
543+
544+
vm.options = ['1', '2']
545+
waitForUpdate(() => {
546+
expect(spy).not.toHaveBeenCalled()
547+
}).then(done)
548+
})
520549
})

0 commit comments

Comments
 (0)