UNPKG

@arolariu/components

Version:

🎨 70+ beautiful, accessible React components built on Radix UI. TypeScript-first, tree-shakeable, SSR-ready. Perfect for modern web apps, design systems & rapid prototyping. Zero config, maximum flexibility! ⚡

147 lines (146 loc) • 5.77 kB
"use client"; import { jsx, jsxs } from "react/jsx-runtime"; import { cn } from "../../lib/utilities.js"; import { motion, useAnimation } from "motion/react"; import { useEffect, useRef, useState } from "react"; const defaultGradientColors = [ "#A97CF8", "#F38CB8", "#FDCC92" ]; const Scratcher = ({ width, height, minScratchPercentage = 50, onComplete, children, className, gradientColors = defaultGradientColors })=>{ const canvasRef = useRef(null); const [isScratching, setIsScratching] = useState(false); const [isComplete, setIsComplete] = useState(false); const controls = useAnimation(); useEffect(()=>{ const canvas = canvasRef.current; const ctx = canvas?.getContext("2d"); if (canvas && ctx) { ctx.fillStyle = "#ccc"; ctx.fillRect(0, 0, canvas.width, canvas.height); const gradient = ctx.createLinearGradient(0, 0, canvas.width, canvas.height); gradient.addColorStop(0, gradientColors[0]); gradient.addColorStop(0.5, gradientColors[1]); gradient.addColorStop(1, gradientColors[2]); ctx.fillStyle = gradient; ctx.fillRect(0, 0, canvas.width, canvas.height); } }, [ gradientColors ]); useEffect(()=>{ const handleDocumentMouseMove = (event)=>{ if (!isScratching) return; scratch(event.clientX, event.clientY); }; const handleDocumentTouchMove = (event)=>{ if (!isScratching) return; const [touch] = event.touches; if (!touch) return; scratch(touch.clientX, touch.clientY); }; const handleDocumentMouseUp = ()=>{ setIsScratching(false); checkCompletion(); }; const handleDocumentTouchEnd = ()=>{ setIsScratching(false); checkCompletion(); }; document.addEventListener("mousedown", handleDocumentMouseMove); document.addEventListener("mousemove", handleDocumentMouseMove); document.addEventListener("touchstart", handleDocumentTouchMove); document.addEventListener("touchmove", handleDocumentTouchMove); document.addEventListener("mouseup", handleDocumentMouseUp); document.addEventListener("touchend", handleDocumentTouchEnd); document.addEventListener("touchcancel", handleDocumentTouchEnd); return ()=>{ document.removeEventListener("mousedown", handleDocumentMouseMove); document.removeEventListener("mousemove", handleDocumentMouseMove); document.removeEventListener("touchstart", handleDocumentTouchMove); document.removeEventListener("touchmove", handleDocumentTouchMove); document.removeEventListener("mouseup", handleDocumentMouseUp); document.removeEventListener("touchend", handleDocumentTouchEnd); document.removeEventListener("touchcancel", handleDocumentTouchEnd); }; }, [ isScratching ]); const handleMouseDown = ()=>setIsScratching(true); const handleTouchStart = ()=>setIsScratching(true); const scratch = (clientX, clientY)=>{ const canvas = canvasRef.current; const ctx = canvas?.getContext("2d"); if (canvas && ctx) { const rect = canvas.getBoundingClientRect(); const x = clientX - rect.left + 16; const y = clientY - rect.top + 16; ctx.globalCompositeOperation = "destination-out"; ctx.beginPath(); ctx.arc(x, y, 30, 0, 2 * Math.PI); ctx.fill(); } }; const startAnimation = async ()=>{ await controls.start({ scale: [ 1, 1.5, 1 ], rotate: [ 0, 10, -10, 10, -10, 0 ], transition: { duration: 0.5 } }); if (onComplete) onComplete(); }; const checkCompletion = ()=>{ if (isComplete) return; const canvas = canvasRef.current; const ctx = canvas?.getContext("2d"); if (canvas && ctx) { const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const pixels = imageData.data; const totalPixels = pixels.length / 4; let clearPixels = 0; for(let i = 3; i < pixels.length; i += 4)if (0 === pixels[i]) clearPixels++; const percentage = clearPixels / totalPixels * 100; if (percentage >= minScratchPercentage) { setIsComplete(true); ctx.clearRect(0, 0, canvas.width, canvas.height); startAnimation(); } } }; return /*#__PURE__*/ jsxs(motion.div, { className: cn("relative select-none", className), style: { width, height, cursor: "url('data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSIzMiIgaGVpZ2h0PSIzMiIgdmlld0JveD0iMCAwIDMyIDMyIj4KICA8Y2lyY2xlIGN4PSIxNiIgY3k9IjE2IiByPSIxNSIgc3R5bGU9ImZpbGw6I2ZmZjtzdHJva2U6IzAwMDtzdHJva2Utd2lkdGg6MXB4OyIgLz4KPC9zdmc+'), auto" }, animate: controls, children: [ /*#__PURE__*/ jsx("canvas", { ref: canvasRef, width: width, height: height, className: "absolute top-0 left-0", onMouseDown: handleMouseDown, onTouchStart: handleTouchStart }), children ] }); }; export { Scratcher }; //# sourceMappingURL=scratcher.js.map