import React, { useState, useEffect, useRef } from 'react'; import { useParams } from 'react-router-dom'; import { Activity, Skull, Heart, LocateFixed, Box, Sword, Clock, Calendar, RefreshCw } from 'lucide-react'; import { motion } from 'framer-motion'; import { io } from 'socket.io-client'; import { formatDate, formatPlayTimeMs } from '../utils/formatters'; // 스티브 3D 스킨 기본 이미지 (로딩 전/실패 시 사용) const STEVE_BODY_BASE64 = ''; // 플레이어 3D 스킨 컴포넌트 - 로딩 중에는 스티브, 로드 완료 시 실제 스킨 표시 const PlayerSkinImage = ({ uuid, playerName }) => { const [src, setSrc] = useState(STEVE_BODY_BASE64); useEffect(() => { const img = new Image(); const realUrl = `https://mc-heads.net/body/${uuid}/80`; img.onload = () => setSrc(realUrl); img.src = realUrl; }, [uuid]); return ( {playerName} ); }; // 아이템/몹 아이콘 기본 이미지 (로딩 실패 시 사용) const DEFAULT_ICON = ''; // 거리 포맷 함수 - 1만 이상이면 "1.2만m" 형식으로 표시 const formatDistance = (meters) => { if (meters >= 100000000) { // 1억 이상 return `${(meters / 100000000).toFixed(1)}억m`; } else if (meters >= 10000000) { // 천만 이상 return `${(meters / 10000000).toFixed(1)}천만m`; } else if (meters >= 10000) { // 만 이상 return `${(meters / 10000).toFixed(1)}만m`; } return `${meters.toLocaleString()}m`; }; // 플레이어 통계 페이지 const PlayerStatsPage = ({ isMobile = false }) => { const { uuid } = useParams(); const [playerName, setPlayerName] = useState(''); const [playerDetail, setPlayerDetail] = useState(null); const [stats, setStats] = useState(null); const [translations, setTranslations] = useState({}); const [icons, setIcons] = useState({}); // 아이콘 캐시 const [loading, setLoading] = useState(true); const socketRef = useRef(null); // 번역 및 아이콘 데이터 로드 useEffect(() => { Promise.all([ fetch('/api/translations').then(res => res.json()), fetch('/api/icons').then(res => res.json()) ]) .then(([transData, iconsData]) => { setTranslations(transData); setIcons(iconsData); }) .catch(err => console.error('데이터 로드 실패:', err)); }, []); useEffect(() => { const socket = io('/', { path: '/socket.io', transports: ['websocket', 'polling'] }); socketRef.current = socket; socket.on('player_stats', (data) => { setStats(data); setLoading(false); }); socket.on('player_detail', (data) => { if (data) { setPlayerName(data.name); setPlayerDetail(data); } }); socket.emit('get_player', uuid); socket.emit('get_player_stats', uuid); const interval = setInterval(() => { socket.emit('get_player', uuid); socket.emit('get_player_stats', uuid); }, 1000); return () => { clearInterval(interval); socket.disconnect(); }; }, [uuid]); const firstPlayed = formatDate(playerDetail?.firstJoin); const lastPlayed = formatDate(playerDetail?.lastLeave > 0 ? playerDetail?.lastLeave : playerDetail?.lastJoin); // 번역 함수 (DB에 없으면 영어 그대로) const translate = (id) => translations[id] || id.replace(/_/g, ' '); // 아이템 통계 정렬 (이름순) const sortedItems = stats?.items ? Object.entries(stats.items) .map(([id, stat]) => ({ id, ...stat, total: (stat.mined || 0) + (stat.used || 0) + (stat.pickedUp || 0) + (stat.crafted || 0) })) .sort((a, b) => translate(a.id).localeCompare(translate(b.id), 'ko')) : []; // 몹 통계 정렬 (이름순) const sortedMobs = stats?.mobs ? Object.entries(stats.mobs) .map(([id, stat]) => ({ id, ...stat, total: (stat.killed || 0) + (stat.killedBy || 0) })) .sort((a, b) => translate(a.id).localeCompare(translate(b.id), 'ko')) : []; return (
{/* 헤더 */}

{playerName || '로딩중...'}

플레이어 통계

{loading ? (
) : !stats ? (

통계를 불러올 수 없습니다. 플레이어가 접속 중이어야 합니다.

) : (
{/* 플레이어 정보 */} {playerDetail && (

플레이어 정보

{/* 현재 세션 플레이타임 (접속 중일 때만) */} {playerDetail.isOnline && (
현재 세션 플레이타임
{formatPlayTimeMs(playerDetail.currentSessionMs)}
)} {/* 누적 플레이타임 */}
누적 플레이타임
{formatPlayTimeMs(playerDetail.totalPlayTimeMs + (playerDetail.isOnline ? playerDetail.currentSessionMs : 0))}
{/* 첫 접속 */}
첫 접속
{firstPlayed.date}
{firstPlayed.time}
{/* 마지막 접속 */}
마지막 접속
{lastPlayed.date}
{lastPlayed.time}
)} {/* 일반 통계 */}

일반 통계

{/* 이동 거리 */}

이동 거리

걸은 거리

{isMobile ? formatDistance(stats.general.distanceWalked) : `${stats.general.distanceWalked.toLocaleString()}m`}

비행 거리

{isMobile ? formatDistance(stats.general.distanceFlown) : `${stats.general.distanceFlown.toLocaleString()}m`}

수영 거리

{isMobile ? formatDistance(stats.general.distanceSwum) : `${stats.general.distanceSwum.toLocaleString()}m`}

{/* 아이템 통계 - 4열 그리드 + 내부 스크롤 */} {sortedItems.length > 0 && (

아이템 통계 ({sortedItems.length}개)

{sortedItems.map((item) => ( ))}
)} {/* 몹 통계 - 4열 그리드 + 내부 스크롤 */} {sortedMobs.length > 0 && (

몹 통계 ({sortedMobs.length}마리)

{sortedMobs.map((mob) => ( ))}
)} {sortedItems.length === 0 && sortedMobs.length === 0 && (

아직 기록된 통계가 없습니다.

)}
)}
); }; // 통계 카드 컴포넌트 const StatCard = ({ icon: Icon, label, value, color }) => (

{label}

{typeof value === 'number' ? value.toLocaleString() : value}

); // 아이템 통계 행 컴포넌트 (온디맨드 아이콘 지원) const ItemStatRow = ({ item, translate, icons }) => { const [iconSrc, setIconSrc] = useState(icons[item.id] || DEFAULT_ICON); const [loading, setLoading] = useState(!icons[item.id]); // 아이콘이 없으면 온디맨드로 가져오기 useEffect(() => { if (!icons[item.id]) { fetch(`/api/icon/item/${item.id}`) .then(res => res.json()) .then(data => { if (data.icon) { setIconSrc(data.icon); } }) .catch(() => {}) .finally(() => setLoading(false)); } }, [item.id, icons]); return (
{item.id} { e.target.src = DEFAULT_ICON; }} />

{translate(item.id)}

{item.mined > 0 && 채굴 {item.mined}} {item.used > 0 && 사용 {item.used}} {item.pickedUp > 0 && 획득 {item.pickedUp}} {item.crafted > 0 && 제작 {item.crafted}}
); }; // 몹 통계 행 컴포넌트 (온디맨드 아이콘 지원) const MobStatRow = ({ mob, translate, icons }) => { const [iconSrc, setIconSrc] = useState(icons[mob.id] || DEFAULT_ICON); const [loading, setLoading] = useState(!icons[mob.id]); // 아이콘이 없으면 온디맨드로 가져오기 useEffect(() => { if (!icons[mob.id]) { fetch(`/api/icon/entity/${mob.id}`) .then(res => res.json()) .then(data => { if (data.icon) { setIconSrc(data.icon); } }) .catch(() => {}) .finally(() => setLoading(false)); } }, [mob.id, icons]); return (
{mob.id} { e.target.src = DEFAULT_ICON; }} />

{translate(mob.id)}

{mob.killed > 0 && 처치 {mob.killed}} {mob.killedBy > 0 && 죽음 {mob.killedBy}}
); }; export default PlayerStatsPage;