Skip to content

Commit 0f7c443

Browse files
committed
fix: block unnecessary input event on textarea placeholder in IE
close #7138
1 parent d2e1d49 commit 0f7c443

File tree

2 files changed

+36
-3
lines changed

2 files changed

+36
-3
lines changed

src/platforms/web/runtime/modules/attrs.js

+18-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/* @flow */
22

3-
import { isIE9, isEdge } from 'core/util/env'
3+
import { isIE, isIE9, isEdge } from 'core/util/env'
44

55
import {
66
extend,
@@ -44,7 +44,7 @@ function updateAttrs (oldVnode: VNodeWithData, vnode: VNodeWithData) {
4444
// #4391: in IE9, setting type can reset value for input[type=radio]
4545
// #6666: IE/Edge forces progress value down to 1 before setting a max
4646
/* istanbul ignore if */
47-
if ((isIE9 || isEdge) && attrs.value !== oldAttrs.value) {
47+
if ((isIE || isEdge) && attrs.value !== oldAttrs.value) {
4848
setAttr(elm, 'value', attrs.value)
4949
}
5050
for (key in oldAttrs) {
@@ -84,6 +84,22 @@ function setAttr (el: Element, key: string, value: any) {
8484
if (isFalsyAttrValue(value)) {
8585
el.removeAttribute(key)
8686
} else {
87+
// #7138: IE10 & 11 fires input event when setting placeholder on
88+
// <textarea>... block the first input event and remove the blocker
89+
// immediately.
90+
/* istanbul ignore if */
91+
if (
92+
isIE && !isIE9 &&
93+
el.tagName === 'TEXTAREA' &&
94+
key === 'placeholder' && !el.__ieph
95+
) {
96+
const blocker = e => {
97+
e.stopImmediatePropagation()
98+
el.removeEventListener('input', blocker)
99+
}
100+
el.addEventListener('input', blocker)
101+
el.__ieph = true /* IE placeholder patched */
102+
}
87103
el.setAttribute(key, value)
88104
}
89105
}

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

+18-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import Vue from 'vue'
2-
import { isIE9, isAndroid } from 'core/util/env'
2+
import { isIE9, isIE, isAndroid } from 'core/util/env'
33

44
describe('Directive v-model text', () => {
55
it('should update value both ways', done => {
@@ -412,4 +412,21 @@ describe('Directive v-model text', () => {
412412
}).then(done)
413413
})
414414
}
415+
416+
// #7138
417+
if (isIE && !isIE9) {
418+
it('should not fire input on initial render of textarea with placeholder in IE10/11', done => {
419+
const el = document.createElement('div')
420+
document.body.appendChild(el)
421+
const vm = new Vue({
422+
el,
423+
data: { foo: null },
424+
template: `<textarea v-model="foo" placeholder="bar"></textarea>`
425+
})
426+
setTimeout(() => {
427+
expect(vm.foo).toBe(null)
428+
done()
429+
}, 17)
430+
})
431+
}
415432
})

0 commit comments

Comments
 (0)