Alert Dialog
Alert Dialog is a modal dialog that interrupts the user with important information and requires a response.
Code
import { motion, AnimatePresence } from "motion/react";
import React from "react";
import { createPortal } from "react-dom";
import { Portal } from "./portal";
export interface AlertDialogContext {
isOpen: boolean;
onClose: () => void;
onOpen: () => void;
onConfirm: () => void;
}
const AlertDialogContext = React.createContext<AlertDialogContext>({
isOpen: false,
onClose: () => {},
onOpen: () => {},
onConfirm: () => {},
});
export function useAlertDialog() {
return React.useContext(AlertDialogContext);
}
export interface AlertDialogProps {
children: React.ReactNode;
isOpen?: boolean;
defaultOpen?: boolean;
onClose?: () => void;
onOpen?: () => void;
onConfirm?: () => void;
}
export function AlertDialog({
children,
isOpen,
defaultOpen,
onClose,
onOpen,
onConfirm,
}: AlertDialogProps) {
const [open, setOpen] = React.useState(defaultOpen ?? false);
function handleClose() {
setOpen(false);
onClose?.();
}
function handleOpen() {
setOpen(true);
onOpen?.();
}
function handleConfirm() {
handleClose();
onConfirm?.();
}
const state = isOpen ?? open;
return (
<AlertDialogContext.Provider
value={{
isOpen: state,
onClose: handleClose,
onOpen: handleOpen,
onConfirm: handleConfirm,
}}
>
{children}
</AlertDialogContext.Provider>
);
}
export interface AlertDialogTriggerProps
extends React.ComponentProps<"button"> {}
export function AlertDialogTrigger(props: AlertDialogTriggerProps) {
const { onOpen } = useAlertDialog();
return (
<button
{...props}
onClick={e => {
onOpen();
props.onClick?.(e);
}}
/>
);
}
export interface AlertDialogContentProps
extends React.ComponentProps<typeof motion.div> {
children: React.ReactNode;
}
export function AlertDialogContent({
children,
...props
}: AlertDialogContentProps) {
const { isOpen } = useAlertDialog();
return (
<AnimatePresence>
{isOpen && (
<Portal>
<motion.div {...props}>{children}</motion.div>
</Portal>
)}
</AnimatePresence>
);
}
export interface AlertDialogBackdropProps
extends React.ComponentProps<typeof motion.div> {}
export function AlertDialogBackdrop(props: AlertDialogBackdropProps) {
const { isOpen } = useAlertDialog();
return (
<AnimatePresence>
{isOpen && (
<Portal>
<motion.div {...props}></motion.div>
</Portal>
)}
</AnimatePresence>
);
}
export interface AlertDialogClose extends React.ComponentProps<"button"> {}
export function AlertDialogClose(props: AlertDialogClose) {
const { onClose } = useAlertDialog();
return (
<button
{...props}
onClick={e => {
onClose();
props.onClick?.(e);
}}
/>
);
}
export interface AlertDialogConfirm extends React.ComponentProps<"button"> {}
export function AlertDialogConfirm(props: AlertDialogConfirm) {
const { onConfirm } = useAlertDialog();
return (
<button
{...props}
onClick={e => {
onConfirm();
props.onClick?.(e);
}}
/>
);
}
Example
import {
AlertDialog,
AlertDialogBackdrop,
AlertDialogClose,
AlertDialogConfirm,
AlertDialogContent,
AlertDialogTrigger,
} from "../ui/primitives/alert-dialog";
export function AlertDialogExample() {
return (
<AlertDialog>
<AlertDialogTrigger className="bg-violet-600 hover:bg-violet-700 transition-colors py-2 px-4 text-white rounded-md">
Alert dialog trigger
</AlertDialogTrigger>
<AlertDialogBackdrop
className="backdrop-blur-xs fixed inset-0"
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
exit={{ opacity: 0 }}
/>
<AlertDialogContent
className="bg-white/80 backdrop-blur-xl shadow-[0_8px_32px_0_rgba(0,0,0,0.2)] border border-white/20 rounded-lg p-4 text-center w-1/3 fixed top-1/2 left-1/2 -translate-1/2"
initial={{ scale: 0.8, opacity: 0 }}
animate={{ scale: 1, opacity: 1 }}
exit={{ scale: 0.8, opacity: 0 }}
>
<h1 className="font-bold text-2xl text-left mb-4">Alert Dialog</h1>
<p className="text-left mb-4">
Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do
eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad
minim veniam, quis nostrud exercitation ullamco laboris nisi ut
aliquip ex ea commodo consequat.
</p>
<div className="flex justify-end gap-4">
<AlertDialogClose className="bg-gray-100 hover:bg-gray-200 transition-colors py-2 px-4 rounded-md">
Cancel
</AlertDialogClose>
<AlertDialogConfirm className="bg-violet-500 hover:bg-violet-600 transition-colors py-2 px-4 rounded-md text-white">
Confirm
</AlertDialogConfirm>
</div>
</AlertDialogContent>
</AlertDialog>
);
}