2026-02-22 21:35:53 +01:00
|
|
|
'use client';
|
|
|
|
|
|
|
|
|
|
import { cn } from '@/lib/utils/cn';
|
|
|
|
|
import { Check, Copy } from 'lucide-react';
|
|
|
|
|
import { useState } from 'react';
|
|
|
|
|
import { toast } from 'sonner';
|
|
|
|
|
|
|
|
|
|
interface ColorSwatchProps {
|
|
|
|
|
color: string;
|
|
|
|
|
size?: 'sm' | 'md' | 'lg';
|
|
|
|
|
showLabel?: boolean;
|
|
|
|
|
onClick?: () => void;
|
|
|
|
|
className?: string;
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-01 08:15:33 +01:00
|
|
|
export function ColorSwatch({ color, size = 'md', showLabel = true, onClick, className }: ColorSwatchProps) {
|
2026-02-22 21:35:53 +01:00
|
|
|
const [copied, setCopied] = useState(false);
|
|
|
|
|
|
2026-03-01 08:15:33 +01:00
|
|
|
const handleClick = () => {
|
|
|
|
|
if (onClick) { onClick(); return; }
|
2026-02-22 21:35:53 +01:00
|
|
|
navigator.clipboard.writeText(color);
|
|
|
|
|
setCopied(true);
|
|
|
|
|
toast.success(`Copied ${color}`);
|
2026-03-01 08:15:33 +01:00
|
|
|
setTimeout(() => setCopied(false), 1500);
|
2026-02-22 21:35:53 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return (
|
2026-03-01 08:15:33 +01:00
|
|
|
<button
|
|
|
|
|
onClick={handleClick}
|
|
|
|
|
title={color}
|
|
|
|
|
aria-label={`Color ${color}`}
|
|
|
|
|
className={cn(
|
|
|
|
|
'group relative w-full rounded-lg overflow-hidden border border-white/8 transition-all',
|
|
|
|
|
'hover:scale-[1.04] hover:border-white/20 hover:shadow-lg hover:shadow-black/20',
|
|
|
|
|
size === 'sm' && 'h-10',
|
|
|
|
|
size === 'md' && 'h-14',
|
|
|
|
|
size === 'lg' && 'h-20',
|
|
|
|
|
className
|
|
|
|
|
)}
|
|
|
|
|
style={{ backgroundColor: color }}
|
|
|
|
|
>
|
|
|
|
|
<div className="absolute inset-0 flex items-center justify-center opacity-0 group-hover:opacity-100 transition-opacity bg-black/25">
|
|
|
|
|
{copied
|
|
|
|
|
? <Check className="w-3.5 h-3.5 text-white drop-shadow" />
|
|
|
|
|
: <Copy className="w-3.5 h-3.5 text-white drop-shadow" />
|
|
|
|
|
}
|
|
|
|
|
</div>
|
2026-02-22 21:35:53 +01:00
|
|
|
{showLabel && (
|
2026-03-01 08:15:33 +01:00
|
|
|
<div className="absolute bottom-0 inset-x-0 px-1 py-0.5 text-[9px] font-mono text-white/70 bg-black/25 truncate text-center leading-tight">
|
|
|
|
|
{color}
|
|
|
|
|
</div>
|
2026-02-22 21:35:53 +01:00
|
|
|
)}
|
2026-03-01 08:15:33 +01:00
|
|
|
</button>
|
2026-02-22 21:35:53 +01:00
|
|
|
);
|
|
|
|
|
}
|