Package Exports
- connected-react-router
- connected-react-router/immutable
- connected-react-router/lib/actions
This package does not declare an exports field, so the exports above have been automatically detected and optimized by JSPM instead. If any package subpath is missing, it is recommended to post an issue to the original package (connected-react-router) to support the "exports" field. If that is not possible, create a JSPM override to customize the exports field for this package.
Readme
Connected React Router 
A Redux binding for React Router v4
Main features
✨ Synchronize router state with redux store with uni-directional flow (history -> store -> router -> components).
🎁 Support React Router v4.
☀️ Support functional component hot reloading while preserving state (with react-hot-reload v3).
🎉 Dispatching history methods (push, replace, go, goBack, goForward) work for both redux-thunk and redux-saga.
⛄ Nested children can access routing state such as current location directly with react-redux's connect.
🕘 Support time traveling in Redux DevTools.
💎 Support Immutable.js
Note:
connected-react-router@4.0.0-betasupports newreact-router@4.0.0-beta(with<Route>and<Switch>)- For old
react-router@4.0.0-alpha(with<Match>and<Miss>), you needconnected-react-router@2.0.0-alpha.5
Installation
Using npm:
$ npm install --save connected-react-routerOr yarn:
$ yarn add connected-react-routerUsage
Step 1
- Create a
historyobject. - Wrap the root reducer with
connectRouterand supply thehistoryobject to get a new root reducer. - Use
routerMiddleware(history)if you want to dispatch history actions (ex. to change URL withpush('/path/to/somewhere')).
...
import { createBrowserHistory } from 'history'
import { applyMiddleware, compose, createStore } from 'redux'
import { connectRouter, routerMiddleware } from 'connected-react-router'
...
const history = createBrowserHistory()
const store = createStore(
connectRouter(history)(rootReducer), // new root reducer with router state
initialState,
compose(
applyMiddleware(
routerMiddleware(history), // for dispatching history actions
// ... other middlewares ...
),
),
)Step 2
- Wrap your react-router v4 routing with
ConnectedRouterand passhistoryobject as a prop. - Place
ConnectedRouteras children ofreact-redux'sProvider.
...
import { Provider } from 'react-redux'
import { Route, Switch } from 'react-router' // react-router v4
import { ConnectedRouter } from 'connected-react-router'
...
ReactDOM.render(
<Provider store={store}>
<ConnectedRouter history={history}> { /* place ConnectedRouter under Provider */ }
<div> { /* your usual react-router v4 routing */ }
<Switch>
<Route exact path="/" render={() => (<div>Match</div>)} />
<Route render={() => (<div>Miss</div>)} />
</Switch>
</div>
</ConnectedRouter>
</Provider>,
document.getElementById('react-root')
)Now, it's ready to work!
Examples
See examples folder
FAQ
- How to navigate with Redux action
- How to get current URL path
- How to hot reload functional components
- How to hot reload reducers
- How to support Immutable.js
How to navigate with Redux action
with store.dispatch
import { push } from 'connected-react-router'
store.dispatch(push('/path/to/somewhere'))in redux thunk
import { push } from 'connected-react-router'
export const login = (username, password) => (dispatch) => {
/* do something before redirection */
dispatch(push('/home'))
}
in redux saga
import { push } from 'connected-react-router'
import { put, call } from 'redux-saga/effects'
export function* login(username, password) {
/* do something before redirection */
yield put(push('/home'))
}How to get current URL path
The current URL path can be accessed directry from the router state with react-redux's connect.
import { connect } from 'react-redux'
const Child = ({ path }) => (
<div>
Child receives path {path}.
</div>
)
const mapStateToProps = state => ({
path: state.router.location.pathname,
})
export default connect(mapStateToProps)(Child)How to hot reload functional components
- Separate main app component to another file.
App.js
import React from 'react'
import { Route, Switch } from 'react-router' /* react-router v4 */
import { ConnectedRouter } from 'connected-react-router'
const App = ({ history }) => ( /* receive history object via props */
<ConnectedRouter history={history}>
<div>
<Switch>
<Route exact path="/" render={() => (<div>Match</div>)} />
<Route render={() => (<div>Miss</div>)} />
</Switch>
</div>
</ConnectedRouter>
)
export default App- Wrap the
Appcomponent withAppContainerfromreact-hot-loaderv3 as a top-level container.
index.js
import React from 'react'
import ReactDOM from 'react-dom'
import { Provider } from 'react-redux'
import { AppContainer } from 'react-hot-loader' /* react-hot-loader v3 */
import App from './App'
...
const render = () => { // this function will be reused
ReactDOM.render(
<AppContainer> { /* AppContainer for hot reloading v3 */ }
<Provider store={store}>
<AppComponent history={history} /> { /* pass history object as props */ }
</Provider>
</AppContainer>,
document.getElementById('react-root')
)
}
render()- Detect change and re-render with hot reload.
index.js
...
if (module.hot) {
module.hot.accept('./App', () => {
/* For Webpack 2.x
Need to disable babel ES2015 modules transformation in .babelrc
presets: [
["es2015", { "modules": false }]
]
*/
render()
/* For Webpack 1.x
const NextApp = require('./App').default
renderWithHotReload(NextApp)
*/
})
}Now, when you change any component that App depends on, it will trigger hot reloading without losing redux state. Thanks react-hot-loader v3!
How to hot reload reducers
Detect change and replace with a new root reducer with router state
index.js
...
if (module.hot) {
module.hot.accept('./reducers', () => {
/* For Webpack 2.x
Need to disable babel ES2015 modules transformation in .babelrc
presets: [
["es2015", { "modules": false }]
]
*/
store.replaceReducer(connectRouter(history)(rootReducer))
/* For Webpack 1.x
const nextRootReducer = require('./reducers').default
store.replaceReducer(connectRouter(history)(nextRootReducer))
*/
})
}How to support Immutable.js
- Use
combineReducersfromredux-immutableto create the root reducer.
import { combineReducers } from 'redux-immutable'
...
const rootReducer = combineReducers({
...
})
...- Import
ConnectedRouter,routerMiddleware, andconnectRouterfromconnected-react-router/immutableinstead ofconnected-react-router.
import { ConnectedRouter, routerMiddleware, connectRouter } from 'connected-react-router/immutable'- (Optional) Initialize state with
Immutabel.Map()
import Immutable from 'immutable'
...
const initialState = Immutable.Map()
...
const store = createStore(
connectRouter(history)(rootReducer),
initialState,
...
)Build
npm run buildGenerated files will be in lib folder.
Cautions
This is still an experimental project. It relies on several alpha and beta things (i.e. react-hot-loader v3 and react-router v4). Anything can be changed. Bugs are certainly waiting for you to wake them up. Please use it at your own risk.
Contributors
See Contributors and Acknowledge.