React Router v4 is a complete rewrite, so there is not a simple migration path. This guide will provide you with a number of steps to help you understand how to upgrade your application.
Note: This migration guide is for both React Router v2 and v3, but for brevity, references to previous versions will only mention v3.
Caution for react-router-redux users: Not all packages are compatible with React Router v4 or are feature complete. In particular, time travel in Redux DevTools is not yet available in the latest react-router-redux (currently in alpha).
In React Router v3, there was a single <Router>
component. It would be provided a history
object as a prop.
Also, you would provide it your application's route configuration to the <Router>
either using the routes
prop or as the children
of the <Router>
.
// v3
import routes from './routes'
<Router history={browserHistory} routes={routes} />
// or
<Router history={browserHistory}>
<Route path='/' component={App}>
// ...
</Route>
</Router>
With React Router v4, one of the big changes is that there are a number of different router components. Each one will create a history
object for you. The <BrowserRouter>
creates a browser history, the <HashRouter>
creates a hash history, and the <MemoryRouter>
creates a memory history.
In v4, there is no centralized route configuration. Anywhere that you need to render content based on a route, you will just render a <Route>
component.
//v4
<BrowserRouter>
<div>
<Route path='/about' component={About} />
<Route path='/contact' component={Contact} />
</div>
</BrowserRouter>
One thing to note is that the router component must only be given one child element.
// yes
<BrowserRouter>
<div>
<Route path='/about' component={About} />
<Route path='/contact' component={Contact} />
</div>
</BrowserRouter>
// no
<BrowserRouter>
<Route path='/about' component={About} />
<Route path='/contact' component={Contact} />
</BrowserRouter>
In v3, the <Route>
was not really a component. Instead, all of your application's <Route>
elements were just used to created a route configuration object.
/// in v3 the element
<Route path='contact' component={Contact} />
// was equivalent to
{
path: 'contact',
component: Contact
}
With v4, you layout your app's components just like a regular React application. Anywhere that you want to render content based on the location (specifically, its pathname
), you render a <Route>
.
The v4 <Route>
component is actually a component, so wherever you render a <Route>
component, content will be rendered. When the <Route>
's path
matches the current location, it will use its rendering prop (component
, render
, or children
) to render. When the <Route>
's path
does not match, it will render null
.
In v3, <Route>
s were nested by passing them as the children
of their parent <Route>
.
<Route path='parent' component={Parent}>
<Route path='child' component={Child} />
<Route path='other' component={Other} />
</Route>
When a nested <Route>
matched, React elements would be created using both the child and parent <Route>
's component
prop. The child element would be passed to the parent element as its children
prop.
<Parent {...routeProps}>
<Child {...routeProps} />
</Parent>
With v4, children <Route>
s should just be rendered by the parent <Route>
's component.
<Route path='parent' component={Parent} />
const Parent = () => (
<div>
<Route path='child' component={Child} />
<Route path='other' component={Other} />
</div>
)
React Router v3 provides onEnter
, onUpdate
, and onLeave
methods. These were essentially recreating React's lifecycle methods.
With v4, you should use the lifecycle methods of the component rendered by a <Route>
. Instead of onEnter
, you would use componentDidMount
or componentWillMount
. Where you would use onUpdate
, you can use componentDidUpdate
or componentWillUpdate
(or possibly componentWillReceiveProps
). onLeave
can be replaced with componentWillUnmount
.
In v3, parameters were made optional with parentheses: path="/entity/:entityId(/:parentId)"
In v4, the syntax is changed to a trailing question mark: path="/entity/:entityId/:parentId?"
In v4, there is no parsing done on query strings. The unparsed query string is available on the location.search
property.
The qhistory library can provide this functionality if it is necessary for your application.
Read more regarding intentions for this change and possible solutions in this issue.
In v3, you could specify a number of child routes, and only the first one that matched would be rendered.
// v3
<Route path='/' component={App}>
<IndexRoute component={Home} />
<Route path='about' component={About} />
<Route path='contact' component={Contact} />
</Route>
v4 provides a similar functionality with the <Switch>
component. When a <Switch>
is rendered, it will only render the first child <Route>
that matches the current location.
// v4
const App = () => (
<Switch>
<Route exact path='/' component={Home} />
<Route path='/about' component={About} />
<Route path='/contact' component={Contact} />
</Switch>
)
In v3, if you wanted to redirect from one path to another, for instance / to /welcome, you would use <IndexRedirect >
.
// v3
<Route path="/" component={App}>
<IndexRedirect to="/welcome" />
</Route>
In v4, you can achieve the same functionality using <Redirect>
.
// v4
<Route exact path="/" render={() => <Redirect to="/welcome" component={App} />} />
<Switch>
<Route exact path="/" component={App} />
<Route path="/login" component={Login} />
<Redirect path="*" to="/" />
</Switch>
In v3, <Redirect>
preserved the query string:
// v3
<Redirect from="/" to="/welcome" />
// /?source=google → /welcome?source=google
In v4, you must re-pass these properties to the to
prop:
// v4
<Redirect from="/" to="/welcome" />
// /?source=google → /welcome
<Redirect from="/" to={{ ...location, pathname: "/welcome" }} />
// /?source=google → /welcome?source=google
In v3, you could use the same matching code used internally to check if a path matched a pattern. In v4 this has been replaced by matchPath which is powered by the path-to-regexp@^1.7.0
library.
In v3, you could use PatternUtils.formatPattern to generate a valid path from a path pattern (perhaps in a constant or in your central routing config) and an object containing the names parameters:
// v3
const THING_PATH = '/thing/:id';
<Link to={PatternUtils.formatPattern(THING_PATH, {id: 1})}>A thing</Link>
In v4, you can achieve the same functionality using the compile
function in path-to-regexp@^1.7.0
.
// v4
const THING_PATH = '/thing/:id';
const thingPath = pathToRegexp.compile(THING_PATH);
<Link to={thingPath({id: 1})}>A thing</Link>
The getParamNames
functionality can be achieved using the parse
function in path-to-regexp@^1.7.0
.
In v3, you could omit to
property or set it to null to create an anchor tag without href
attribute.
// v3
<Link to={disabled ? null : `/item/${id}`} className="item">
// item content
</Link>
In v4, you should always provide to
. In case you are rely on empty to
you can make a simple wrapper.
// v4
import { Link } from 'react-router-dom'
const LinkWrapper = (props) => {
const Component = props.to ? Link : 'a'
return (
<Component {...props}>
{ props.children }
</Component>
)
}
<LinkWrapper to={disabled ? null : `/item/${id}`} className="item">
// item content
</LinkWrapper>