Skip to content

Commit 6f312d6

Browse files
committed
fix(v-model): fix input listener with modifier blocking v-model update
fix #6552
1 parent a977740 commit 6f312d6

File tree

2 files changed

+83
-2
lines changed

2 files changed

+83
-2
lines changed

src/core/vdom/helpers/update-listeners.js

+22-2
Original file line numberDiff line numberDiff line change
@@ -5,18 +5,22 @@ import { cached, isUndef } from 'shared/util'
55

66
const normalizeEvent = cached((name: string): {
77
name: string,
8+
plain: boolean,
89
once: boolean,
910
capture: boolean,
10-
passive: boolean
11+
passive: boolean,
12+
handler?: Function
1113
} => {
1214
const passive = name.charAt(0) === '&'
1315
name = passive ? name.slice(1) : name
1416
const once = name.charAt(0) === '~' // Prefixed last, checked first
1517
name = once ? name.slice(1) : name
1618
const capture = name.charAt(0) === '!'
1719
name = capture ? name.slice(1) : name
20+
const plain = !(passive || once || capture)
1821
return {
1922
name,
23+
plain,
2024
once,
2125
capture,
2226
passive
@@ -40,6 +44,11 @@ export function createFnInvoker (fns: Function | Array<Function>): Function {
4044
return invoker
4145
}
4246

47+
// #6552
48+
function prioritizePlainEvents (a, b) {
49+
return a.plain ? -1 : b.plain ? 1 : 0
50+
}
51+
4352
export function updateListeners (
4453
on: Object,
4554
oldOn: Object,
@@ -48,10 +57,13 @@ export function updateListeners (
4857
vm: Component
4958
) {
5059
let name, cur, old, event
60+
const toAdd = []
61+
let hasModifier = false
5162
for (name in on) {
5263
cur = on[name]
5364
old = oldOn[name]
5465
event = normalizeEvent(name)
66+
if (!event.plain) hasModifier = true
5567
if (isUndef(cur)) {
5668
process.env.NODE_ENV !== 'production' && warn(
5769
`Invalid handler for event "${event.name}": got ` + String(cur),
@@ -61,12 +73,20 @@ export function updateListeners (
6173
if (isUndef(cur.fns)) {
6274
cur = on[name] = createFnInvoker(cur)
6375
}
64-
add(event.name, cur, event.once, event.capture, event.passive)
76+
event.handler = cur
77+
toAdd.push(event)
6578
} else if (cur !== old) {
6679
old.fns = cur
6780
on[name] = old
6881
}
6982
}
83+
if (toAdd.length) {
84+
if (hasModifier) toAdd.sort(prioritizePlainEvents)
85+
for (let i = 0; i < toAdd.length; i++) {
86+
const event = toAdd[i]
87+
add(event.name, event.handler, event.once, event.capture, event.passive)
88+
}
89+
}
7090
for (name in oldOn) {
7191
if (isUndef(on[name])) {
7292
event = normalizeEvent(name)

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

+61
Original file line numberDiff line numberDiff line change
@@ -293,5 +293,66 @@ describe('Directive v-model text', () => {
293293
triggerEvent(vm.$el, 'compositionend')
294294
expect(spy.calls.count()).toBe(2)
295295
})
296+
297+
// #4392
298+
it('should not update value with modifiers when in focus if post-conversion values are the same', done => {
299+
const vm = new Vue({
300+
data: {
301+
a: 1,
302+
foo: false
303+
},
304+
template: '<div>{{ foo }}<input ref="input" v-model.number="a"></div>'
305+
}).$mount()
306+
307+
document.body.appendChild(vm.$el)
308+
vm.$refs.input.focus()
309+
vm.$refs.input.value = '1.000'
310+
vm.foo = true
311+
312+
waitForUpdate(() => {
313+
expect(vm.$refs.input.value).toBe('1.000')
314+
}).then(done)
315+
})
316+
317+
// #6552
318+
// Root cause: input listeners with modifiers are added as a separate
319+
// DOM listener. If a change is triggered inside this listener, an update
320+
// will happen before the second listener is fired! (obviously microtasks
321+
// can be processed in between DOM events on the same element)
322+
// This causes the domProps module to receive state that has not been
323+
// updated by v-model yet (because v-model's listener has not fired yet)
324+
// Solution: make sure to always fire v-model's listener first
325+
it('should not block input when another input listener with modifier is used', done => {
326+
const vm = new Vue({
327+
data: {
328+
a: 'a',
329+
foo: false
330+
},
331+
template: `
332+
<div>
333+
<input ref="input" v-model="a" @input.capture="onInput">{{ a }}
334+
<div v-if="foo">foo</div>
335+
</div>
336+
`,
337+
methods: {
338+
onInput (e) {
339+
this.foo = true
340+
}
341+
}
342+
}).$mount()
343+
344+
document.body.appendChild(vm.$el)
345+
vm.$refs.input.focus()
346+
vm.$refs.input.value = 'b'
347+
triggerEvent(vm.$refs.input, 'input')
348+
349+
// not using wait for update here because there will be two update cycles
350+
// one caused by onInput in the first listener
351+
setTimeout(() => {
352+
expect(vm.a).toBe('b')
353+
expect(vm.$refs.input.value).toBe('b')
354+
done()
355+
}, 16)
356+
})
296357
}
297358
})

0 commit comments

Comments
 (0)