From 3d29ba863b89fd90dabd0856c0507eacdf5fef22 Mon Sep 17 00:00:00 2001
From: zrh122 <46116414+zrh122@users.noreply.github.com>
Date: Thu, 5 Aug 2021 17:21:12 +0800
Subject: [PATCH] fix(v-on): add removing all dom event listeners when vnode
destroyed (#10085)
---
src/platforms/web/runtime/modules/events.js | 8 +++--
.../component/component-keep-alive.spec.js | 32 +++++++++++++++++++
2 files changed, 38 insertions(+), 2 deletions(-)
diff --git a/src/platforms/web/runtime/modules/events.js b/src/platforms/web/runtime/modules/events.js
index ecb08af32d..3c60fd5d1a 100644
--- a/src/platforms/web/runtime/modules/events.js
+++ b/src/platforms/web/runtime/modules/events.js
@@ -5,6 +5,7 @@ import { updateListeners } from 'core/vdom/helpers/index'
import { isIE, isFF, supportsPassive, isUsingMicroTask } from 'core/util/index'
import { RANGE_TOKEN, CHECKBOX_RADIO_TOKEN } from 'web/compiler/directives/model'
import { currentFlushTimestamp } from 'core/observer/scheduler'
+import { emptyNode } from 'core/vdom/patch'
// normalize v-model event tokens that can only be determined at runtime.
// it's important to place the event as the first in the array because
@@ -108,7 +109,9 @@ function updateDOMListeners (oldVnode: VNodeWithData, vnode: VNodeWithData) {
}
const on = vnode.data.on || {}
const oldOn = oldVnode.data.on || {}
- target = vnode.elm
+ // vnode is empty when removing all listeners,
+ // and use old vnode dom element
+ target = vnode.elm || oldVnode.elm
normalizeEvents(on)
updateListeners(on, oldOn, add, remove, createOnceHandler, vnode.context)
target = undefined
@@ -116,5 +119,6 @@ function updateDOMListeners (oldVnode: VNodeWithData, vnode: VNodeWithData) {
export default {
create: updateDOMListeners,
- update: updateDOMListeners
+ update: updateDOMListeners,
+ destroy: (vnode: VNodeWithData) => updateDOMListeners(vnode, emptyNode)
}
diff --git a/test/unit/features/component/component-keep-alive.spec.js b/test/unit/features/component/component-keep-alive.spec.js
index c873495882..9d30d2c7d5 100644
--- a/test/unit/features/component/component-keep-alive.spec.js
+++ b/test/unit/features/component/component-keep-alive.spec.js
@@ -1249,5 +1249,37 @@ describe('Component keep-alive', () => {
}).then(done)
}
})
+
+ // #10083
+ it('should not attach event handler repeatedly', done => {
+ const vm = new Vue({
+ template: `
+
+
+
+ `,
+ data: { showBtn: true, n: 0 },
+ methods: {
+ add () {
+ this.n++
+ }
+ },
+ components: {
+ btn: { template: '' }
+ }
+ }).$mount()
+
+ const btn = vm.$el
+ expect(vm.n).toBe(0)
+ btn.click()
+ expect(vm.n).toBe(1)
+ vm.showBtn = false
+ waitForUpdate(() => {
+ vm.showBtn = true
+ }).then(() => {
+ btn.click()
+ expect(vm.n).toBe(2)
+ }).then(done)
+ })
}
})