Skip to content

Commit a64ff19

Browse files
haoqunjiangyyx990803
authored andcommitted
fix(lifecycle): updated should not be called after component being destroyed (#8381)
fix #8076
1 parent a71853b commit a64ff19

File tree

2 files changed

+38
-1
lines changed

2 files changed

+38
-1
lines changed

src/core/observer/scheduler.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -98,7 +98,7 @@ function callUpdatedHooks (queue) {
9898
while (i--) {
9999
const watcher = queue[i]
100100
const vm = watcher.vm
101-
if (vm._watcher === watcher && vm._isMounted) {
101+
if (vm._watcher === watcher && vm._isMounted && !vm._isDestroyed) {
102102
callHook(vm, 'updated')
103103
}
104104
}

test/unit/features/options/lifecycle.spec.js

+37
Original file line numberDiff line numberDiff line change
@@ -199,6 +199,43 @@ describe('Options lifecycle hooks', () => {
199199
expect(calls).toEqual(['child', 'parent'])
200200
}).then(done)
201201
})
202+
203+
// #8076
204+
it('should not be called after destroy', done => {
205+
const updated = jasmine.createSpy('updated')
206+
const destroyed = jasmine.createSpy('destroyed')
207+
208+
Vue.component('todo', {
209+
template: '<div>{{todo.done}}</div>',
210+
props: ['todo'],
211+
destroyed,
212+
updated
213+
})
214+
215+
const vm = new Vue({
216+
template: `
217+
<div>
218+
<todo v-for="t in pendingTodos" :todo="t" :key="t.id"></todo>
219+
</div>
220+
`,
221+
data () {
222+
return {
223+
todos: [{ id: 1, done: false }]
224+
}
225+
},
226+
computed: {
227+
pendingTodos () {
228+
return this.todos.filter(t => !t.done)
229+
}
230+
}
231+
}).$mount()
232+
233+
vm.todos[0].done = true
234+
waitForUpdate(() => {
235+
expect(destroyed).toHaveBeenCalled()
236+
expect(updated).not.toHaveBeenCalled()
237+
}).then(done)
238+
})
202239
})
203240

204241
describe('beforeDestroy', () => {

0 commit comments

Comments
 (0)