React Hooks aim to solve the difficulties of logic reuse by enabling us to write components that have access to features like state, context, lifecycle methods, ref, etc. In this article, we’ll be spending our focus on the useContext hook—a useful hook that allows components to access data without the need to rely on props.
When managing data between parent and child components, React gives us the ability to use something known as props to pass data down from parent to child. Props can only flow in one direction, from parent components to child components (and further down). When state changes occur on parent elements, React will re-render components that depend on those values.
Using props works well in most cases. However, when working in large applications with a large number of components in the component tree, props can become hard to maintain since props need to be declared in each and every component in the component tree.
Context, in React, can help make things easier for us in situations like this.
Context in React provides a way to pass data through a component tree without the need to prop-drill (i.e., pass props down manually at every level). We’ll go through a simple example to illustrate how this can be done.
To begin, we first need to create the Context object with the createContext()
function.
import React, { createContext } from "react";
const Context = createContext();
To have context available in the entire component tree, we need to wrap the parent component markup with the Context provider to have all child components subscribe to Context changes.
If we have an <App />
component that acts as the parent component of the application, using the Context provider would look something can look something like the following:
import React, { createContext } from "react";
const Context = createContext();
const App = () => {
return <Context.Provider>/* ... */</Context.Provider>;
};
When declaring the Context provider, we’ll go ahead and specify the context value in the value
prop of <Context.Provider>
. This is how we’ll provide some initial data
for our example.
import React, { createContext } from "react";
const Context = createContext();
const App = () => {
return (
<Context.Provider value={{ data: "Data from context!" }}>
/* ... */
</Context.Provider>
);
};
We’ll now assume <App />
is to render a child component named <Child />
.
import React, { createContext } from "react";
const Context = createContext();
const Child = () => {
return <div>This is the child component!</div>;
};
const App = () => {
return (
<Context.Provider>
<Child />
</Context.Provider>
);
};
We can have the <Child />
component retrieve the current context value with the help of the useContext
Hook.
The useContext
Hook provides function components access to the context value for a context object. It:
React.createContext
) as the one argument it accepts.With this in mind, we’ll have the <Child />
component in our example use the useContext
hook to access the data
property available in our application
context and render its value in its markup.
import React, { createContext, useContext } from "react";
const Context = createContext();
const Child = () => {
const context = useContext(Context);
return <div>{context.data}</div>;
};
const App = () => {
return (
<Context.Provider value={{ data: "Data from context!" }}>
<Child />
</Context.Provider>
);
};
With these changes, we’ll notice the data from the parent <App />
component be rendered in the <Child />
component with the help of the Context API.
I’m using the React Developer Tools Google Chrome extension to visualize the context data in the component tree.
With Context, we would notice the same behavior as we’ve seen above even if we had numerous child components within the component hierarchy tree. As an example, assume we had <Child />
, <Child2 />
,
<Child3 />
,
<Child4 />
and <Child5 />
components where each child component is a parent to the other.
We’ll have the lowest-most component (<Child5 />
) attempt to access and render the Context data value in its markup.
import React, { createContext, useContext } from "react";
const Context = createContext();
const Child5 = () => {
const context = useContext(Context);
return <div>{context.data}</div>;
};
const Child4 = () => {
return <Child5 />;
};
const Child3 = () => {
return <Child4 />;
};
const Child2 = () => {
return <Child3 />;
};
const Child = () => {
return <Child2 />;
};
const App = () => {
return (
<Context.Provider value={{ data: "Data from context!" }}>
<Child />
</Context.Provider>
);
};
Data from the parent <App />
component will be rendered in the <Child5 />
component without the need to prop-drill data through every component in the tree, thanks to Context!
When should one decide to use props vs. context? There are pros and cons to each approach.
With props:
With Context:
Context is best used for application-wide client data. Things like theme information, locale/language preferences, user authentication information, etc. are best kept in Context since any component within an application may have the need to access this data at any time.
Props is best used when data needs to be isolated to a few set of components only.
This article covers the main concept and use case with using the useContext
hook. At any time we want to have client data be available within a large number of React components, we should think of leveraging the Context
API and the useContext
hook.
Hassan is a senior frontend engineer and has helped build large production applications at-scale at organizations like Doordash, Instacart and Shopify. Hassan is also a published author and course instructor where he’s helped thousands of students learn in-depth frontend engineering skills like React, Vue, TypeScript, and GraphQL.