Skip to content

Commit 684248e

Browse files
alisd23timdorr
authored andcommitted
feat: cancel pending enter/change hooks on location change (#4063)
* feat: cancel pending enter/change hooks on location change * chore: fix npm build scripts on windows
1 parent cd3479b commit 684248e

File tree

3 files changed

+108
-10
lines changed

3 files changed

+108
-10
lines changed

modules/TransitionUtils.js

+38-8
Original file line numberDiff line numberDiff line change
@@ -1,31 +1,47 @@
11
import { loopAsync } from './AsyncUtils'
22

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) => {
518
hook.apply(route, args)
619

7-
if (hook.length < asyncArity) {
20+
if (isSync) {
821
let callback = args[args.length - 1]
922
// Assume hook executes synchronously and
1023
// automatically call the callback.
1124
callback()
1225
}
1326
}
27+
28+
pendingHooks.add(transitionHook)
29+
30+
return transitionHook
1431
}
1532

1633
function getEnterHooks(routes) {
1734
return routes.reduce(function (hooks, route) {
1835
if (route.onEnter)
19-
hooks.push(createTransitionHook(route.onEnter, route, 3))
20-
36+
hooks.push(createTransitionHook(route.onEnter, route, 3, enterHooks))
2137
return hooks
2238
}, [])
2339
}
2440

2541
function getChangeHooks(routes) {
2642
return routes.reduce(function (hooks, route) {
2743
if (route.onChange)
28-
hooks.push(createTransitionHook(route.onChange, route, 4))
44+
hooks.push(createTransitionHook(route.onChange, route, 4, changeHooks))
2945
return hooks
3046
}, [])
3147
}
@@ -63,9 +79,16 @@ function runTransitionHooks(length, iter, callback) {
6379
* which could lead to a non-responsive UI if the hook is slow.
6480
*/
6581
export function runEnterHooks(routes, nextState, callback) {
82+
enterHooks.clear()
6683
const hooks = getEnterHooks(routes)
6784
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)
6992
}, callback)
7093
}
7194

@@ -80,9 +103,16 @@ export function runEnterHooks(routes, nextState, callback) {
80103
* which could lead to a non-responsive UI if the hook is slow.
81104
*/
82105
export function runChangeHooks(routes, state, nextState, callback) {
106+
changeHooks.clear()
83107
const hooks = getChangeHooks(routes)
84108
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)
86116
}, callback)
87117
}
88118

modules/__tests__/transitionHooks-test.js

+68
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import createHistory from '../createMemoryHistory'
55
import { routerShape } from '../PropTypes'
66
import execSteps from './execSteps'
77
import Router from '../Router'
8+
import Route from '../Route'
89

910
describe('When a router enters a branch', function () {
1011
let
@@ -374,5 +375,72 @@ describe('When a router enters a branch', function () {
374375
})
375376
})
376377
})
378+
})
379+
380+
describe('Changing location', () => {
381+
let node, onEnterSpy, onChangeSpy
382+
383+
const Text = text => () => <p>{text}</p>
384+
const noop = () => {}
377385

386+
const onEnter = (state, replace, cb) => {
387+
setTimeout(() => {
388+
onEnterSpy()
389+
replace('/bar')
390+
cb()
391+
})
392+
}
393+
const onChange = (prevState, nextState, replace, cb) => {
394+
setTimeout(() => {
395+
onChangeSpy()
396+
replace('/bar')
397+
cb()
398+
})
399+
}
400+
const createRoutes = ({ enter, change }) => [
401+
<Route path="/" onChange={change ? onChange : noop} component={Text('Home')}>
402+
<Route path="child1" component={Text('Child1')} />
403+
<Route path="child2" component={Text('Child2')} />
404+
</Route>,
405+
<Route path="/foo" onEnter={enter ? onEnter : noop} component={Text('Foo')} />,
406+
<Route path="/bar" component={Text('Bar')} />
407+
]
408+
409+
beforeEach(() => {
410+
node = document.createElement('div')
411+
onEnterSpy = expect.createSpy()
412+
onChangeSpy = expect.createSpy()
413+
})
414+
415+
it('cancels pending async onEnter hook', (done) => {
416+
const history = createHistory('/')
417+
const routes = createRoutes({ enter: true })
418+
419+
render(<Router history={history} routes={routes} />, node, () => {
420+
history.push('/foo')
421+
history.push('/')
422+
expect(onEnterSpy.calls.length).toEqual(0)
423+
setTimeout(() => {
424+
expect(onEnterSpy.calls.length).toEqual(1)
425+
expect(node.innerHTML).toContain('Home')
426+
done()
427+
})
428+
})
429+
})
430+
431+
it('cancels pending async onChange hook', (done) => {
432+
const history = createHistory('/')
433+
const routes = createRoutes({ change: true })
434+
435+
render(<Router history={history} routes={routes} />, node, () => {
436+
history.push('/child1')
437+
history.push('/bar')
438+
expect(onChangeSpy.calls.length).toEqual(0)
439+
setTimeout(() => {
440+
expect(onChangeSpy.calls.length).toEqual(1)
441+
expect(node.innerHTML).toContain('Bar')
442+
done()
443+
})
444+
})
445+
})
378446
})

package.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -16,8 +16,8 @@
1616
"bugs": "https://github.com/ReactTraining/react-router/issues",
1717
"scripts": {
1818
"build": "npm run build-cjs && npm run build-es",
19-
"build-cjs": "rimraf lib && cross-env BABEL_ENV=cjs babel ./modules -d lib --ignore '__tests__'",
20-
"build-es": "rimraf es && cross-env BABEL_ENV=es babel ./modules -d es --ignore '__tests__'",
19+
"build-cjs": "rimraf lib && cross-env BABEL_ENV=cjs babel ./modules -d lib --ignore __tests__",
20+
"build-es": "rimraf es && cross-env BABEL_ENV=es babel ./modules -d es --ignore __tests__",
2121
"build-umd": "cross-env NODE_ENV=development webpack modules/index.js umd/ReactRouter.js",
2222
"build-min": "cross-env NODE_ENV=production webpack -p modules/index.js umd/ReactRouter.min.js",
2323
"lint": "eslint examples modules scripts tools *.js",

0 commit comments

Comments
 (0)