1. 25
    Redux: Passing the Store Down Implicitly via Context
    7m 56s

Redux: Passing the Store Down Implicitly via Context

Dan Abramov
InstructorDan Abramov
Share this video with your friends

Social Share Links

Send Tweet
Published 9 years ago
Updated 4 years ago

Learn how to make the store object available to all components by using the advanced React feature called “context”.

[00:00] In the previous lesson, we got rid of the top level chore variable and instead starting passing this chore as a prop to the to-do app component. So every component below receives this chore as a prop. We even have to do this for presentational components because sometimes they contain container components that need this chore to subscribe to the changes.

[00:24] We have to write a lot of boiler plate code to pass this chore down as a prop. But there is another way, using the advanced React feature called context.

[00:36] I'm creating a new component called provider. From its render method, it just returns whatever its child is. We can wrap any component in a provider, and it's going to render that component.

[00:51] I'm changing the render call to render a to-do app inside the provider. I'm moving this tool prop from the to-do app to the provider component. The provider component will use the React advanced context feature to make this chore available to any component inside it, including grandchildren.

[01:14] To do this, it has to define a special method get child context that will be called by React by using this props tool which corresponds to this chore that is passed to the provider as a prop just once.

[01:30] This tool will be part of the context that the provider specifies for any its children and grandchildren. The to-do app is going to receive this context, and any component inside to do app is also going to receive this context object with this tool inside it.

[01:50] However, there is an important condition for the context to work. This condition is that you have to specify child context types on the component that defines get child context. These are just React prop types definition, but unlike prop types, the child context types are essential for the context to be turned on. If you don't specify them, no child components will receive this context.

[02:18] The container components currently access tool by props, but we're going to change this to read this chore from React context. To do that, we just wrap it to this.context. Similarly, in the render method, I'm also going to read this chore from the context instead of the props.

[02:39] Finally, the context is opt-in for the receiving components, too, so you have to specify a special field called context types, which are similar to child context type. But, in this case, we are specifying which context we want to receive and not pass down. If you forget to declare the context types, the component will not receive the relevant context, so it is essential to remember to declare them.

[03:09] What about the functional components that don't have this? It turns out that they also receive the context but as a second argument after the props. I'm destructuring the second argument and getting this chore from there. The second argument is the context.

[03:28] Just like with the class components, I still have to add a property called, "context types" that specifies which context I want to receive. In this case, I want to receive the chore from the provider. If I forget to declare the context types, my functional component will not receive the relevant context as a second argument. It's important to remember to declare them any time you use the context.

[03:57] Finally, I'm replacing the props with the context when getting this chore for the filter link. I'm adding the context type declaration to the filter link so it receives the relevant context from the provider.

[04:12] Now that the filter link receives this chore by context, I no longer need to pass it as a prop, so I'm removing its usage. I'm also removing the store prop from the footer because it doesn't need to pass it down anymore. I'm also removing this chore prop from the to-do app component because I no longer need to pass it down to the containers.

[04:35] Now, instead of explicitly passing this chore down by props, we pass it implicitly by context.

[04:43] Let's recap how we use the context to pass this chore down. We start by rendering the to-do app inside the provider component we defined above. The provider component just renders whatever you pass through it. In this case, it renders its children or the to do app component. However, it also provides the context to any components inside it including grandchildren.

[05:14] The context contains just one key called the store. It corresponds to the store we passed as a prop to the provider component.

[05:23] We pass this chore to the provider component in our render code and make it available to child components by defining the get child context with the store key pointing to that prop.

[05:36] It is essential that the get child context is matched by child context types where we specify that the store key has prop type of object. Note that the child context types definition is absolutely required if you want to pass the context down the tree.

[05:54] The benefit is that we don't need to pass this chore through the intermediate components. Instead, we can declare the context types on the container components that need access to the store so that they can retrieve it from the context instead of retrieving it from the props.

[06:11] The context creates something like a wormhole between the visible to-do list component that reads the context and the provider that provides the context. This wormhole is only enabled because the context types declared on the visible to-do list include this chore that is defined in child context types of the provider component.

[06:35] The at to-do is another component that needs access to this chore. It also opts into receiving it in the context by specifying the context types. This is why, in addition to props, it receives a second argument, which is the context. I'm using the destruction syntax to grab this chore from the context here.

[06:58] The context works at any depth, so it is not necessary to put context types on the footer. The filter link is the component that directly uses the context, so this is the component that has to specify the context types so that it can use this chore by reading it from the context.

[07:19] Context is a powerful feature, but in a way it contradicts the React philosophy of the explicit data flow. The context essentially allows global variables across the component tree. But global variables are usually a bad idea. Unless you're using it for dependency injection, like here when we need to make a single object available to all components, then probably you shouldn't use context.

[07:46] Finally, the context API is not stable in React. It has changed before, and it is likely to change again. Try your best not to rely on it too much.

Jiwon
Jiwon
~ 7 years ago

I do not understand why you declared it twice.

const { store } = this.context;

like this ''' componentDidMount() { const { store } = this.context; }

render() { const { store } = this.context; } '''

Joe Talarovich
Joe Talarovich
~ 7 years ago

Life cycle methods. componentDidMount is only called on the first render. (Component is receiving the state at time of render) render is called each time the state changes, each time the state changes you will have access to the new state.

Kevin Pinny
Kevin Pinny
~ 7 years ago

Hah - I see the next lesson covers this. Still seems like marginal gains, to have to have all those contextTypes etc declared.

I assume the reason being (however I'm not an experienced React developer), that React likes to be very declarative and explicit. Using context is kinda providing a global variable, which feels implicit, which is the reason for those contextTypes. So it's explicit, like hey, you are kinda using globals.

Robert Smith
Robert Smith
~ 7 years ago

Hah - I see the next lesson covers this. Still seems like marginal gains, to have to have all those contextTypes etc declared.

I assume the reason being (however I'm not an experienced React developer), that React likes to be very declarative and explicit. Using context is kinda providing a global variable, which feels implicit, which is the reason for those contextTypes. So it's explicit, like hey, you are kinda using globals.

Yeah they are like a global. The gains are not marginal when an application scales however. What about components that are very deep in the tree? What happens when you want to move a component to another part of the tree. Without this Provider/Connect pattern it would require a lot of refactoring... This pattern makes things less brittle in that regard.

Sean Cooper
Sean Cooper
~ 7 years ago

I see that the video declares that store: React.PropTypes.object but the example code uses store: ReactPropTypes, the former is not working for me while the latter does, could someone please explain what's going on here?

Bayuadji
Bayuadji
~ 6 years ago

so im guessing this is different with the React.Context yea?

Loïc
Loïc
~ 5 years ago

yet another change with newer react versions (mine is 16.13.1), instead of store: React.PropTypes.object it is store: React.PropTypes

Mathieu
Mathieu
~ 5 years ago

Just a heads up for anyone currently watching this in 2020, the context API changed in React since the recording of this tutorial: https://reactjs.org/docs/context.html

Markdown supported.
Become a member to join the discussionEnroll Today