DEV Community

Cover image for How To Create Alert In React JS. Master Alerts In React JS.
Sajith P J
Sajith P J

Posted on

How To Create Alert In React JS. Master Alerts In React JS.

Heyy DevelopersπŸ™‹β€β™‚οΈ.

Did you know how to create an Alert box in React JS. I know, you already know this. Maybe I can share a different method, which will help you ignore repeated state usage and other lines of code.

Let's Explore πŸ™Œ

Before We Start

Be ready with the react js project with you and install classnames. Run the following command to install classnames.

npm i classnames
Enter fullscreen mode Exit fullscreen mode

We are using some dynamic classNames to style our component.

The Basic Method To Create An Alert

Usually, we create the Alert component inside of our components and we use useState to conditionally render the alert.

First, Let's see how it will be

import { useState } from "react";
import SuccessAlert from "./component/SuccessAlert";
import WarningAlert from "./component/WarningAlert";
import ErrorAlert from "./component/ErrorAlert";

const App = () => {
  const [alert, setAlert] = useState("");

  const showAlert = (type) => setAlert(type);
  return (
    <div>
      {/* rest of the UI */}
      <button type="button" onClick={() => showAlert("warning")}>
        Show Warning Alert
      </button>
      <button type="button" onClick={() => showAlert("success")}>
        Show Success Alert
      </button>
      <button type="button" onClick={() => showAlert("error")}>
        Show Error Alert
      </button>

      {alert == "success" ? (
        <SuccessAlert />
      ) : alert == "warning" ? (
        <WarningAlert />
      ) : alert == "error" ? (
        <ErrorAlert />
      ) : (
        ""
      )}
    </div>
  );
};

export default App
Enter fullscreen mode Exit fullscreen mode

Probably, this is how you will create alerts. Here, the alerts are loaded by using the alerts and they are not reusable, and these components will mount into the root div of React Js.

Why Don't You Use This Method?❗❗❗

  • Less Reusability.
  • Repeated use of state.
  • Mounting into the root div/inside parent elements.

Let's Create A Better Version Of AlertπŸ™Œ

Step 1: Create A Function, Which Creates Alerts Dynamically
Let's create a folder in src directory called components,
Inside of components create another folder called alertist.
Create a file called index.js inside alertist.

So the folder structure will be like this.

my-vite-react-app/
β”œβ”€β”€ node_modules/
β”œβ”€β”€ public/
β”‚   └── vite.svg
β”œβ”€β”€ src/
β”‚   β”œβ”€β”€ assets/
β”‚   β”‚   └── react.svg
β”‚   β”œβ”€β”€ components/
β”‚   β”‚   └── alertist/
β”‚   β”‚       └── index.js
β”‚   β”œβ”€β”€ App.css
β”‚   β”œβ”€β”€ App.jsx
β”‚   β”œβ”€β”€ index.css
β”‚   β”œβ”€β”€ main.jsx
β”‚   └── vite-env.d.ts
β”œβ”€β”€ .gitignore
β”œβ”€β”€ index.html
β”œβ”€β”€ package.json
β”œβ”€β”€ README.md
β”œβ”€β”€ vite.config.js
└── package-lock.json
Enter fullscreen mode Exit fullscreen mode

We add the code first and I will explain about it.

src/components/alertist/index.js

import { createRoot } from "react-dom/client";
import Alertist from "./components/Alertist";
import { createPortal } from "react-dom";
let alertRoots = [];
const alertist = (props) => {
  const alertCount = document.querySelectorAll(".alert").length;
  const portalNode = document.querySelector("#alert_1");
  if (!portalNode) {
    const div = document.createElement("div");
    div.id = "alert_1";
    div.className = "alert";
    document.querySelector("#root")?.append(div);
    alertRoots.push(createRoot(div));
  } else {
    const div = document.createElement("div");
    div.id = `alert_${modalCount + 1}`;
    div.className = "alert";
    document.querySelector("#root")?.append(div);
    alertRoots.push(createRoot(div));
  }
  if (alertRoots.length > 0) {
    let root = alertRoots.find(
      (root) => root._internalRoot.containerInfo.id == `alert_${alertCount + 1}`
    );
    const addedPortalNode = document.querySelector(`#alert_${alertCount + 1}`);
    if (addedPortalNode) {
      root.render(createPortal(<Alertist {...props} addedPortalNode={addedPortalNode} alertRoots={alertRoots} />, addedPortalNode));
    }
  }
};

export { alertist };
Enter fullscreen mode Exit fullscreen mode

The code defines a function called alertist that dynamically creates an alert in a React application. Each alert modal is created in a new div element and rendered using React's createPortal function. The Alertist component is responsible for the content of these alert modals.

  • Determine the Number of Existing Alerts:
    alertCount: Counts the number of existing alert modals by querying
    elements with the class name "alert".

  • Check for Existing Alert Node:
    portalNode: Looks for an existing element with the ID #alert_1.

  • Create New Alert Node:
    If no element with ID #alert_1 is found:

    • Creates a new div with ID alert_1 and class name "alert".
    • Appends this div to the #root element in the DOM.
    • Creates a React root for this div and adds it to alertRoots.

If an element with ID #alert_1 is found:
- Creates a new div with a dynamically generated ID
(alert_${alertCount + 1}) and class name "alert".
- Appends this div to the #root element in the DOM.
- Creates a React root for this div and adds it to alertRoots.

  • Render the Alert Component:

    • If alertRoots array is not empty:
      • Finds the root in alertRoots that corresponds to the newly created alert div.
      • Uses createPortal to render the Alertist component into the newly created div.
      • Passes props, the newly created alert div (as addedPortalNode), and the alertRoots array to the Alertist component.
  • createRoot: A React function from react-dom/client used to create a root for rendering React components.

  • createPortal: A React function from react-dom used to render children into a DOM node that exists outside the DOM hierarchy of the parent component.

After this, you will notice that we are using a component called Alertist in this alertist function. It's time to create that function.

Step 2: Create All The Components For Alert

First, Create the <Alertist />.
So, create a folder called components inside src/components/Alert.
Add a file named Alertist.jsx.

And the following code to Alertist.jsx.


import { useState } from "react";
import Alert from "./Alert";

const Alertist = (props) => {
  const [show, setShow] = useState(true);

  return show ? <Alert {...props} setShow={setShow} /> : <div />;
};

export default Alertist;

Enter fullscreen mode Exit fullscreen mode

This is just a simple component that renders the <Alert/> according to the state show.

Here, You can see that I am importing and using a component called <Alert />.

Let's create Alert.jsx.
So, create a file called Alert.jsx, inside src/components/Alert/components directory.

Add the below code inside Alert.jsx.

import { useEffect } from "react";
import AlertOverlay from "./AlertOverlay";
import AlertLayout from "./AlertLayout";
import AlertIcon from "./AlertIcon";
import classNames from "classnames";
const Alert = ({
  show,
  setShow,
  type,
  title,
  subtitle,
  subContent,
  successButton,
  cancelButton,
  timer,
  onClose,
  alertRoots,
  addedPortalNode,
}) => {
  const closeAlert = () => {
    if (onClose) onClose();
    document.body.style.overflow = "auto";
    alertRoots.pop();
    addedPortalNode?.remove();
  };
  const autoCloseAfterTimer = () => {
    setTimeout(() => closeAlert(), timer);
  };
  const closingOnClick = () => {
    closeAlert();
  };

  const onCancelButtonClick = () => {
    cancelButton !== undefined &&
      cancelButton.onClick !== undefined &&
      cancelButton?.onClick();
    closeAlert();
  };
  const onSuccessButtonClick = () => {
    successButton !== undefined &&
      successButton.onClick !== undefined &&
      successButton?.onClick();
    closeAlert();
  };

  useEffect(() => {
    if (timer) {
      autoCloseAfterTimer();
    }
  }, []);

  return (
    <>
      <AlertOverlay onClick={closingOnClick} />
      <AlertLayout onClose={closingOnClick}>
        <AlertIcon type={type} width={100} />
        <div className="flex justify-center items-center flex-col">
          <div className="w-full flex justify-center items-center flex-col">
            <h1 className="text-[24px] font-extrabold  tracking-wide">
              {title}
            </h1>
            <p className="text-black tracking-wide text-center">
              {subtitle}
            </p>
          </div>
        </div>
        {typeof subContent !== "undefined" && <div>{subContent()}</div>}
        <div className="flex justify-center items-center my-[16px] gap-4">
          {cancelButton?.show && (
            <div>
              <button
                type="button"
                className={classNames(
                  "rounded-[10px]",
                  cancelButton?.className
                )}
                onClick={onCancelButtonClick}
              >
                {cancelButton?.displayText}
              </button>
            </div>
          )}
          {successButton?.show && (
            <div>
              <button
                type="button"
                className={classNames(
                  "rounded-[10px]",
                  successButton?.className
                )}
                onClick={onSuccessButtonClick}
              >
                {successButton?.displayText}
              </button>
            </div>
          )}
        </div>
      </AlertLayout>
    </>
  );
};

export default Alert;
Enter fullscreen mode Exit fullscreen mode

Explanation

Props

  • show: Boolean to control the visibility of the alert.
  • setShow: Function to set the visibility of the alert.
  • type: Type of the alert, used to determine the icon and style.
  • title: Title text of the alert.
  • subtitle: Subtitle text of the alert.
  • subContent: Additional content to display in the alert.
  • successButton: Object with properties to configure the success button.
  • cancelButton: Object with properties to configure the cancel button.
  • timer: Time in milliseconds after which the alert will automatically close.
  • onClose: Function to call when the alert is closed.
  • alertRoots: Array of alert root elements.
  • addedPortalNode: The DOM node where the alert is rendered.

Internal Functions

  • closeAlert:
    • Calls the onClose function if provided.
    • Restores the body overflow style to allow scrolling.
    • Removes the last element from the alertRoots array.
    • Removes the addedPortalNode from the DOM.
  • autoCloseAfterTimer:
    • Calls closeAlert after the specified timer duration using setTimeout.
  • closingOnClick: Calls closeAlert when the alert overlay is clicked.
  • onCancelButtonClick:
    • Calls the onClick handler of the cancelButton if provided.
    • Calls closeAlert.
  • onSuccessButtonClick:
    • Calls the onClick handler of the successButton if provided.
    • Calls closeAlert.

useEffect Hook
The **useEffect** hook is used to set up a timer to automatically close the alert if the timer prop is provided. This effect runs once when the component mounts because the dependency array is empty ([]).

Return Statement
The Alert component return a UI for a basic, styled with Tailwind CSS, alert box containing overlay, buttons, close button, etc.

Other Components Used

  • : AlertOverlay component to capture click events for closing the alert, and return a blurred overlay for the alert.

To create this component, create a file called AlertOverlay.jsx, inside src/components/Alert/components.

Add the below code to AlertOverlay.jsx

import { useEffect } from "react";

const AlertOverlay = ({ onClick }) => {
  useEffect(() => {
    document.body.style.overflow = "hidden";
  }, []);
  return (
    <div
      className="fixed top-0 left-0 w-full h-screen bg-gradient-to-r from-[#ffffff07] to-[#ffffff31] backdrop-blur-[5px] z-50 overflow-hidden"
      onClick={onClick}
    ></div>
  );
};

export default AlertOverlay;
Enter fullscreen mode Exit fullscreen mode

This component will receive onClick.
The onClick function will used to close the alert when you click outside of the alert.

  • **<AlertLayout />**: This is a component, that contains the alert's content, and it returns a basic UI for the alert.

To create this component, create a file called AlertLayout.jsx, inside src/components/Alert/components.

Add the following code to AlertLayout.jsx


import React from "react";
import { AnimatePresence, motion } from "framer-motion";
import CloseIcon from "../../../assets/icons/Close";

const AlertLayout = ({ children, onClose }) => {
  return (
    <div className={`fixed max-w-[450px] w-full min-h-[300px] z-[51] bg-white top-1/2 left-1/2 transform translate-x-[-50%] translate-y-[-50%] rounded-[10px]  shadow-xl shadow-[#eeeeee]`}>
      <div
        className="w-full h-full relative p-5 min-h-[300px] flex justify-center items-center flex-col "
      >
        <button type="button" className="absolute top-[10px] left-[10px]" onClick={onClose}>
          <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 384 512"  width="20" height="20">
      <path
        d="M342.6 150.6c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L192 210.7 86.6 105.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3L146.7 256 41.4 361.4c-12.5 12.5-12.5 32.8 0 45.3s32.8 12.5 45.3 0L192 301.3 297.4 406.6c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3L237.3 256 342.6 150.6z"
      />
    </svg>
        </button>
        {children}
      </div>
    </div>
  );
};

export default AlertLayout;
Enter fullscreen mode Exit fullscreen mode
  • **<AlertIcon />**: This is a component used to display an icon based on the alert type.

To create this component, create a file called AlertIcon.jsx, inside src/components/Alert/components.
Add the following code to AlertIcon.jsx.

import ErrorIcon from "../icons/Alertist/ErrorIcon";
import InfoIcon from "../icons/Alertist/InfoIcon";
import OopsIcon from "../icons/Alertist/OopsIcon";
import SuccessIcon from "../icons/Alertist/SuccessIcon";
import WarningIcon from "../icons/Alertist/WarningIcon";

const ICONS = [
  {
    type: "oops",
    src: (props) => <OopsIcon {...props} />,
  },
  {
    type: "error",
    src: (props) => <ErrorIcon {...props} />,
  },
  {
    type: "warning",
    src: (props) => <WarningIcon {...props} />,
  },
  {
    type: "success",
    src: (props) => <SuccessIcon {...props} />,
  },
  {
    type: "info",
    src: (props) => <InfoIcon {...props} />,
  },
];

const AlertIcon = ({ type, width, height }) => {
  const getIcon = (type) => {
    return ICONS.filter((icon) => icon.type === type)[0].src;
  };

  let iconComponent = getIcon(type);
  // iconComponent.props = Object.assign({}, iconComponent.props, { width, height });

  return (
    <div className="flex justify-center items-center">
      <div>{iconComponent({ width, height })}</div>
    </div>
  );
};

export default AlertIcon;
Enter fullscreen mode Exit fullscreen mode

Add A folder for the icons, called icons, In the src/components/Alert

Add the SVG code for the icons in the specified directories.

src/components/Alert/icons/ErrorIcon.jsx

const ErrorIcon = (props) => {
  return (
    <svg
      width="45"
      height="45"
      viewBox="0 0 45 45"
      fill="none"
      xmlns="http://www.w3.org/2000/svg"
      {...props}
    >
      <path
        d="M31.3403 14.7871L13.3403 30.7871"
        stroke="#F31717"
        strokeWidth="3.61702"
        strokeLinecap="round"
        strokeLinejoin="round"
      />
      <path
        d="M13.3403 14.7871L31.3403 30.7871"
        stroke="#F31717"
        strokeWidth="3.61702"
        strokeLinecap="round"
        strokeLinejoin="round"
      />
      <circle
        cx="22.5"
        cy="22.5"
        r="20.6915"
        stroke="#F31717"
        strokeWidth="3.61702"
      />
    </svg>
  );
};

export default ErrorIcon;
Enter fullscreen mode Exit fullscreen mode

src/components/Alert/icons/InfoIcon.jsx

const InfoIcon = (props) => {
  return (
    <svg
      xmlns="http://www.w3.org/2000/svg"
      viewBox="0 0 48 48"
      width="48px"
      height="48px"
      {...props}
    >
      <circle cx="28" cy="28" r="18.5" fill="#90caf9" />
      <path
        fill="none"
        stroke="#18193f"
        strokeLinecap="round"
        strokeLinejoin="round"
        strokeMiterlimit="10"
        strokeWidth="3"
        d="M31.4,41c-2.3,1-4.8,1.5-7.4,1.5C13.8,42.5,5.5,34.2,5.5,24c0-4.5,1.6-8.6,4.2-11.8"
      />
      <path
        fill="none"
        stroke="#18193f"
        strokeLinecap="round"
        strokeLinejoin="round"
        strokeMiterlimit="10"
        strokeWidth="3"
        d="M16.3,7.2c2.3-1.1,5-1.7,7.7-1.7c10.2,0,18.5,8.3,18.5,18.5c0,4-1.3,7.7-3.4,10.7"
      />
      <circle cx="24" cy="16" r="2" fill="#18193f" />
      <line
        x1="24"
        x2="24"
        y1="22.5"
        y2="33.5"
        fill="none"
        stroke="#18193f"
        strokeLinecap="round"
        strokeMiterlimit="10"
        strokeWidth="3"
      />
    </svg>
  );
};

export default InfoIcon;
Enter fullscreen mode Exit fullscreen mode

src/components/Alert/icons/OopsIcon.jsx

import React from "react";

const OopsIcon = (props) => {
  return (
    <svg
      width="153"
      height="153"
      viewBox="0 0 153 153"
      fill="none"
      xmlns="http://www.w3.org/2000/svg"
      {...props}
    >
      <rect width="153" height="153" fill="url(#pattern0)" />
      <defs>
        <pattern
          id="pattern0"
          patternContentUnits="objectBoundingBox"
          width="1"
          height="1"
        >
          <use xlinkHref="#image0_629_1106" transform="scale(0.00195312)" />
        </pattern>
        <image
          id="image0_629_1106"
          width="512"
          height="512"
          xlinkHref=""
        />
      </defs>
    </svg>
  );
};

export default OopsIcon;
Enter fullscreen mode Exit fullscreen mode

I took this image and converted it as svg😁.

src/components/Alert/icons/SuccessIcon.jsx

const SuccessIcon = (props) => {
  return (
    <svg
      width="60"
      height="60"
      viewBox="0 0 60 60"
      fill="none"
      xmlns="http://www.w3.org/2000/svg"
      {...props}
    >
      <path
        d="M44.6597 20.2129L27.7561 37.2129L21.6597 30.6962"
        stroke="#209625"
        strokeWidth="3.61702"
        strokeLinecap="round"
        strokeLinejoin="round"
      />
      <circle
        cx="33.1597"
        cy="28.7129"
        r="20.6915"
        stroke="#209625"
        strokeWidth="3.61702"
      />
    </svg>
  );
};

export default SuccessIcon;
Enter fullscreen mode Exit fullscreen mode

src/components/Alert/icons/WarningIcon.jsx

import React from "react";

const WarningIcon = (props) => {
  return (
    <svg
      width="153"
      height="153"
      viewBox="0 0 153 153"
      fill="none"
      xmlns="http://www.w3.org/2000/svg"
      xmlnsXlink="http://www.w3.org/1999/xlink"
      {...props}
    >
      <rect width="153" height="153" fill="url(#pattern0)" />
      <defs>
        <pattern
          id="pattern0"
          patternContentUnits="objectBoundingBox"
          width="1"
          height="1"
        >
          <use xlinkHref="#image0_629_1117" transform="scale(0.00195312)" />
        </pattern>
        <image
          id="image0_629_1117"
          width="512"
          height="512"
          xlinkHref=""
        />
      </defs>
    </svg>
  );
};

export default WarningIcon;
Enter fullscreen mode Exit fullscreen mode

FEEL FREE TO REPLACE THE ICONS

Whoooohh, That a loooongg setup. Isnt it?

That's all we need for the Alert.

How To Use In The Code ?

Let's take the example of buttons, that we use as the example.

import { useState } from "react";
import { alertist } from './components/Alert';

const App = () => {
  const showAlert = (type) => setAlert(type);

  const showWarningAlert = () => {
    alertist({
      type: "warning",
      title: "Warning",
      subtitle: "The warning message",
      timer: 2000, // close in 2s,
      successButton: {
        show: true,
        displayText: "Okay",
        className: "bg-primary text-white px-4 py-2 rounded-lg text-[14px]",
        onClick: () => handleErrorAlertSuccessClick(status),
      },
      cancelButton: {
        show: false, // cancel button will be hidden
      },
    });
  }

   const showSuccessAlert = () => {
   // timer is disabled
    alertist({
      type: "success",
      title: "Success",
      subtitle: "The success message",
      successButton: {
        show: true,
        displayText: "Okay",
        className: "bg-primary text-white px-4 py-2 rounded-lg text-[14px]",
        onClick: () => handleErrorAlertSuccessClick(status),
      },
      cancelButton: {
        show: false, // cancel button will be hidden
      },
    });
  }
  return (
    <div>
      {/* rest of the UI */}
      <button type="button" onClick={showWarningAlert}>
        Show Warning Alert
      </button>
      <button type="button" onClick={showSuccessAlert}>
        Show Success Alert
      </button>
    </div>
  );
};

export default App
Enter fullscreen mode Exit fullscreen mode

Conclucion

By creating this component, you dont need to manage the different states for different Alerts anymore, and you can modify this code for moree resuability.

Cool😎, I think you learned something new today.

About Me

I am Sajith P J, Senior React JS Developer, A JavaScript developer with the entrepreneurial mindset. I combine my experience with the super powers of JavaScript to satisfy your requirements.

*Reach Me Out *

Thanks !!!πŸ™Œ

Top comments (0)