@@ -5,6 +5,7 @@ import { warn } from './warning'
5
5
6
6
export interface SchedulerJob extends Function {
7
7
id ?: number
8
+ pre ?: boolean
8
9
active ?: boolean
9
10
computed ?: boolean
10
11
/**
@@ -39,19 +40,13 @@ let isFlushPending = false
39
40
const queue : SchedulerJob [ ] = [ ]
40
41
let flushIndex = 0
41
42
42
- const pendingPreFlushCbs : SchedulerJob [ ] = [ ]
43
- let activePreFlushCbs : SchedulerJob [ ] | null = null
44
- let preFlushIndex = 0
45
-
46
43
const pendingPostFlushCbs : SchedulerJob [ ] = [ ]
47
44
let activePostFlushCbs : SchedulerJob [ ] | null = null
48
45
let postFlushIndex = 0
49
46
50
47
const resolvedPromise = /*#__PURE__*/ Promise . resolve ( ) as Promise < any >
51
48
let currentFlushPromise : Promise < void > | null = null
52
49
53
- let currentPreFlushParentJob : SchedulerJob | null = null
54
-
55
50
const RECURSION_LIMIT = 100
56
51
type CountMap = Map < SchedulerJob , number >
57
52
@@ -89,12 +84,11 @@ export function queueJob(job: SchedulerJob) {
89
84
// allow it recursively trigger itself - it is the user's responsibility to
90
85
// ensure it doesn't end up in an infinite loop.
91
86
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
+ )
98
92
) {
99
93
if ( job . id == null ) {
100
94
queue . push ( job )
@@ -119,71 +113,44 @@ export function invalidateJob(job: SchedulerJob) {
119
113
}
120
114
}
121
115
122
- function queueCb (
123
- cb : SchedulerJobs ,
124
- activeQueue : SchedulerJob [ ] | null ,
125
- pendingQueue : SchedulerJob [ ] ,
126
- index : number
127
- ) {
116
+ export function queuePostFlushCb ( cb : SchedulerJobs ) {
128
117
if ( ! isArray ( cb ) ) {
129
118
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
+ )
132
124
) {
133
- pendingQueue . push ( cb )
125
+ pendingPostFlushCbs . push ( cb )
134
126
}
135
127
} else {
136
128
// if cb is an array, it is a component lifecycle hook which can only be
137
129
// triggered by a job, which is already deduped in the main queue, so
138
130
// we can skip duplicate check here to improve perf
139
- pendingQueue . push ( ...cb )
131
+ pendingPostFlushCbs . push ( ...cb )
140
132
}
141
133
queueFlush ( )
142
134
}
143
135
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 ) ) {
172
144
continue
173
145
}
174
- activePreFlushCbs [ preFlushIndex ] ( )
146
+ queue . splice ( i , 1 )
147
+ i --
148
+ cb ( )
175
149
}
176
- activePreFlushCbs = null
177
- preFlushIndex = 0
178
- currentPreFlushParentJob = null
179
- // recursively flush until it drains
180
- flushPreFlushCbs ( seen , parentJob )
181
150
}
182
151
}
183
152
184
153
export function flushPostFlushCbs ( seen ?: CountMap ) {
185
- // flush any pre cbs queued during the flush (e.g. pre watchers)
186
- flushPreFlushCbs ( )
187
154
if ( pendingPostFlushCbs . length ) {
188
155
const deduped = [ ...new Set ( pendingPostFlushCbs ) ]
189
156
pendingPostFlushCbs . length = 0
@@ -222,23 +189,30 @@ export function flushPostFlushCbs(seen?: CountMap) {
222
189
const getId = ( job : SchedulerJob ) : number =>
223
190
job . id == null ? Infinity : job . id
224
191
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
+
225
201
function flushJobs ( seen ?: CountMap ) {
226
202
isFlushPending = false
227
203
isFlushing = true
228
204
if ( __DEV__ ) {
229
205
seen = seen || new Map ( )
230
206
}
231
207
232
- flushPreFlushCbs ( seen )
233
-
234
208
// Sort queue before flush.
235
209
// This ensures that:
236
210
// 1. Components are updated from parent to child. (because parent is always
237
211
// created before the child so its render effect will have smaller
238
212
// priority number)
239
213
// 2. If a component is unmounted during a parent component's update,
240
214
// its update can be skipped.
241
- queue . sort ( ( a , b ) => getId ( a ) - getId ( b ) )
215
+ queue . sort ( comparator )
242
216
243
217
// conditional usage of checkRecursiveUpdate must be determined out of
244
218
// try ... catch block since Rollup by default de-optimizes treeshaking
@@ -270,11 +244,7 @@ function flushJobs(seen?: CountMap) {
270
244
currentFlushPromise = null
271
245
// some postFlushCb queued jobs!
272
246
// keep flushing until it drains.
273
- if (
274
- queue . length ||
275
- pendingPreFlushCbs . length ||
276
- pendingPostFlushCbs . length
277
- ) {
247
+ if ( queue . length || pendingPostFlushCbs . length ) {
278
248
flushJobs ( seen )
279
249
}
280
250
}
0 commit comments