import React, { useEffect, useRef } from 'react'; import * as skinview3d from 'skinview3d'; const STEVE_SKIN_BASE64 = ""; // 3D 스킨 뷰어 컴포넌트 const SkinViewer = ({ skinUrl }) => { const containerRef = useRef(null); const canvasRef = useRef(null); const viewerRef = useRef(null); const isDraggingRef = useRef(false); const lastXRef = useRef(0); useEffect(() => { if (!canvasRef.current || !containerRef.current) return; const viewer = new skinview3d.SkinViewer({ canvas: canvasRef.current, width: containerRef.current.clientWidth, height: containerRef.current.clientHeight, skin: STEVE_SKIN_BASE64, }); // 설정 viewer.fov = 70; viewer.zoom = 0.9; viewer.autoRotate = false; viewer.animation = new skinview3d.IdleAnimation(); viewer.playerObject.rotation.y = Math.PI / 6; viewer.playerObject.position.y = -0.5; viewerRef.current = viewer; // 수동 드래그 회전 구현 const canvas = canvasRef.current; const handleMouseDown = (e) => { isDraggingRef.current = true; lastXRef.current = e.clientX; canvas.style.cursor = 'grabbing'; }; const handleMouseMove = (e) => { if (!isDraggingRef.current || !viewerRef.current) return; const deltaX = e.clientX - lastXRef.current; viewerRef.current.playerObject.rotation.y += deltaX * 0.01; lastXRef.current = e.clientX; }; const handleMouseUp = () => { isDraggingRef.current = false; canvas.style.cursor = 'grab'; }; // 터치 이벤트 지원 const handleTouchStart = (e) => { if (e.touches.length === 1) { isDraggingRef.current = true; lastXRef.current = e.touches[0].clientX; } }; const handleTouchMove = (e) => { if (!isDraggingRef.current || !viewerRef.current || e.touches.length !== 1) return; e.preventDefault(); const deltaX = e.touches[0].clientX - lastXRef.current; viewerRef.current.playerObject.rotation.y += deltaX * 0.01; lastXRef.current = e.touches[0].clientX; }; const handleTouchEnd = () => { isDraggingRef.current = false; }; canvas.addEventListener('mousedown', handleMouseDown); window.addEventListener('mousemove', handleMouseMove); window.addEventListener('mouseup', handleMouseUp); canvas.addEventListener('touchstart', handleTouchStart, { passive: false }); canvas.addEventListener('touchmove', handleTouchMove, { passive: false }); canvas.addEventListener('touchend', handleTouchEnd); // Resize Observer for responsive sizing const resizeObserver = new ResizeObserver(entries => { for (let entry of entries) { const { width, height } = entry.contentRect; viewer.setSize(width, height); } }); resizeObserver.observe(containerRef.current); return () => { resizeObserver.disconnect(); canvas.removeEventListener('mousedown', handleMouseDown); window.removeEventListener('mousemove', handleMouseMove); window.removeEventListener('mouseup', handleMouseUp); canvas.removeEventListener('touchstart', handleTouchStart); canvas.removeEventListener('touchmove', handleTouchMove); canvas.removeEventListener('touchend', handleTouchEnd); viewer.dispose(); }; }, []); useEffect(() => { if (viewerRef.current) { // 로딩 상태를 보여주기 위해 스티브 스킨으로 초기화 (동기/즉시) viewerRef.current.loadSkin(STEVE_SKIN_BASE64); const targetSkin = skinUrl || STEVE_SKIN_BASE64; if (targetSkin !== STEVE_SKIN_BASE64) { // 그 다음 실제 스킨 로드 viewerRef.current.loadSkin(targetSkin); } } }, [skinUrl]); return (
); }; export default SkinViewer;