Skip to content

Commit 38c2923

Browse files
committed
fix(scheduler): job ordering when the post queue is flushing (vuejs#12090)
1 parent 5e7005f commit 38c2923

File tree

2 files changed

+61
-15
lines changed

2 files changed

+61
-15
lines changed

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

+54
Original file line numberDiff line numberDiff line change
@@ -441,6 +441,29 @@ describe('scheduler', () => {
441441
await nextTick()
442442
expect(calls).toEqual(['job1', 'job2', 'cb1', 'cb2'])
443443
})
444+
445+
test('jobs added during post flush are ordered correctly', async () => {
446+
const calls: string[] = []
447+
448+
const job1: SchedulerJob = () => {
449+
calls.push('job1')
450+
}
451+
job1.id = 1
452+
453+
const job2: SchedulerJob = () => {
454+
calls.push('job2')
455+
}
456+
job2.id = 2
457+
458+
queuePostFlushCb(() => {
459+
queueJob(job2)
460+
queueJob(job1)
461+
})
462+
463+
await nextTick()
464+
465+
expect(calls).toEqual(['job1', 'job2'])
466+
})
444467
})
445468

446469
test('sort job based on id', async () => {
@@ -758,6 +781,37 @@ describe('scheduler', () => {
758781
expect(spy).toHaveBeenCalledTimes(1)
759782
})
760783

784+
test('flushPreFlushCbs inside a post job', async () => {
785+
const calls: string[] = []
786+
const callsAfterFlush: string[] = []
787+
788+
const job1: SchedulerJob = () => {
789+
calls.push('job1')
790+
}
791+
job1.id = 1
792+
job1.flags! |= SchedulerJobFlags.PRE
793+
794+
const job2: SchedulerJob = () => {
795+
calls.push('job2')
796+
}
797+
job2.id = 2
798+
job2.flags! |= SchedulerJobFlags.PRE
799+
800+
queuePostFlushCb(() => {
801+
queueJob(job2)
802+
queueJob(job1)
803+
804+
// e.g. nested app.mount() call
805+
flushPreFlushCbs()
806+
callsAfterFlush.push(...calls)
807+
})
808+
809+
await nextTick()
810+
811+
expect(callsAfterFlush).toEqual(['job1', 'job2'])
812+
expect(calls).toEqual(['job1', 'job2'])
813+
})
814+
761815
it('nextTick should return promise', async () => {
762816
const fn = vi.fn(() => {
763817
return 1

packages/runtime-core/src/scheduler.ts

+7-15
Original file line numberDiff line numberDiff line change
@@ -40,13 +40,10 @@ export interface SchedulerJob extends Function {
4040
}
4141

4242
export type SchedulerJobs = SchedulerJob | SchedulerJob[]
43-
44-
let isFlushing = false
45-
let isFlushPending = false
4643
let auto = true
4744

4845
const queue: SchedulerJob[] = []
49-
let flushIndex = 0
46+
let flushIndex = -1
5047

5148
const pendingPostFlushCbs: SchedulerJob[] = []
5249
let activePostFlushCbs: SchedulerJob[] | null = null
@@ -76,7 +73,7 @@ export function nextTick<T = void, R = void>(
7673
// watcher should be inserted immediately before the update job. This allows
7774
// watchers to be skipped if the component is unmounted by the parent update.
7875
function findInsertionIndex(id: number) {
79-
let start = isFlushing ? flushIndex + 1 : 0
76+
let start = flushIndex + 1
8077
let end = queue.length
8178

8279
while (start < end) {
@@ -196,8 +193,7 @@ export function endFlush(): void {
196193
}
197194

198195
function queueFlush() {
199-
if (!isFlushing && !isFlushPending) {
200-
isFlushPending = true
196+
if (!currentFlushPromise) {
201197
currentFlushPromise = Promise.all([
202198
resolvedPromise.then(flushJobs),
203199
trackManualFlush(),
@@ -239,8 +235,8 @@ export function queuePostFlushCb(cb: SchedulerJobs): {
239235
export function flushPreFlushCbs(
240236
instance?: ComponentInternalInstance,
241237
seen?: CountMap,
242-
// if currently flushing, skip the current job itself
243-
i: number = isFlushing ? flushIndex + 1 : 0,
238+
// skip the current job
239+
i: number = flushIndex + 1,
244240
): void {
245241
if (__DEV__) {
246242
seen = seen || new Map()
@@ -382,8 +378,6 @@ export function flushPostJobsUntil(index: number, clear: boolean = true): void {
382378
}
383379

384380
function flushJobs(seen?: CountMap) {
385-
isFlushPending = false
386-
isFlushing = true
387381
if (!auto) return
388382
if (__DEV__) {
389383
seen = seen || new Map()
@@ -427,16 +421,14 @@ function flushJobs(seen?: CountMap) {
427421
}
428422
}
429423

430-
flushIndex = 0
424+
flushIndex = -1
431425
queue.length = 0
432426

433427
flushPostFlushCbs(seen)
434428

435429
endFlush()
436-
isFlushing = false
437430
currentFlushPromise = null
438-
// some postFlushCb queued jobs!
439-
// keep flushing until it drains.
431+
// If new jobs have been added to either queue, keep flushing
440432
if (queue.length || pendingPostFlushCbs.length) {
441433
flushJobs(seen)
442434
}

0 commit comments

Comments
 (0)