Toast

Toast Notifications Demo

Installation

1

Install the following packages if you do not have it.

npm i framer-motion lucide-react
2

Copy and paste the following code into your project.

import * as React from "react";
import { motion, AnimatePresence } from "framer-motion";
import { X, CheckCircle, AlertCircle, Info } from "lucide-react";

interface ToastProps {
  message: string;
  type?: "success" | "error" | "info";
  duration?: number;
  onClose: () => void;
}

const Toast: React.FC<ToastProps> = ({
  message,
  type = "info",
  duration = 5000,
  onClose,
}) => {
  const [progress, setProgress] = React.useState(100);

  React.useEffect(() => {
    const timer = setInterval(() => {
      setProgress((prev) => {
        if (prev > 0) {
          return prev - 100 / (duration / 1000);
        }
        clearInterval(timer);
        return 0;
      });
    }, 1000);

    const closeTimer = setTimeout(onClose, duration);

    return () => {
      clearInterval(timer);
      clearTimeout(closeTimer);
    };
  }, [duration, onClose]);

  const typeStyles = {
    success: {
      gradient: "from-green-400 to-emerald-600",
      icon: <CheckCircle className="w-6 h-6 text-emerald-100" />,
    },
    error: {
      gradient: "from-red-400 to-rose-600",
      icon: <AlertCircle className="w-6 h-6 text-rose-100" />,
    },
    info: {
      gradient: "from-blue-400 to-indigo-600",
      icon: <Info className="w-6 h-6 text-indigo-100" />,
    },
  }[type];

  return (
    <motion.div
      layout
      initial={{ opacity: 0, y: 50, scale: 0.5 }}
      animate={{
        opacity: 1,
        y: 0,
        scale: 1,
        transition: { type: "spring", stiffness: 200, damping: 20 },
      }}
      exit={{
        opacity: 0,
        scale: 0.5,
        transition: { duration: 0.2, ease: "easeInOut" },
      }}
      className={`relative flex items-center w-full max-w-sm p-4 rounded-lg shadow-lg bg-gradient-to-br ${typeStyles.gradient} backdrop-blur-xl`}
    >
      <div className="flex items-center justify-center w-10 h-10 mr-4 bg-white bg-opacity-25 rounded-full">
        {typeStyles.icon}
      </div>
      <div className="flex-grow pr-8">
        <p className="text-sm font-medium text-white">{message}</p>
      </div>
      <motion.button
        onClick={onClose}
        className="absolute top-2 right-2 flex items-center justify-center w-6 h-6 text-white bg-black bg-opacity-25 rounded-full hover:bg-opacity-40 transition-colors"
        whileHover={{ scale: 1.1 }}
        whileTap={{ scale: 0.95 }}
        aria-label="Close"
      >
        <X size={14} />
      </motion.button>
      <div
        className="absolute bottom-0 left-0 h-1 bg-white bg-opacity-50 rounded-full"
        style={{ width: `${progress}%`, transition: "width 1s linear" }}
      />
    </motion.div>
  );
};

export const ToastContainer: React.FC<{
  toasts: Array<{
    id: number;
    message: string;
    type: "success" | "error" | "info";
  }>;
}> = ({ toasts }) => (
  <div className="fixed bottom-6 right-6 flex flex-col space-y-4 items-end sm:bottom-8 sm:right-8 z-50">
    <AnimatePresence>
      {toasts.map((toast, index) => (
        <motion.div
          key={toast.id}
          initial={{ opacity: 0, y: 50, scale: 0.3 }}
          animate={{
            opacity: 1,
            y: 0,
            scale: 1,
            transition: { delay: index * 0.1 },
          }}
          exit={{
            opacity: 0,
            scale: 0.5,
            transition: { duration: 0.2 },
          }}
        >
          <Toast
            message={toast.message}
            type={toast.type}
            onClose={() => console.log("Toast removed")}
          />
        </motion.div>
      ))}
    </AnimatePresence>
  </div>
);
3

Implement the code as demonstrated in the preview