Initial commit

This commit is contained in:
dk
2026-06-23 09:35:55 +00:00
commit f29bb8d1c0
313 changed files with 37703 additions and 0 deletions

View File

@@ -0,0 +1,69 @@
"use client";
import { useState, useEffect } from "react";
import type { ReactNode } from "react";
import { motion, AnimatePresence } from "motion/react";
import { X } from "lucide-react";
import { cls } from "@/lib/utils";
interface ModalProps {
trigger: ReactNode;
title: string;
description?: string;
children: ReactNode;
className?: string;
}
const Modal = ({ trigger, title, description, children, className = "" }: ModalProps) => {
const [isOpen, setIsOpen] = useState(false);
useEffect(() => {
const handleEscape = (e: KeyboardEvent) => {
if (e.key === "Escape") setIsOpen(false);
};
if (isOpen) document.addEventListener("keydown", handleEscape);
return () => document.removeEventListener("keydown", handleEscape);
}, [isOpen]);
return (
<>
<div onClick={() => setIsOpen(true)} className="cursor-pointer">{trigger}</div>
<AnimatePresence>
{isOpen && (
<div className="fixed inset-0 z-50 flex items-center justify-center">
<motion.div
className="absolute inset-0 bg-background/30 backdrop-blur-[1px]"
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
exit={{ opacity: 0 }}
onClick={() => setIsOpen(false)}
/>
<motion.div
className={cls("relative card rounded p-3 xl:p-4 2xl:p-5", className)}
initial={{ opacity: 0, scale: 0.95 }}
animate={{ opacity: 1, scale: 1 }}
exit={{ opacity: 0, scale: 0.95 }}
transition={{ duration: 0.2, ease: [0.4, 0, 0.2, 1] }}
>
<div className="flex items-center justify-between gap-3 xl:gap-4 2xl:gap-5 mb-3 xl:mb-4 2xl:mb-5">
<div className="min-w-0">
<h2 className="text-lg font-medium text-foreground truncate">{title}</h2>
{description && <p className="text-sm text-foreground truncate">{description}</p>}
</div>
<button
onClick={() => setIsOpen(false)}
className="shrink-0 flex items-center justify-center size-9 rounded secondary-button text-secondary-cta-text cursor-pointer"
>
<X className="size-4" />
</button>
</div>
{children}
</motion.div>
</div>
)}
</AnimatePresence>
</>
);
};
export default Modal;