Design Pattern
HOC Pattern
Within our application, we often want to use the same logic in multiple components. This logic can include applying a certain styling to components, requiring authorization, or adding a global state.
One way of being able to reuse the same logic in multiple components, is by using the higher order component pattern. This pattern allows us to reuse component logic throughout our application.
A Higher Order Component (HOC) is a component that receives another component. The HOC contains certain logic that we want to apply to the component that we pass as a parameter. After applying that logic, the HOC returns the element with the additional logic.
Say that we always wanted to add a certain styling to multiple components in our application. Instead of creating a style
object locally each time, we can simply create a HOC that adds the style
objects to the component that we pass to it
function withStyles(Component) {
return props => {
const style = { padding: '0.2rem', margin: '1rem' }
return <Component style={style} {...props} />
}
}
const Button = () = <button>Click me!</button>
const Text = () => <p>Hello World!</p>
const StyledButton = withStyles(Button)
const StyledText = withStyles(Text)
We just created a StyledButton and StyledText component, which are the modified versions of the Button and Text component. They now both contain the style that got added in the withStyles
HOC!
Let’s take a look at the same DogImages example that was previously used in the Container/Presentational pattern! The application does nothing more than rendering a list of dog images, fetched from an API.
Let’s improve the user experience a little bit. When we’re fetching the data, we
want to show a "Loading..."
screen to the user. Instead of adding data to the DogImages
component directly, we can use a Higher Order Component that adds this logic for
us.
Let’s create a HOC called withLoader
. A HOC should receive an component, and return that component. In this case, the withLoader
HOC should receive the element which should display Loading…
until the data is fetched.
Let’s create the bare minimum version of the withLoader
HOC that we want to use!
function withLoader(Element) {
return (props) => <Element />;
}
However, we don’t just want to return the element it received. Instead, we want this element to contain logic that tells us whether the data is still loading or not.
To make the withLoader
HOC very reusable, we won’t hardcode the Dog API url in that component. Instead, we can pass the URL as an argument to the withLoader
HOC, so this loader can be used on any component that needs a loading indicator while fetching data from a different API endpoint.
function withLoader(Element, url) {
return (props) => {};
}
A HOC returns an element, a functional component props => {}
in this case, to which we want to add the logic that allows us to display a text with Loading…
as the data is still being fetched. Once the data has been fetched, the component should pass the fetched data as a prop.
1import React, { useEffect, useState } from "react";23export default function withLoader(Element, url) {4 return (props) => {5 const [data, setData] = useState(null);67 useEffect(() => {8 async function getData() {9 const res = await fetch(url);10 const data = await res.json();11 setData(data);12 }1314 getData();15 }, []);1617 if (!data) {18 return <div>Loading...</div>;19 }2021 return <Element {...props} data={data} />;22 };23}
Perfect! We just created a HOC that can receive any component and url.
- In the
useEffect
hook, thewithLoader
HOC fetches the data from the API endpoint that we pass as the value ofurl
. While the data hasn’t returned yet, we return the element containing theLoading...
text. - Once the data has been fetched, we set
data
equal to the data that has been fetched. Sincedata
is no longernull
, we can display the element that we passed to the HOC!
So, how can we add this behavior to our application, so it’ll actually show the Loading...
indicator on the DogImages
list?
In DogImages.js
, we no longer want to just export the plain DogImages
component. Instead, we want to export the “wrapped” withLoading
HOC around the DogImages
component.
export default withLoading(DogImages);
The withLoader
HOC also expects the url to know which endpoint to fetch the data from. In this case, we want to add the Dog API endpoint.
export default withLoader(
DogImages,
"https://dog.ceo/api/breed/labrador/images/random/6"
);
Since the withLoader
HOC returned the element with an extra data
prop, DogImages
in this case, we can access the data
prop in the DogImages
component.
1import React from "react";2import withLoader from "./withLoader";34function DogImages(props) {5 return props.data.message.map((dog, index) => (6 <img src={dog} alt="Dog" key={index} />7 ));8}910export default withLoader(11 DogImages,12 "https://dog.ceo/api/breed/labrador/images/random/6"13);
Perfect! We now see a Loading...
screen while the data is being fetched.
The Higher Order Component pattern allows us to provide the same logic to multiple components, while keeping all the logic in one single place. The withLoader
HOC doesn’t care about the component or url it receives: as long as it’s a valid component and a valid API endpoint, it’ll simply pass the data from that API endpoint to the component that we pass.
Composing
We can also compose multiple Higher Order Components. Let’s say that we also want to add functionality that shows a Hovering!
text box when the user hovers over the DogImages
list.
We need to create a HOC that provides a hovering
prop to the element that we pass. Based on that prop, we can conditionally render the text box based on whether the user is hovering over the DogImages
list.
1import React, { useState } from "react";23export default function withHover(Element) {4 return props => {5 const [hovering, setHover] = useState(false);67 return (8 <Element9 {...props}10 hovering={hovering}11 onMouseEnter={() => setHover(true)}12 onMouseLeave={() => setHover(false)}13 />14 );15 };16}
We can now wrap the withHover
HOC around the withLoader
HOC.
1import React from "react";2import withLoader from "./withLoader";3import withHover from "./withHover";45function DogImages(props) {6 return (7 <div {...props}>8 {props.hovering && <div id="hover">Hovering!</div>}9 <div id="list">10 {props.data.message.map((dog, index) => (11 <img src={dog} alt="Dog" key={index} />12 ))}13 </div>14 </div>15 );16}1718export default withHover(19 withLoader(DogImages, "https://dog.ceo/api/breed/labrador/images/random/6")20);
The DogImages
element now contains all props that we passed from both withHover
and withLoader
. We can now conditionally render the Hovering!
text box, based on whether the value of the hovering
prop is true
or false
.
A well-known library used for composing HOCs is recompose. Since HOCs can largely be replaced by React Hooks, the recompose library is no longer maintained, thus won’t be covered in this article.
Hooks
In some cases, we can replace the HOC pattern with React Hooks.
Let’s replace the withHover
HOC with a useHover
hook. Instead of having a higher order component, we export a hook that adds a mouseOver
and mouseLeave
event listener to the element. We cannot pass the element anymore like we did with the HOC. Instead, we’ll return a ref
from the hook for that should get the mouseOver
and mouseLeave
events.
1import { useState, useRef, useEffect } from "react";23export default function useHover() {4 const [hovering, setHover] = useState(false);5 const ref = useRef(null);67 const handleMouseOver = () => setHover(true);8 const handleMouseOut = () => setHover(false);910 useEffect(() => {11 const node = ref.current;12 if (node) {13 node.addEventListener("mouseover", handleMouseOver);14 node.addEventListener("mouseout", handleMouseOut);1516 return () => {17 node.removeEventListener("mouseover", handleMouseOver);18 node.removeEventListener("mouseout", handleMouseOut);19 };20 }21 }, [ref.current]);2223 return [ref, hovering];24}
The useEffect
hook adds an event listener to the component, and sets the value hovering
to true
or false
, depending on whether the user is currently hovering over the element. Both the ref
and hovering
values need to be returned from the hook: ref
to add a ref to the component that should receive the mouseOver
and mouseLeave
events, and hovering
in order to be able to conditionally render the Hovering!
text box.
Instead of wrapping the DogImages
component with the withHover
HOC, we can use the useHover
hook right inside the DogImages
component.
1import React from "react";2import withLoader from "./withLoader";3import useHover from "./useHover";45function DogImages(props) {6 const [hoverRef, hovering] = useHover();78 return (9 <div ref={hoverRef} {...props}>10 {hovering && <div id="hover">Hovering!</div>}11 <div id="list">12 {props.data.message.map((dog, index) => (13 <img src={dog} alt="Dog" key={index} />14 ))}15 </div>16 </div>17 );18}1920export default withLoader(21 DogImages,22 "https://dog.ceo/api/breed/labrador/images/random/6"23);
Perfect! Instead of wrapping the DogImages
component with the withHover
component, we can simply use the useHover
hook within the component directly.
Generally speaking, React Hooks don’t replace the HOC pattern.
“In most cases, Hooks will be sufficient and can help reduce nesting in your tree.” - React Docs
As the React docs tell us, using Hooks can reduce the depth of the component tree. Using the HOC pattern, it’s easy to end up with a deeply nested component tree.
<withAuth>
<withLayout>
<withLogging>
<Component />
</withLogging>
</withLayout>
</withAuth>
By adding a Hook to the component directly, we no longer have to wrap components.
Using Higher Order Components makes it possible to provide the same logic to many components, while keeping that logic all in one single place. Hooks allow us to add custom behavior from within the component, which could potentially increase the risk of introducing bugs compared to the HOC pattern if multiple components rely on this behavior.
Best use-cases for a HOC:
- The same, uncustomized behavior needs to be used by many components throughout the application.
- The component can work standalone, without the added custom logic.
Best use-cases for Hooks:
- The behavior has to be customized for each component that uses it.
- The behavior is not spread throughout the application, only one or a few components use the behavior.
- The behavior adds many properties to the component
Case Study
Some libraries that relied on the HOC pattern added Hooks support after the release. A good example of this is Apollo Client.
No experience with Apollo Client is needed to understand this example.
One way to use Apollo Client is through the graphql()
higher order component.
1import React from "react";2import "./styles.css";34import { graphql } from "react-apollo";5import { ADD_MESSAGE } from "./resolvers";67class Input extends React.Component {8 constructor() {9 super();10 this.state = { message: "" };11 }1213 handleChange = (e) => {14 this.setState({ message: e.target.value });15 };1617 handleClick = () => {18 this.props.mutate({ variables: { message: this.state.message } });19 };2021 render() {22 return (23 <div className="input-row">24 <input25 onChange={this.handleChange}26 type="text"27 placeholder="Type something..."28 />29 <button onClick={this.handleClick}>Add</button>30 </div>31 );32 }33}3435export default graphql(ADD_MESSAGE)(Input);
With the graphql()
HOC, we can make data from the client available to components that are wrapped by the higher order component! Although we can still use the graphql()
HOC currently, there are some downsides to using it.
When a component needs access to multiple resolvers, we need to compose multiple graphql()
higher order components in order to do so. Composing multiple HOCs can make it difficult to understand how the data is passed to your components. The order of the HOCs can matter in some cases, which can easily lead to bugs when refactoring the code.
After the release of Hooks, Apollo added Hooks support to the Apollo Client library. Instead of using the graphql()
higher order component, developers can now directly access the data through the hooks that the library provides.
Let’s look at an example that uses the exact same data as we previously saw in the example with the graphql()
higher order component. This time, we’ll provide the data to the component by using the useMutation
hook that Apollo Client provided for us.
1import React, { useState } from "react";2import "./styles.css";34import { useMutation } from "@apollo/react-hooks";5import { ADD_MESSAGE } from "./resolvers";67export default function Input() {8 const [message, setMessage] = useState("");9 const [addMessage] = useMutation(ADD_MESSAGE, {10 variables: { message }11 });1213 return (14 <div className="input-row">15 <input16 onChange={(e) => setMessage(e.target.value)}17 type="text"18 placeholder="Type something..."19 />20 <button onClick={addMessage}>Add</button>21 </div>22 );23}
By using the useMutation
hook, we reduced the amount of code that was needed in order to provide the data to the component.
Besides a reduction in boilerplate, it’s also much easier to use the data of multiple resolvers in a component. Instead of having to compose multiple higher order components, we can simply write multiple hooks in the component. Knowing how data gets passed to the component is much easier this way, and improves developer experience when refactoring components, or breaking them down into smaller pieces.
Pros
Using the Higher Order Component pattern allows us to keep logic that we want to re-use all in one place. This reduces the risk of accidentally spreading bugs throughout the application by duplicating code over and over, potentially introducing new bugs each time. By keeping the logic all in one place, we can keep our code DRY
and easily enforce separation of concerns.
Cons
The name of the prop that a HOC can pass to an element, can cause a naming collision.
function withStyles(Component) {
return props => {
const style = { padding: '0.2rem', margin: '1rem' }
return <Component style={style} {...props} />
}
}
const Button = () = <button style={{ color: 'red' }}>Click me!</button>
const StyledButton = withStyles(Button)
In this case, the withStyles
HOC adds a prop called style
to the element that we pass to it. However, the Button
component already had a prop called style
, which will be overwritten! Make sure that the HOC can handle accidental name collision, by either renaming the prop or merging the props.
function withStyles(Component) {
return props => {
const style = {
padding: '0.2rem',
margin: '1rem',
...props.style
}
return <Component style={style} {...props} />
}
}
const Button = () = <button style={{ color: 'red' }}>Click me!</button>
const StyledButton = withStyles(Button)
When using multiple composed HOCs that all pass props to the element that’s wrapped within them, it can be difficult to figure out which HOC is responsible for which prop. This can hinder debugging and scaling an application easily.