Skip to content

Commit 78c199d

Browse files
committedAug 15, 2022
fix(watch): flush:pre watchers should not fire if state change causes
owner component to unmount fix #2291
1 parent a95554d commit 78c199d

File tree

5 files changed

+68
-142
lines changed

5 files changed

+68
-142
lines changed
 

‎packages/runtime-core/__tests__/apiWatch.spec.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -509,7 +509,8 @@ describe('api: watch', () => {
509509
expect(cb).not.toHaveBeenCalled()
510510
})
511511

512-
it('should fire on component unmount w/ flush: pre', async () => {
512+
// #2291
513+
it('should not fire on component unmount w/ flush: pre', async () => {
513514
const toggle = ref(true)
514515
const cb = jest.fn()
515516
const Comp = {
@@ -527,7 +528,7 @@ describe('api: watch', () => {
527528
expect(cb).not.toHaveBeenCalled()
528529
toggle.value = false
529530
await nextTick()
530-
expect(cb).toHaveBeenCalledTimes(1)
531+
expect(cb).not.toHaveBeenCalled()
531532
})
532533

533534
// #1763

‎packages/runtime-core/__tests__/scheduler.spec.ts

+23-71
Original file line numberDiff line numberDiff line change
@@ -3,9 +3,8 @@ import {
33
nextTick,
44
queuePostFlushCb,
55
invalidateJob,
6-
queuePreFlushCb,
7-
flushPreFlushCbs,
8-
flushPostFlushCbs
6+
flushPostFlushCbs,
7+
flushPreFlushCbs
98
} from '../src/scheduler'
109

1110
describe('scheduler', () => {
@@ -114,65 +113,7 @@ describe('scheduler', () => {
114113
})
115114
})
116115

117-
describe('queuePreFlushCb', () => {
118-
it('basic usage', async () => {
119-
const calls: string[] = []
120-
const cb1 = () => {
121-
calls.push('cb1')
122-
}
123-
const cb2 = () => {
124-
calls.push('cb2')
125-
}
126-
127-
queuePreFlushCb(cb1)
128-
queuePreFlushCb(cb2)
129-
130-
expect(calls).toEqual([])
131-
await nextTick()
132-
expect(calls).toEqual(['cb1', 'cb2'])
133-
})
134-
135-
it('should dedupe queued preFlushCb', async () => {
136-
const calls: string[] = []
137-
const cb1 = () => {
138-
calls.push('cb1')
139-
}
140-
const cb2 = () => {
141-
calls.push('cb2')
142-
}
143-
const cb3 = () => {
144-
calls.push('cb3')
145-
}
146-
147-
queuePreFlushCb(cb1)
148-
queuePreFlushCb(cb2)
149-
queuePreFlushCb(cb1)
150-
queuePreFlushCb(cb2)
151-
queuePreFlushCb(cb3)
152-
153-
expect(calls).toEqual([])
154-
await nextTick()
155-
expect(calls).toEqual(['cb1', 'cb2', 'cb3'])
156-
})
157-
158-
it('chained queuePreFlushCb', async () => {
159-
const calls: string[] = []
160-
const cb1 = () => {
161-
calls.push('cb1')
162-
// cb2 will be executed after cb1 at the same tick
163-
queuePreFlushCb(cb2)
164-
}
165-
const cb2 = () => {
166-
calls.push('cb2')
167-
}
168-
queuePreFlushCb(cb1)
169-
170-
await nextTick()
171-
expect(calls).toEqual(['cb1', 'cb2'])
172-
})
173-
})
174-
175-
describe('queueJob w/ queuePreFlushCb', () => {
116+
describe('pre flush jobs', () => {
176117
it('queueJob inside preFlushCb', async () => {
177118
const calls: string[] = []
178119
const job1 = () => {
@@ -183,8 +124,9 @@ describe('scheduler', () => {
183124
calls.push('cb1')
184125
queueJob(job1)
185126
}
127+
cb1.pre = true
186128

187-
queuePreFlushCb(cb1)
129+
queueJob(cb1)
188130
await nextTick()
189131
expect(calls).toEqual(['cb1', 'job1'])
190132
})
@@ -194,17 +136,23 @@ describe('scheduler', () => {
194136
const job1 = () => {
195137
calls.push('job1')
196138
}
139+
job1.id = 1
140+
197141
const cb1 = () => {
198142
calls.push('cb1')
199143
queueJob(job1)
200144
// cb2 should execute before the job
201-
queuePreFlushCb(cb2)
145+
queueJob(cb2)
202146
}
147+
cb1.pre = true
148+
203149
const cb2 = () => {
204150
calls.push('cb2')
205151
}
152+
cb2.pre = true
153+
cb2.id = 1
206154

207-
queuePreFlushCb(cb1)
155+
queueJob(cb1)
208156
await nextTick()
209157
expect(calls).toEqual(['cb1', 'cb2', 'job1'])
210158
})
@@ -216,19 +164,21 @@ describe('scheduler', () => {
216164
// when updating the props of a child component. This is handled
217165
// directly inside `updateComponentPreRender` to avoid non atomic
218166
// cb triggers (#1763)
219-
queuePreFlushCb(cb1)
220-
queuePreFlushCb(cb2)
221-
flushPreFlushCbs(undefined, job1)
167+
queueJob(cb1)
168+
queueJob(cb2)
169+
flushPreFlushCbs()
222170
calls.push('job1')
223171
}
224172
const cb1 = () => {
225173
calls.push('cb1')
226174
// a cb triggers its parent job, which should be skipped
227175
queueJob(job1)
228176
}
177+
cb1.pre = true
229178
const cb2 = () => {
230179
calls.push('cb2')
231180
}
181+
cb2.pre = true
232182

233183
queueJob(job1)
234184
await nextTick()
@@ -237,12 +187,14 @@ describe('scheduler', () => {
237187

238188
// #3806
239189
it('queue preFlushCb inside postFlushCb', async () => {
240-
const cb = jest.fn()
190+
const spy = jest.fn()
191+
const cb = () => spy()
192+
cb.pre = true
241193
queuePostFlushCb(() => {
242-
queuePreFlushCb(cb)
194+
queueJob(cb)
243195
})
244196
await nextTick()
245-
expect(cb).toHaveBeenCalled()
197+
expect(spy).toHaveBeenCalled()
246198
})
247199
})
248200

‎packages/runtime-core/src/apiWatch.ts

+4-2
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ import {
99
EffectScheduler,
1010
DebuggerOptions
1111
} from '@vue/reactivity'
12-
import { SchedulerJob, queuePreFlushCb } from './scheduler'
12+
import { SchedulerJob, queueJob } from './scheduler'
1313
import {
1414
EMPTY_OBJ,
1515
isObject,
@@ -345,7 +345,9 @@ function doWatch(
345345
scheduler = () => queuePostRenderEffect(job, instance && instance.suspense)
346346
} else {
347347
// default: 'pre'
348-
scheduler = () => queuePreFlushCb(job)
348+
job.pre = true
349+
if (instance) job.id = instance.uid
350+
scheduler = () => queueJob(job)
349351
}
350352

351353
const effect = new ReactiveEffect(getter, scheduler)

‎packages/runtime-core/src/renderer.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1590,7 +1590,7 @@ function baseCreateRenderer(
15901590
pauseTracking()
15911591
// props update may have triggered pre-flush watchers.
15921592
// flush them before the render update.
1593-
flushPreFlushCbs(undefined, instance.update)
1593+
flushPreFlushCbs()
15941594
resetTracking()
15951595
}
15961596

@@ -2331,6 +2331,7 @@ function baseCreateRenderer(
23312331
} else {
23322332
patch(container._vnode || null, vnode, container, null, null, null, isSVG)
23332333
}
2334+
flushPreFlushCbs()
23342335
flushPostFlushCbs()
23352336
container._vnode = vnode
23362337
}

‎packages/runtime-core/src/scheduler.ts

+36-66
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import { warn } from './warning'
55

66
export interface SchedulerJob extends Function {
77
id?: number
8+
pre?: boolean
89
active?: boolean
910
computed?: boolean
1011
/**
@@ -39,19 +40,13 @@ let isFlushPending = false
3940
const queue: SchedulerJob[] = []
4041
let flushIndex = 0
4142

42-
const pendingPreFlushCbs: SchedulerJob[] = []
43-
let activePreFlushCbs: SchedulerJob[] | null = null
44-
let preFlushIndex = 0
45-
4643
const pendingPostFlushCbs: SchedulerJob[] = []
4744
let activePostFlushCbs: SchedulerJob[] | null = null
4845
let postFlushIndex = 0
4946

5047
const resolvedPromise = /*#__PURE__*/ Promise.resolve() as Promise<any>
5148
let currentFlushPromise: Promise<void> | null = null
5249

53-
let currentPreFlushParentJob: SchedulerJob | null = null
54-
5550
const RECURSION_LIMIT = 100
5651
type CountMap = Map<SchedulerJob, number>
5752

@@ -89,12 +84,11 @@ export function queueJob(job: SchedulerJob) {
8984
// allow it recursively trigger itself - it is the user's responsibility to
9085
// ensure it doesn't end up in an infinite loop.
9186
if (
92-
(!queue.length ||
93-
!queue.includes(
94-
job,
95-
isFlushing && job.allowRecurse ? flushIndex + 1 : flushIndex
96-
)) &&
97-
job !== currentPreFlushParentJob
87+
!queue.length ||
88+
!queue.includes(
89+
job,
90+
isFlushing && job.allowRecurse ? flushIndex + 1 : flushIndex
91+
)
9892
) {
9993
if (job.id == null) {
10094
queue.push(job)
@@ -119,71 +113,44 @@ export function invalidateJob(job: SchedulerJob) {
119113
}
120114
}
121115

122-
function queueCb(
123-
cb: SchedulerJobs,
124-
activeQueue: SchedulerJob[] | null,
125-
pendingQueue: SchedulerJob[],
126-
index: number
127-
) {
116+
export function queuePostFlushCb(cb: SchedulerJobs) {
128117
if (!isArray(cb)) {
129118
if (
130-
!activeQueue ||
131-
!activeQueue.includes(cb, cb.allowRecurse ? index + 1 : index)
119+
!activePostFlushCbs ||
120+
!activePostFlushCbs.includes(
121+
cb,
122+
cb.allowRecurse ? postFlushIndex + 1 : postFlushIndex
123+
)
132124
) {
133-
pendingQueue.push(cb)
125+
pendingPostFlushCbs.push(cb)
134126
}
135127
} else {
136128
// if cb is an array, it is a component lifecycle hook which can only be
137129
// triggered by a job, which is already deduped in the main queue, so
138130
// we can skip duplicate check here to improve perf
139-
pendingQueue.push(...cb)
131+
pendingPostFlushCbs.push(...cb)
140132
}
141133
queueFlush()
142134
}
143135

144-
export function queuePreFlushCb(cb: SchedulerJob) {
145-
queueCb(cb, activePreFlushCbs, pendingPreFlushCbs, preFlushIndex)
146-
}
147-
148-
export function queuePostFlushCb(cb: SchedulerJobs) {
149-
queueCb(cb, activePostFlushCbs, pendingPostFlushCbs, postFlushIndex)
150-
}
151-
152-
export function flushPreFlushCbs(
153-
seen?: CountMap,
154-
parentJob: SchedulerJob | null = null
155-
) {
156-
if (pendingPreFlushCbs.length) {
157-
currentPreFlushParentJob = parentJob
158-
activePreFlushCbs = [...new Set(pendingPreFlushCbs)]
159-
pendingPreFlushCbs.length = 0
160-
if (__DEV__) {
161-
seen = seen || new Map()
162-
}
163-
for (
164-
preFlushIndex = 0;
165-
preFlushIndex < activePreFlushCbs.length;
166-
preFlushIndex++
167-
) {
168-
if (
169-
__DEV__ &&
170-
checkRecursiveUpdates(seen!, activePreFlushCbs[preFlushIndex])
171-
) {
136+
export function flushPreFlushCbs(seen?: CountMap, i = flushIndex) {
137+
if (__DEV__) {
138+
seen = seen || new Map()
139+
}
140+
for (; i < queue.length; i++) {
141+
const cb = queue[i]
142+
if (cb && cb.pre) {
143+
if (__DEV__ && checkRecursiveUpdates(seen!, cb)) {
172144
continue
173145
}
174-
activePreFlushCbs[preFlushIndex]()
146+
queue.splice(i, 1)
147+
i--
148+
cb()
175149
}
176-
activePreFlushCbs = null
177-
preFlushIndex = 0
178-
currentPreFlushParentJob = null
179-
// recursively flush until it drains
180-
flushPreFlushCbs(seen, parentJob)
181150
}
182151
}
183152

184153
export function flushPostFlushCbs(seen?: CountMap) {
185-
// flush any pre cbs queued during the flush (e.g. pre watchers)
186-
flushPreFlushCbs()
187154
if (pendingPostFlushCbs.length) {
188155
const deduped = [...new Set(pendingPostFlushCbs)]
189156
pendingPostFlushCbs.length = 0
@@ -222,23 +189,30 @@ export function flushPostFlushCbs(seen?: CountMap) {
222189
const getId = (job: SchedulerJob): number =>
223190
job.id == null ? Infinity : job.id
224191

192+
const comparator = (a: SchedulerJob, b: SchedulerJob): number => {
193+
const diff = getId(a) - getId(b)
194+
if (diff === 0) {
195+
if (a.pre && !b.pre) return -1
196+
if (b.pre && !a.pre) return 1
197+
}
198+
return diff
199+
}
200+
225201
function flushJobs(seen?: CountMap) {
226202
isFlushPending = false
227203
isFlushing = true
228204
if (__DEV__) {
229205
seen = seen || new Map()
230206
}
231207

232-
flushPreFlushCbs(seen)
233-
234208
// Sort queue before flush.
235209
// This ensures that:
236210
// 1. Components are updated from parent to child. (because parent is always
237211
// created before the child so its render effect will have smaller
238212
// priority number)
239213
// 2. If a component is unmounted during a parent component's update,
240214
// its update can be skipped.
241-
queue.sort((a, b) => getId(a) - getId(b))
215+
queue.sort(comparator)
242216

243217
// conditional usage of checkRecursiveUpdate must be determined out of
244218
// try ... catch block since Rollup by default de-optimizes treeshaking
@@ -270,11 +244,7 @@ function flushJobs(seen?: CountMap) {
270244
currentFlushPromise = null
271245
// some postFlushCb queued jobs!
272246
// keep flushing until it drains.
273-
if (
274-
queue.length ||
275-
pendingPreFlushCbs.length ||
276-
pendingPostFlushCbs.length
277-
) {
247+
if (queue.length || pendingPostFlushCbs.length) {
278248
flushJobs(seen)
279249
}
280250
}

0 commit comments

Comments
 (0)