Package Exports
- react-redux
- react-redux/lib/components/createAll
- react-redux/lib/utils/isPlainObject
- react-redux/lib/utils/shallowEqual
- react-redux/lib/utils/wrapActionCreators
- react-redux/native
- react-redux/package.json
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 (react-redux) to support the "exports" field. If that is not possible, create a JSPM override to customize the exports field for this package.
Readme
React Redux
Official React bindings for Redux.
Performant and flexible.
Note: There is a project called
redux-react
on NPM that is completely unrelated to the official bindings. This documentation (and any other official Redux documentation) is forreact-redux
.
Table of Contents
Installation
npm install --save react-redux
Be careful! “React” goes first.
There is a project called redux-react
, but it is completely unrelated.
React Redux requires React 0.13 or later.
React Native
What you get from react-redux
is for React.
For React Native, import from react-redux/native
instead.
Quick Start
React bindings for Redux embrace the idea of dividing “smart” and “dumb” components.
It is advisable that only top-level components of your app (such as route handlers, for example) are aware of Redux. Components below them should be “dumb” and receive all data via props.
Location | Use React-Redux | To read data, they | To change data, they | |
---|---|---|---|---|
“Smart” Components | Top level, route handlers | Yes | Subscribe to Redux state | Dispatch Redux actions |
“Dumb” Components | Middle and leaf components | No | Read data from props | Invoke callbacks from props |
“Dumb” components are unaware of Redux
Let’s say we have a <Counter />
“dumb” component with a number value
prop, and an onIncrement
function prop that it will call when user presses an “Increment” button:
import { Component } from 'react';
export default class Counter extends Component {
render() {
return (
<button onClick={this.props.onIncrement}>
{this.props.value}
</button>
);
}
}
“Smart” components are connect()
-ed to Redux
Here’s how we hook it up to the Redux Store.
We will use the connect()
function provided by react-redux
to turn a “dumb” Counter
into a smart component. The connect()
function lets you specify which exact state from the Redux store your component wants to track. This lets you subscribe on any level of granularity.
containers/CounterContainer.js
import { Component } from 'react';
import { connect } from 'react-redux';
import Counter from '../components/Counter';
import { increment } from '../actionsCreators';
// Which part of the Redux global state does our component want to receive as props?
function mapStateToProps(state) {
return {
value: state.counter
};
}
// Which action creators does it want to receive by props?
function mapDispatchToProps(dispatch) {
return {
onIncrement: () => dispatch(increment())
};
}
export default connect(
mapStateToProps,
mapDispatchToProps
)(Counter);
// You can also pass an object instead of defining `mapDispatchToProps`:
// export default connect(mapStateToProps, CounterActionCreators)(Counter);
// Or you can pass `dispatch` down as a prop if you omit `mapDispatchToProps`:
// export default connect(mapStateToProps)(Counter);
// See more recipes in detailed connect() examples below.
Whether to put the connect()
call in the same file as the “dumb” component, or separately, is up to you.
Ask yourself whether you'd want to reuse this component but bind it to different data, or not.
Nesting
You can have many connect()
-ed components in your app at any depth, and you can even nest them. It is, however, preferable that you try to only connect()
top-level components such as route handlers, so the data flow in your application stays predictable.
Support for Decorators
You might have noticed that we used parens twice when calling connect()
. This is called partial application, and it lets people
use ES7’s proposed decorator syntax:
// Unstable syntax! It might change or break in production.
@connect(mapStateToProps)
export default class CounterContainer { ... }
Don’t forget decorators are experimental! They desugar to function calls anyway as the example above demonstrates.
Additional Flexibility
This is the most basic usage, but connect()
supports many other patterns: just passing the vanilla dispatch()
function down, binding multiple action creators, passing them in an actions
prop, selecting parts of state and binding action creators depending on props
, and so on. Check out the connect()
docs below to learn more.
Injecting Redux Store
Finally, how do we actually hook it up to the Redux store? We need to create the store somewhere at the root of our component hierarchy. For client apps, the root component is a good place. For server rendering, you can do this in the request handler.
The trick is to wrap the whole view hierarchy into a <Provider>{() => ... }</Provider>
where Provider
is imported from react-redux
. One gotcha is that the child of Provider
must be a function. This is to work around an issue about how context (undocumented feature we have to rely on to pass Redux data to components below) works in React 0.13. In React 0.14, you will be able to put your view hierarchy in <Provider>
without wrapping it into a function.
import { Component } from 'react';
import { Provider } from 'react-redux';
class App extends Component {
render() {
// ...
}
}
const targetEl = document.getElementById('root');
React.render((
<Provider store={store}>
{() => <App />}
</Provider>
), targetEl);
API
<Provider store>
Makes the Redux store available to the connect()
calls in the component hierarchy below. Normally, you can’t use connect()
without wrapping the root component in <Provider>
. (If you really need to, you can manually pass store
as a prop to every connect()
ed component, but we only recommend to do this for stubbing store
in unit tests, or in non-fully-React codebases. Normally, you should just use <Provider>
.)
Props
store
: (Redux Store): The single Redux store in your application.children
: (Function): Unlike most React components,<Provider>
accepts a function as a child with your root component. This is a temporary workaround for a React 0.13 context issue, which will be fixed when React 0.14 comes out.
Example
Vanilla React
React.render(
<Provider store={store}>
{() => <MyRootComponent />}
</Provider>,
rootEl
);
React Router 0.13
Router.run(routes, Router.HistoryLocation, (Handler, routerState) => { // note "routerState" here
React.render(
<Provider store={store}>
{() => <Handler routerState={routerState} />} // note "routerState" here: important to pass it down
</Provider>,
document.getElementById('root')
);
});
React Router 1.0
React.render(
<Provider store={store}>
{() => <Router history={history}>...</Router>}
</Provider>,
targetEl
);
connect([mapStateToProps], [mapDispatchToProps], [mergeProps])
Connects a React component to a Redux store.
Arguments
[
mapStateToProps(state, [ownProps]): stateProps
] (Function): If specified, the component will subscribe to Redux store updates. Any time it updates,mapStateToProps
will be called. Its result must be a plain object, and it will be merged into the component’s props. If you omit it, the component will not be subscribed to the Redux store. IfownProps
is specified as a second argument, thenmapStateToProps
will be re-invoked whenever the component receives new props.[
mapDispatchToProps(dispatch, [ownProps]): dispatchProps
] (Object or Function): If an object is passed, each function inside it will be assumed to be a Redux action creator. An object with the same function names, but bound to a Redux store, will be merged into the component’s props. If a function is passed, it will be givendispatch
. It’s up to you to return an object that somehow usesdispatch
to bind action creators in your own way. (Tip: you may use thebindActionCreators()
helper from Redux.) If you omit it, the default implementation just injectsdispatch
into your component’s props. IfownProps
is specified as a second argument, thenmapDispatchToProps
will be re-invoked whenever the component receives new props.[
mergeProps(stateProps, dispatchProps, ownProps): props
] (Function): If specified, it is passed the result ofmapStateToProps()
,mapDispatchToProps()
, and the parentprops
. The plain object you return from it will be passed as props to the wrapped component. You may specify this function to select a slice of the state based on props, or to bind action creators to a particular variable from props. If you omit it,Object.assign({}, ownProps, stateProps, dispatchProps)
is used by default.
Returns
A React component class that injects state and action creators into your component according to the specified options.
Remarks
It needs to be invoked two times. The first time with its arguments described above, and a second time, with the component:
connect(mapStateToProps, mapDispatchToProps, mergeProps)(MyComponent)
.The
mapStateToProps
function takes a single argument of the entire Redux store’s state and returns an object to be passed as props. It is often called a selector. Use reselect to efficiently compose selectors and compute derived data.To use
connect()
, the root component of your app must be wrapped into<Provider>{() => ... }</Provider>
before being rendered. You may also passstore
as a prop to theconnect()
ed component, but it's not recommended, because it's just too much trouble. Only do this for non-fully-React codebases or to stub the store in a unit test.
Examples
Inject just dispatch
and don't listen to store
export default connect()(TodoApp);
Inject dispatch
and every field in the global state
Don’t do this! It kills any performance optimisations because
TodoApp
will rerender after every action.
It’s better to have more granularconnect()
on several components in your view hierarchy.
export default connect(state => state)(TodoApp);
Inject dispatch
and todos
function mapStateToProps(state) {
return { todos: state.todos };
}
export default connect(mapStateToProps)(TodoApp);
Inject todos
and all action creators (addTodo
, completeTodo
, ...)
import * as actionCreators from './actionCreators';
function mapStateToProps(state) {
return { todos: state.todos };
}
export default connect(mapStateToProps, actionCreators)(TodoApp);
Inject todos
and all action creators (addTodo
, completeTodo
, ...) as actions
import * as actionCreators from './actionCreators';
import { bindActionCreators } from 'redux';
function mapStateToProps(state) {
return { todos: state.todos };
}
function mapDispatchToProps(dispatch) {
return { actions: bindActionCreators(actionCreators, dispatch) };
}
export default connect(mapStateToProps, mapDispatchToProps)(TodoApp);
Inject todos
and a specific action creator (addTodo
)
import { addTodo } from './actionCreators';
import { bindActionCreators } from 'redux';
function mapStateToProps(state) {
return { todos: state.todos };
}
function mapDispatchToProps(dispatch) {
return bindActionCreators({ addTodo }, dispatch);
}
export default connect(mapStateToProps, mapDispatchToProps)(TodoApp);
Inject todos
, todoActionCreators as todoActions
, and counterActionCreators as counterActions
import * as todoActionCreators from './todoActionCreators';
import * as counterActionCreators from './counterActionCreators';
import { bindActionCreators } from 'redux';
function mapStateToProps(state) {
return { todos: state.todos };
}
function mapDispatchToProps(dispatch) {
return {
todoActions: bindActionCreators(todoActionCreators, dispatch),
counterActions: bindActionCreators(counterActionCreators, dispatch)
};
}
export default connect(mapStateToProps, mapDispatchToProps)(TodoApp);
Inject todos
, and todoActionCreators and counterActionCreators together as actions
import * as todoActionCreators from './todoActionCreators';
import * as counterActionCreators from './counterActionCreators';
import { bindActionCreators } from 'redux';
function mapStateToProps(state) {
return { todos: state.todos };
}
function mapDispatchToProps(dispatch) {
return {
actions: bindActionCreators(Object.assign({}, todoActionCreators, counterActionCreators), dispatch)
};
}
export default connect(mapStateToProps, mapDispatchToProps)(TodoApp);
Inject todos
, and all todoActionCreators and counterActionCreators directly as props
import * as todoActionCreators from './todoActionCreators';
import * as counterActionCreators from './counterActionCreators';
import { bindActionCreators } from 'redux';
function mapStateToProps(state) {
return { todos: state.todos };
}
function mapDispatchToProps(dispatch) {
return bindActionCreators(Object.assign({}, todoActionCreators, counterActionCreators), dispatch);
}
export default connect(mapStateToProps, mapDispatchToProps)(TodoApp);
Inject todos
of a specific user depending on props
import * as actionCreators from './actionCreators';
function mapStateToProps(state, ownProps) {
return { todos: state.todos[ownProps.userId] };
}
export default connect(mapStateToProps)(TodoApp);
Inject todos
of a specific user depending on props, and inject props.userId
into the action
import * as actionCreators from './actionCreators';
function mapStateToProps(state) {
return { todos: state.todos };
}
function mergeProps(stateProps, dispatchProps, ownProps) {
return Object.assign({}, ownProps, {
todos: stateProps.todos[ownProps.userId],
addTodo: (text) => dispatchProps.addTodo(ownProps.userId, text)
});
}
export default connect(mapStateToProps, actionCreators, mergeProps)(TodoApp);
Troubleshooting
Make sure to check out Troubleshooting Redux first.
My views aren’t updating!
See the link above. In short,
- Reducers should never mutate state, they must return new objects, or React Redux won’t see the updates.
- Make sure you either bind action creators with the
mapDispatchToProps
argument toconnect()
or with thebindActionCreators()
method, or that you manually calldispatch()
. Just calling yourMyActionCreators.addTodo()
function won’t work because it just returns an action, but does not dispatch it.
My views aren’t updating on route change with React Router 0.13
If you’re using React Router 0.13, you might bump into this problem. The solution is simple: whenever you use <RouteHandler>
or the Handler
provided by Router.run
, pass the router state to it.
Root view:
Router.run(routes, Router.HistoryLocation, (Handler, routerState) => { // note "routerState" here
React.render(
<Provider store={store}>
{() => <Handler routerState={routerState} />} // note "routerState" here
</Provider>,
document.getElementById('root')
);
});
Nested view:
render() {
// Keep passing it down
return <RouteHandler routerState={this.props.routerState} />;
}
Conveniently, this gives your components access to the router state! You can also upgrade to React Router 1.0 which shouldn’t have this problem. (Let us know if it does!)
Could not find "store" in either the context or props
If you have context issues,
- Make sure you don’t have a duplicate instance of React on the page.
- Make sure you didn’t forget to wrap your root component in
<Provider>
. - If you use React Router, something like
<Provider>{() => routes}</Provider>
won’t work. Due to the way context works in React 0.13, it’s important that the<Provider>
children are created inside that function. Just referencing an outside variable doesn’t do the trick. Instead of<Provider>{() => routes}</Provider>
, write<Provider>{createRoutes}</Provider>
wherecreateRoutes()
is a function that actually creates (and returns) the route configuration.
Invariant Violation: addComponentAsRefTo(...): Only a ReactOwner can have refs. This usually means that you're trying to add a ref to a component that doesn't have an owner
If you’re using React for web, this usually means you have a duplicate React. Follow the linked instructions to fix this.
If you’re using React Native, make sure you’re importing react-redux/native
both for <Provider>
and any connect()
call. Importing from react-redux
will not work on React Native.
License
MIT