28. Generating Containers with connect() from React Redux (AddTodo)

Video Link

In the last section we used connect() to set up our VisibleTodoList component.

Since these examples have all of our JavaScript written in a single file, we need to rename our mapStateToProps and mapDispatchToProps functions to be more specific. Note that this doesn't need to be done if we keep all of our components in separate files.

Recall that our AddTodo component wasn't clearly a presentational or container component. However, it does rely on store for the dispatch() function.

Instead of reading store from the context, we are going to refactor AddTodo to read dispatch from the props. This is because AddTodo only needs dispatch(), not the whole store.

We will be creating a container component using connect() that will inject the dispatch function as a prop. We will remove AddTodo.contextTypes because the component generated by the connect() function will take care of reading the store from the context.

Before:
const AddTodo = (props, { store }) => {
  .
  . // inside `return`
  .
    <button onClick={() => {
      store.dispatch({
      .
      .
      .

}
AddTodo.contextTypes = {
  store: React.PropTypes.object
}
After:
let AddTodo = ({ dispatch }) => {
  .
  . // inside `return`
  .
    <button onClick={() => {
      dispatch({
      .
      .
      .
}
// No more `AddTodo.contextTypes`

Notice that we changed const to let in our declaration. This lets us reassign AddTodo so the consuming component doesn't need to specify the dispatch prop. We don't have to specify dispatch as a prop because it will be injected by the component generated by the connect call.

The first argument to connect() is mapStateToProps, but there aren't any props for our AddTodo component that depend on the current state. Because of this, we'll have our first parameter return an empty object.

The second argument to connect() is mapDispatchToProps, but AddTodo doesn't need any callback props. Because of this, we'll just return the dispatch function itself as a prop with the same name.

We'll call the function a second time to specify the component we want to wrap (in this case AddTodo itself).

AddTodo = connect(
  state => {
    return {};
  },
  dispatch => {
    return { dispatch };
  }
)(AddTodo);

Now AddTodo won't pass any props dependent on state, but it will pass dispatch() itself as a function so that the component can read from the props and use it without worrying about context or specifying any ContextTypes.

But it's wasteful...

Why subscribe to the store if we aren't going to calculate props from the state? Because we don't need to subcribe to the store, we can call connect() without mapStateToProps as an argument, instead passing in null. What this does is tell connect that there is no need to subscribe to the store.

It's a common pattern to inject just the dispatch function, so if connect() sees that the second argument is null (or any falsey value), you'll get dispatch injected as a prop.

What this means is that we can accomplish the same effect as the above code by removing the arguments from the connect function:

AddTodo = connect()(AddTodo)

Now the default behavior to not subscribe to the store, and inject dispatch as a prop.

3:43 in the video has the recap.

results matching ""

    No results matching ""