The Tech Pulse

The Tech Pulse

Higher-Order Components in React

Higher-Order Components in React

Higher-order components (HOCs) are one of those concepts in React that might seem a bit confusing at first. But once you get the hang of them, they can be incredibly powerful. Essentially, an HOC is a function that takes a component and returns a new component. This might sound a bit abstract, so let's break it down.

Most React components return JSX, which represents the DOM elements that should be rendered. HOCs, on the other hand, return another component. Think of them as component factories: functions that create new components. This extra step allows us to add additional behavior or functionality to existing components without modifying them directly.

Why Use Higher-Order Components?

One of the main reasons to use HOCs is to share behavior between multiple components. This is similar to what we do with container components, where we wrap different components in the same container to give them similar behavior. HOCs allow us to do this in a more flexible and reusable way.

Another use case for HOCs is to add extra functionality to an existing component. For example, if you have a component built by someone else and you want to add new features to it, you can use an HOC to wrap the original component and extend its capabilities.

Printing Props with HOCs

Let's start with a simple example: creating an HOC that prints the props of a component. We'll call this HOC printProps. Here's how it works:

  1. Create a new file called printProps.js.
  2. Define the HOC as a function that takes a component as an argument.
  3. Return a new component that logs the props and renders the original component with those props.
export const printProps = (Component) => {
  return (props) => {
    console.log(props);
    return <Component {...props} />;
  };
};

To use this HOC, import it into your main file and wrap your component with it:

import { printProps } from './printProps';
import UserInfo from './UserInfo';

const UserInfoWithProps = printProps(UserInfo);

<UserInfoWithProps a={1} b="hello" c={{ name: 'Shaun' }} />;

When you run this code, you'll see the props logged in the console. This is a simple example, but it shows how HOCs can be used to add functionality to components.

Loading Data with HOCs

Next, let's create an HOC that loads data from a server and passes it to a component. We'll call this HOC withUser. It will fetch user data based on an ID and pass it as a prop to the wrapped component.

  1. Create a new file called withUser.js.
  2. Import necessary hooks and Axios for data fetching.
  3. Define the HOC to take a component and user ID as arguments.
  4. Use useState and useEffect to manage and fetch user data.
  5. Return the original component with the fetched user data as a prop.
import React, { useState, useEffect } from 'react';
import axios from 'axios';

export const withUser = (Component, userId) => {
  return (props) => {
    const [user, setUser] = useState(null);

    useEffect(() => {
      const fetchUser = async () => {
        const response = await axios.get(`/users/${userId}`);
        setUser(response.data);
      };
      fetchUser();
    }, [userId]);

    return <Component {...props} user={user} />;
  };
};

To use this HOC, import it and wrap your component:

import { withUser } from './withUser';
import UserInfo from './UserInfo';

const UserInfoWithLoader = withUser(UserInfo, 234);

<UserInfoWithLoader />;

This will load user data for user ID 234 and pass it to UserInfo as a prop.

Modifying Data with HOCs

Now let's take it a step further and create an HOC that not only loads data but also allows editing it. We'll call this HOC withEditableUser.

  1. Create a new file called withEditableUser.js.
  2. Import necessary hooks and Axios.
  3. Define the HOC to take a component and user ID as arguments.
  4. Use useState and useEffect to manage and fetch user data.
  5. Add functions for changing, saving, and resetting user data.
  6. Return the original component with these functions as props.
import React, { useState, useEffect } from 'react';
import axios from 'axios';

export const withEditableUser = (Component, userId) => {
  return (props) => {
    const [originalUser, setOriginalUser] = useState(null);
    const [user, setUser] = useState(null);

    useEffect(() => {
      const fetchUser = async () => {
        const response = await axios.get(`/users/${userId}`);
        setOriginalUser(response.data);
        setUser(response.data);
      };
      fetchUser();
    }, [userId]);

    const onChangeUser = (changes) => {
      setUser({ ...user, ...changes });
    };

    const onSaveUser = async () => {
      const response = await axios.post(`/users/${userId}`, user);
      setOriginalUser(response.data);
      setUser(response.data);
    };

    const onResetUser = () => {
      setUser(originalUser);
    };

    return (
      <Component
        {...props}
        user={user}
        onChangeUser={onChangeUser}
        onSaveUser={onSaveUser}
        onResetUser={onResetUser}
      />
    );
  };
};

To use this HOC, import it and wrap your component:

import { withEditableUser } from './withEditableUser';
import UserInfoForm from './UserInfoForm';

const UserInfoFormWithEditable = withEditableUser(UserInfoForm, 123);

<UserInfoFormWithEditable />;

This will allow editing user data for user ID 123.

Higher-Order Component Improvements

Finally, let's generalize our withEditableUser HOC to work with any resource. We'll call this new HOC withEditableResource.

  1. Create a new file called withEditableResource.js.
  2. Copy the code from withEditableUser.
  3. Modify it to accept resource path and name as arguments.
  4. Update state variables and functions to be more generic.
import React, { useState, useEffect } from 'react';
import axios from 'axios';

export const withEditableResource = (Component, resourcePath, resourceName) => {
  return (props) => {
    const [originalData, setOriginalData] = useState(null);
    const [data, setData] = useState(null);

    useEffect(() => {
      const fetchData = async () => {
        const response = await axios.get(resourcePath);
        setOriginalData(response.data);
        setData(response.data);
      };
      fetchData();
    }, [resourcePath]);

    const onChange = (changes) => {
      setData({ ...data, ...changes });
    };

    const onSave = async () => {
      const response = await axios.post(resourcePath, { [resourceName]: data });
      setOriginalData(response.data);
      setData(response.data);
    };

    const onReset = () => {
      setData(originalData);
    };

    return (
      <Component
        {...props}
        {...{ [resourceName]: data }}
        {...{ [`onChange${capitalize(resourceName)}`]: onChange }}
        {...{ [`onSave${capitalize(resourceName)}`]: onSave }}
        {...{ [`onReset${capitalize(resourceName)}`]: onReset }}
      />
    );
  };
};

const capitalize = (str) => str.charAt(0).toUpperCase() + str.slice(1);

To use this generalized HOC:

import { withEditableResource } from './withEditableResource';
import UserInfoForm from './UserInfoForm';

const UserInfoFormWithResource = withEditableResource(UserInfoForm, '/users/123', 'user');

<UserInfoFormWithResource />;

This will allow editing any resource by specifying its path and name.

Higher-order components are powerful tools in React for sharing behavior and adding functionality to components in a reusable way. Once you understand how they work, you'll find many opportunities to use them in your projects.

© 2024 The Tech Pulse. All rights reserved.