diff --git a/modules/TransitionUtils.js b/modules/TransitionUtils.js index baff7cde63..807e316544 100644 --- a/modules/TransitionUtils.js +++ b/modules/TransitionUtils.js @@ -1,23 +1,39 @@ import { loopAsync } from './AsyncUtils' -function createTransitionHook(hook, route, asyncArity) { - return function (...args) { +class PendingHooks { + hooks = [] + add = hook => this.hooks.push(hook) + remove = hook => this.hooks = this.hooks.filter(h => h !== hook) + has = hook => this.hooks.indexOf(hook) !== -1 + clear = () => this.hooks = [] +} + +const enterHooks = new PendingHooks() +const changeHooks = new PendingHooks() + +function createTransitionHook(hook, route, asyncArity, pendingHooks) { + const isSync = hook.length < asyncArity + + const transitionHook = (...args) => { hook.apply(route, args) - if (hook.length < asyncArity) { + if (isSync) { let callback = args[args.length - 1] // Assume hook executes synchronously and // automatically call the callback. callback() } } + + pendingHooks.add(transitionHook) + + return transitionHook } function getEnterHooks(routes) { return routes.reduce(function (hooks, route) { if (route.onEnter) - hooks.push(createTransitionHook(route.onEnter, route, 3)) - + hooks.push(createTransitionHook(route.onEnter, route, 3, enterHooks)) return hooks }, []) } @@ -25,7 +41,7 @@ function getEnterHooks(routes) { function getChangeHooks(routes) { return routes.reduce(function (hooks, route) { if (route.onChange) - hooks.push(createTransitionHook(route.onChange, route, 4)) + hooks.push(createTransitionHook(route.onChange, route, 4, changeHooks)) return hooks }, []) } @@ -63,9 +79,16 @@ function runTransitionHooks(length, iter, callback) { * which could lead to a non-responsive UI if the hook is slow. */ export function runEnterHooks(routes, nextState, callback) { + enterHooks.clear() const hooks = getEnterHooks(routes) return runTransitionHooks(hooks.length, (index, replace, next) => { - hooks[index](nextState, replace, next) + const wrappedNext = () => { + if (enterHooks.has(hooks[index])) { + next() + enterHooks.remove(hooks[index]) + } + } + hooks[index](nextState, replace, wrappedNext) }, callback) } @@ -80,9 +103,16 @@ export function runEnterHooks(routes, nextState, callback) { * which could lead to a non-responsive UI if the hook is slow. */ export function runChangeHooks(routes, state, nextState, callback) { + changeHooks.clear() const hooks = getChangeHooks(routes) return runTransitionHooks(hooks.length, (index, replace, next) => { - hooks[index](state, nextState, replace, next) + const wrappedNext = () => { + if (changeHooks.has(hooks[index])) { + next() + changeHooks.remove(hooks[index]) + } + } + hooks[index](state, nextState, replace, wrappedNext) }, callback) } diff --git a/modules/__tests__/transitionHooks-test.js b/modules/__tests__/transitionHooks-test.js index 586428a5f1..e77575d9b5 100644 --- a/modules/__tests__/transitionHooks-test.js +++ b/modules/__tests__/transitionHooks-test.js @@ -5,6 +5,7 @@ import createHistory from '../createMemoryHistory' import { routerShape } from '../PropTypes' import execSteps from './execSteps' import Router from '../Router' +import Route from '../Route' describe('When a router enters a branch', function () { let @@ -374,5 +375,72 @@ describe('When a router enters a branch', function () { }) }) }) +}) + +describe('Changing location', () => { + let node, onEnterSpy, onChangeSpy + + const Text = text => () =>
{text}
+ const noop = () => {} + const onEnter = (state, replace, cb) => { + setTimeout(() => { + onEnterSpy() + replace('/bar') + cb() + }) + } + const onChange = (prevState, nextState, replace, cb) => { + setTimeout(() => { + onChangeSpy() + replace('/bar') + cb() + }) + } + const createRoutes = ({ enter, change }) => [ +