File tree 4 files changed +46
-2
lines changed
4 files changed +46
-2
lines changed Original file line number Diff line number Diff line change @@ -59,6 +59,15 @@ if (inBrowser && !isIE) {
59
59
}
60
60
}
61
61
62
+ const sortCompareFn = ( a : Watcher , b : Watcher ) : number => {
63
+ if ( a . post ) {
64
+ if ( ! b . post ) return 1
65
+ } else if ( b . post ) {
66
+ return - 1
67
+ }
68
+ return a . id - b . id
69
+ }
70
+
62
71
/**
63
72
* Flush both queues and run the watchers.
64
73
*/
@@ -75,7 +84,7 @@ function flushSchedulerQueue() {
75
84
// user watchers are created before the render watcher)
76
85
// 3. If a component is destroyed during a parent component's watcher run,
77
86
// its watchers can be skipped.
78
- queue . sort ( ( a , b ) => a . id - b . id )
87
+ queue . sort ( sortCompareFn )
79
88
80
89
// do not cache length because more watchers might be pushed
81
90
// as we run existing watchers
Original file line number Diff line number Diff line change @@ -58,6 +58,7 @@ export default class Watcher implements DepTarget {
58
58
noRecurse ?: boolean
59
59
getter : Function
60
60
value : any
61
+ post : boolean
61
62
62
63
// dev only
63
64
onTrack ?: ( ( event : DebuggerEvent ) => void ) | undefined
@@ -93,6 +94,7 @@ export default class Watcher implements DepTarget {
93
94
this . cb = cb
94
95
this . id = ++ uid // uid for batching
95
96
this . active = true
97
+ this . post = false
96
98
this . dirty = this . lazy // for lazy watchers
97
99
this . deps = [ ]
98
100
this . newDeps = [ ]
Original file line number Diff line number Diff line change @@ -313,7 +313,7 @@ function doWatch(
313
313
if ( flush === 'sync' ) {
314
314
watcher . update = watcher . run
315
315
} else if ( flush === 'post' ) {
316
- watcher . id = Infinity
316
+ watcher . post = true
317
317
watcher . update = ( ) => queueWatcher ( watcher )
318
318
} else {
319
319
// pre
Original file line number Diff line number Diff line change @@ -1167,4 +1167,37 @@ describe('api: watch', () => {
1167
1167
set ( r . value , 'foo' , 1 )
1168
1168
expect ( spy ) . not . toHaveBeenCalled ( )
1169
1169
} )
1170
+
1171
+ // #12664
1172
+ it ( 'queueing multiple flush: post watchers' , async ( ) => {
1173
+ const parentSpy = vi . fn ( )
1174
+ const childSpy = vi . fn ( )
1175
+
1176
+ const Child = {
1177
+ setup ( ) {
1178
+ const el = ref ( )
1179
+ watch ( el , childSpy , { flush : 'post' } )
1180
+ return { el }
1181
+ } ,
1182
+ template : `<div><span ref="el">hello child</span></div>`
1183
+ }
1184
+ const App = {
1185
+ components : { Child } ,
1186
+ setup ( ) {
1187
+ const el = ref ( )
1188
+ watch ( el , parentSpy , { flush : 'post' } )
1189
+ return { el }
1190
+ } ,
1191
+ template : `<div><Child /><span ref="el">hello app1</span></div>`
1192
+ }
1193
+
1194
+ const container = document . createElement ( 'div' )
1195
+ const root = document . createElement ( 'div' )
1196
+ container . appendChild ( root )
1197
+ new Vue ( App ) . $mount ( root )
1198
+
1199
+ await nextTick ( )
1200
+ expect ( parentSpy ) . toHaveBeenCalledTimes ( 1 )
1201
+ expect ( childSpy ) . toHaveBeenCalledTimes ( 1 )
1202
+ } )
1170
1203
} )
You can’t perform that action at this time.
0 commit comments