minecraft-web/frontend/src/components/SkinViewer.jsx

123 lines
5.6 KiB
JavaScript

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 (
<div ref={containerRef} className="w-full h-full flex items-center justify-center">
<canvas ref={canvasRef} style={{ cursor: 'grab' }} />
</div>
);
};
export default SkinViewer;