React Context
Table of Contents:
Terms
Context (English Definition) — The circumstances that form the setting for an event, statement, or idea, and in terms of which it can be fully understood and assessed.
Props Drilling - When we pass a piece of state through 2 or more child components before reaching the component that uses the state.
React Context — the combination of:
an object where we can store global data and
a "wrapper" component that provides its data to all descendants
createContext
— A function exported fromreact
that creates a newContext
object.Context.Provider
— A component created from aContext
object that provides all of its descendants with access to theContext
object's values.useContext
— A React hook for retrieving the values of aContext
object.
Problem: Props Drilling
Consider the instagram clone app that renders something like this:
Note how there is a total tally of likes at the top that sums the likes on each individual picture.
Passing State From App to LikesButton
To achieve the totalLikes
feature, The App
component defines a piece of state called totalLikes
and a function incrementTotalLikes
that invokes setTotalLikes
.
However, the component that uses incrementTotalLikes
is LikesButton
which is 3 layers away from App
.
So we first pass
incrementTotalLikes
toPicturesList
...Which passes it to each
InstagramPost
instance...Which passes it down to
LikesButton
...Which finally uses it.
This is called props drilling — when we pass a piece of state through 2 or more child components before reaching the component that uses the state.
It is okay to pass a piece of state through 1 intermediate component but passing state through 2 or more can start to feel tedious.
Solution: useContext
The solution is to create something called a Context.
In plain english, "context" means:
the circumstances that form the setting for an event, statement, or idea, and in terms of which it can be fully understood and assessed.
In React, Context is the combination of:
an object where we can store global data and
a "wrapper" component that provides its data to all descendants
To use React's Context API, there are 3 concepts to understand.
Making a Context object -
const Context = createContext(startingValue)
Rendering a Context Provider -
<Context.Provider values={values} />
Getting values from a Context -
useContext(Context)
1. Create a context object
This is certainly the simplest step. It will almost always look like this:
We create a new folder called
context/
in oursrc
foldercreateContext
is a named export of thereact
libraryWe invoke
createContext
to create a newInstagramContext
object which we export.It can take in a
defaultValue
argument. For now, we'll just use an empty object.
The InstagramContext
object is the "glue" that makes context possible. We are going to use it to:
Wrap our app's components in a
<InstagramContext.Provider value={contextValues} />
component, providing all descendants with access to thecontextValues
that we choose.Use the
useContext(InstagramContext)
hook to get access to thecontextValues
within a descendent component.
2. Render a Context Provider
<InstagramContext.Provider values={contextValues}/>
is a React component that we wrap around a portion of our app that we want to have access to the InstagramContext
(we can even wrap it around the entire App
if we wanted to).
We first import the InstagramContext
we just created.
InstagramContext.Provider
is a component that we can wrap around any piece of the application that we want to have access to theInstagramContext
.The
value
prop of theInstagramContext.Provider
determines the data available to the children of theInstagramContext.Provider
Now, we can safely remove the
incrementTotalLikes
prop from thePicturesList
and all intermediate components.We can also remove the
totalLikes
prop from theHeader
3. Use the Context
Any component that is a descendant from a InstagramContext.Provider
may utilize the value
of that provider using the useContext
hook from react
:
useContext
is imported fromreact
alongsideuseState
The
InstagramContext
itself is also imported. This will be needed when we invokeuseContext
useContext
is invoked at the top of theLikesButton
component. It takes in aContext
object and returns thevalue
prop of the associatedInstagramContext.Provider
.
⚠️ We can take this even further and use the Context for every value in the application. However, there is a delicate balance between storing TOO much in context and keeping the state close to the components that need it.
Summary
Context provides an alternative to props drilling.
Think of Context as an object where we can store global data and that any component within that context's scope can access that data.
To use React's Context API, there are 3 concepts to understand.
Making a Context object -
const Context = createContext()
Rendering a Context Provider -
<Context.Provider values={} />
Getting values from a Context -
useContext()
Last updated