Files
ExcaliDash/frontend/src/components/ConfirmModal.tsx
T
2025-11-22 10:06:58 -08:00

96 lines
3.7 KiB
TypeScript

import React from 'react';
import { createPortal } from 'react-dom';
import { AlertTriangle, CheckCircle, X } from 'lucide-react';
interface ConfirmModalProps {
isOpen: boolean;
title: string;
message: string;
confirmText?: string;
cancelText?: string;
onConfirm: () => void;
onCancel: () => void;
isDangerous?: boolean; // Makes confirm button red
showCancel?: boolean;
variant?: 'warning' | 'success'; // Controls icon and styling
}
export const ConfirmModal: React.FC<ConfirmModalProps> = ({
isOpen,
title,
message,
confirmText = "Delete",
cancelText = "Cancel",
onConfirm,
onCancel,
isDangerous = true,
showCancel = true,
variant = 'warning'
}) => {
if (!isOpen) return null;
// Icon and styling based on variant
const isSuccess = variant === 'success';
const IconComponent = isSuccess ? CheckCircle : AlertTriangle;
const iconClasses = isSuccess
? "w-12 h-12 rounded-full bg-emerald-100 dark:bg-emerald-900/30 flex items-center justify-center text-emerald-600 dark:text-emerald-300 border-2 border-emerald-200 dark:border-emerald-900/30"
: "w-12 h-12 rounded-full bg-rose-100 dark:bg-rose-900/30 flex items-center justify-center text-rose-600 dark:text-rose-300 border-2 border-rose-200 dark:border-rose-900/30";
return createPortal(
<div className="fixed inset-0 z-[100] flex items-center justify-center p-4">
{/* Backdrop */}
<div
className="absolute inset-0 bg-neutral-900/20 backdrop-blur-sm"
onClick={onCancel}
/>
{/* Modal */}
<div className="relative w-full max-w-md bg-white dark:bg-neutral-900 rounded-2xl border-2 border-black dark:border-neutral-700 shadow-[4px_4px_0px_0px_rgba(0,0,0,1)] dark:shadow-[2px_2px_0px_0px_rgba(255,255,255,0.08)] p-6 animate-in fade-in zoom-in-95 duration-200">
<button
onClick={onCancel}
className="absolute right-4 top-4 text-neutral-400 hover:text-neutral-900 dark:hover:text-white transition-colors"
>
<X size={20} />
</button>
<div className="flex flex-col items-center text-center gap-4">
<div className={iconClasses}>
<IconComponent size={24} strokeWidth={2.5} />
</div>
<div className="space-y-2">
<h3 className="text-xl font-bold text-neutral-900 dark:text-neutral-100 tracking-tight">{title}</h3>
<p className="text-sm font-medium text-neutral-500 dark:text-neutral-400 leading-relaxed">
{message}
</p>
</div>
<div className="flex gap-3 w-full mt-2">
{/* Green for Cancel/No */}
{showCancel && (
<button
onClick={onCancel}
className="flex-1 px-4 py-2.5 bg-emerald-50 dark:bg-neutral-800 text-emerald-700 dark:text-emerald-200 font-bold rounded-xl border-2 border-emerald-200 dark:border-neutral-700 hover:bg-emerald-100 dark:hover:bg-neutral-700 hover:border-emerald-300 dark:hover:border-neutral-600 hover:-translate-y-0.5 transition-all duration-200"
>
{cancelText}
</button>
)}
{/* Red for Confirm/Action */}
<button
onClick={onConfirm}
className={`flex-1 px-4 py-2.5 font-bold rounded-xl border-2 border-black shadow-[2px_2px_0px_0px_rgba(0,0,0,1)] hover:shadow-[4px_4px_0px_0px_rgba(0,0,0,1)] hover:-translate-y-0.5 active:translate-y-0 active:shadow-[2px_2px_0px_0px_rgba(0,0,0,1)] transition-all duration-200 ${isDangerous
? 'bg-rose-600 text-white'
: 'bg-indigo-600 text-white'
}`}
>
{confirmText}
</button>
</div>
</div>
</div>
</div>,
document.body
);
};