/** * 관리자 활동 로그 페이지 */ import { useState, useEffect } from 'react'; import { Link } from 'react-router-dom'; import { useQuery, keepPreviousData } from '@tanstack/react-query'; import { motion, AnimatePresence } from 'framer-motion'; import { Home, ChevronRight, Search, ChevronLeft, ChevronDown, User, Bot, ScrollText, X, Loader2, } from 'lucide-react'; import { AdminLayout, DatePicker } from '@/components/pc/admin'; import { useAdminAuth } from '@/hooks/pc/admin'; import { adminLogApi } from '@/api/admin'; // 카테고리 목록 const CATEGORIES = [ { value: 'album', label: '앨범' }, { value: 'schedule', label: '일정' }, { value: 'member', label: '멤버' }, { value: 'bot', label: '봇' }, { value: 'category', label: '카테고리' }, { value: 'dict', label: '사전' }, { value: 'concert', label: '콘서트' }, { value: 'sync', label: '동기화' }, ]; // 액션 뱃지 색상 const ACTION_STYLES = { create: 'bg-emerald-100 text-emerald-700', upload: 'bg-emerald-100 text-emerald-700', update: 'bg-blue-100 text-blue-700', delete: 'bg-red-100 text-red-700', sync_complete: 'bg-purple-100 text-purple-700', error: 'bg-red-100 text-red-700', start: 'bg-amber-100 text-amber-700', stop: 'bg-amber-100 text-amber-700', }; // 액션 한글 라벨 const ACTION_LABELS = { create: '생성', upload: '업로드', update: '수정', delete: '삭제', sync_complete: '동기화', error: '에러', start: '시작', stop: '정지', }; const ITEMS_PER_PAGE = 15; function Logs() { const { user } = useAdminAuth(); // 필터 상태 const [searchQuery, setSearchQuery] = useState(''); const [selectedCategories, setSelectedCategories] = useState([]); const [actorFilter, setActorFilter] = useState('all'); // all, admin, bot const [dateFrom, setDateFrom] = useState(''); const [dateTo, setDateTo] = useState(''); const [currentPage, setCurrentPage] = useState(1); const [actorDropdownOpen, setActorDropdownOpen] = useState(false); // 검색어 디바운스 const [debouncedSearch, setDebouncedSearch] = useState(''); useEffect(() => { const timer = setTimeout(() => setDebouncedSearch(searchQuery), 300); return () => clearTimeout(timer); }, [searchQuery]); // API 호출 const { data, isLoading } = useQuery({ queryKey: ['admin', 'logs', { page: currentPage, category: selectedCategories.join(','), actor: actorFilter === 'all' ? '' : actorFilter, search: debouncedSearch, from: dateFrom, to: dateTo }], queryFn: () => adminLogApi.getLogs({ page: currentPage, limit: ITEMS_PER_PAGE, category: selectedCategories.join(',') || undefined, actor: actorFilter === 'all' ? undefined : actorFilter, search: debouncedSearch || undefined, from: dateFrom || undefined, to: dateTo || undefined, }), placeholderData: keepPreviousData, }); const logs = data?.logs || []; const total = data?.total || 0; const totalPages = data?.totalPages || 0; // 카테고리 토글 const toggleCategory = (cat) => { setSelectedCategories((prev) => prev.includes(cat) ? prev.filter((c) => c !== cat) : [...prev, cat] ); setCurrentPage(1); }; // 날짜/시간 포맷 const formatDateTime = (dateStr) => { const date = new Date(dateStr); const y = date.getFullYear(); const month = String(date.getMonth() + 1).padStart(2, '0'); const day = String(date.getDate()).padStart(2, '0'); const hours = String(date.getHours()).padStart(2, '0'); const minutes = String(date.getMinutes()).padStart(2, '0'); return `${y}.${month}.${day} ${hours}:${minutes}`; }; // 행위자 아이콘 const renderActorBadge = (actor) => { if (actor === 'admin') { return ( 관리자 ); } return ( {actor} ); }; // 필터 초기화 const clearFilters = () => { setSearchQuery(''); setSelectedCategories([]); setActorFilter('all'); setDateFrom(''); setDateTo(''); setCurrentPage(1); }; const hasActiveFilters = searchQuery || selectedCategories.length > 0 || actorFilter !== 'all' || dateFrom || dateTo; return (
{/* 브레드크럼 */}
활동 로그
{/* 타이틀 */}

활동 로그

모든 관리자 및 봇 활동 기록을 확인합니다

{/* 필터 영역 */}
{/* 상단: 검색 + 행위자 + 날짜 */}
{/* 검색 */}
{ setSearchQuery(e.target.value); setCurrentPage(1); }} placeholder="로그 검색..." className="w-full pl-10 pr-4 py-2 border border-gray-200 rounded-lg focus:outline-none focus:ring-2 focus:ring-primary focus:border-transparent text-sm" />
{/* 행위자 드롭다운 */}
{actorDropdownOpen && ( <>
setActorDropdownOpen(false)} /> {[ { value: 'all', label: '전체 행위자' }, { value: 'admin', label: '관리자' }, { value: 'bot', label: '봇' }, ].map((opt) => ( ))} )}
{/* 날짜 필터 */}
{ setDateFrom(v); setCurrentPage(1); }} placeholder="시작일" />
~
{ setDateTo(v); setCurrentPage(1); }} placeholder="종료일" />
{/* 필터 초기화 */} {hasActiveFilters && ( )}
{/* 하단: 카테고리 칩 */}
카테고리 {CATEGORIES.map((cat) => ( ))}
{/* 결과 개수 */}

{total}개의 로그

{/* 로그 테이블 */} {logs.map((log, index) => ( ))}
시간 행위자 액션 카테고리 내용
{formatDateTime(log.created_at)} {renderActorBadge(log.actor)} {ACTION_LABELS[log.action] || log.action} {CATEGORIES.find((c) => c.value === log.category)?.label || log.category} {log.summary}
{isLoading && logs.length === 0 && (

로그를 불러오는 중...

)} {!isLoading && logs.length === 0 && (

{hasActiveFilters ? '검색 결과가 없습니다.' : '활동 로그가 없습니다.'}

)}
{/* 페이지네이션 */} {totalPages > 1 && (
{Array.from({ length: totalPages }, (_, i) => i + 1) .filter((page) => { // 페이지가 많을 때 현재 페이지 주변만 표시 if (totalPages <= 7) return true; if (page === 1 || page === totalPages) return true; if (Math.abs(page - currentPage) <= 2) return true; return false; }) .reduce((acc, page, i, arr) => { // 생략 부호(...) 추가 if (i > 0 && page - arr[i - 1] > 1) { acc.push({ type: 'ellipsis', key: `e-${page}` }); } acc.push({ type: 'page', value: page, key: page }); return acc; }, []) .map((item) => item.type === 'ellipsis' ? ( ... ) : ( ) )}
)}
); } export default Logs;