1
1
import { loopAsync } from './AsyncUtils'
2
2
3
- function createTransitionHook ( hook , route , asyncArity ) {
4
- return function ( ...args ) {
3
+ class PendingHooks {
4
+ hooks = [ ]
5
+ add = hook => this . hooks . push ( hook )
6
+ remove = hook => this . hooks = this . hooks . filter ( h => h !== hook )
7
+ has = hook => this . hooks . indexOf ( hook ) !== - 1
8
+ clear = ( ) => this . hooks = [ ]
9
+ }
10
+
11
+ const enterHooks = new PendingHooks ( )
12
+ const changeHooks = new PendingHooks ( )
13
+
14
+ function createTransitionHook ( hook , route , asyncArity , pendingHooks ) {
15
+ const isSync = hook . length < asyncArity
16
+
17
+ const transitionHook = ( ...args ) => {
5
18
hook . apply ( route , args )
6
19
7
- if ( hook . length < asyncArity ) {
20
+ if ( isSync ) {
8
21
let callback = args [ args . length - 1 ]
9
22
// Assume hook executes synchronously and
10
23
// automatically call the callback.
11
24
callback ( )
12
25
}
13
26
}
27
+
28
+ pendingHooks . add ( transitionHook )
29
+
30
+ return transitionHook
14
31
}
15
32
16
33
function getEnterHooks ( routes ) {
17
34
return routes . reduce ( function ( hooks , route ) {
18
35
if ( route . onEnter )
19
- hooks . push ( createTransitionHook ( route . onEnter , route , 3 ) )
20
-
36
+ hooks . push ( createTransitionHook ( route . onEnter , route , 3 , enterHooks ) )
21
37
return hooks
22
38
} , [ ] )
23
39
}
24
40
25
41
function getChangeHooks ( routes ) {
26
42
return routes . reduce ( function ( hooks , route ) {
27
43
if ( route . onChange )
28
- hooks . push ( createTransitionHook ( route . onChange , route , 4 ) )
44
+ hooks . push ( createTransitionHook ( route . onChange , route , 4 , changeHooks ) )
29
45
return hooks
30
46
} , [ ] )
31
47
}
@@ -63,9 +79,16 @@ function runTransitionHooks(length, iter, callback) {
63
79
* which could lead to a non-responsive UI if the hook is slow.
64
80
*/
65
81
export function runEnterHooks ( routes , nextState , callback ) {
82
+ enterHooks . clear ( )
66
83
const hooks = getEnterHooks ( routes )
67
84
return runTransitionHooks ( hooks . length , ( index , replace , next ) => {
68
- hooks [ index ] ( nextState , replace , next )
85
+ const wrappedNext = ( ) => {
86
+ if ( enterHooks . has ( hooks [ index ] ) ) {
87
+ next ( )
88
+ enterHooks . remove ( hooks [ index ] )
89
+ }
90
+ }
91
+ hooks [ index ] ( nextState , replace , wrappedNext )
69
92
} , callback )
70
93
}
71
94
@@ -80,9 +103,16 @@ export function runEnterHooks(routes, nextState, callback) {
80
103
* which could lead to a non-responsive UI if the hook is slow.
81
104
*/
82
105
export function runChangeHooks ( routes , state , nextState , callback ) {
106
+ changeHooks . clear ( )
83
107
const hooks = getChangeHooks ( routes )
84
108
return runTransitionHooks ( hooks . length , ( index , replace , next ) => {
85
- hooks [ index ] ( state , nextState , replace , next )
109
+ const wrappedNext = ( ) => {
110
+ if ( changeHooks . has ( hooks [ index ] ) ) {
111
+ next ( )
112
+ changeHooks . remove ( hooks [ index ] )
113
+ }
114
+ }
115
+ hooks [ index ] ( state , nextState , replace , wrappedNext )
86
116
} , callback )
87
117
}
88
118
0 commit comments