maplestory/frontend/src/features/boss-crystal/admin/BossList.jsx

203 lines
7.4 KiB
React
Raw Normal View History

import { useState, useEffect } from 'react'
import { Link } from 'react-router-dom'
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query'
import {
DndContext, DragOverlay, closestCenter, PointerSensor, KeyboardSensor,
useSensor, useSensors,
} from '@dnd-kit/core'
import {
SortableContext, sortableKeyboardCoordinates, useSortable, rectSortingStrategy,
arrayMove,
} from '@dnd-kit/sortable'
import { CSS } from '@dnd-kit/utilities'
import { api } from '../../../api/client'
import Tooltip from '../../../components/Tooltip'
import { DIFFICULTIES, formatMeso, getDifficultyBadgeStyle } from './constants'
function BossCardContent({ boss, dragging = false }) {
return (
<div className={`flex items-stretch rounded-2xl border bg-gradient-to-br from-gray-900/80 to-gray-900/40 ${
dragging
? 'border-emerald-500/60 shadow-2xl shadow-emerald-500/30'
: 'border-white/5'
}`}>
{/* 핸들 자리 */}
<div className="flex items-center px-2 text-gray-700 cursor-grab active:cursor-grabbing">
<svg width="14" height="20" viewBox="0 0 14 20" fill="currentColor">
<circle cx="4" cy="4" r="1.5" />
<circle cx="10" cy="4" r="1.5" />
<circle cx="4" cy="10" r="1.5" />
<circle cx="10" cy="10" r="1.5" />
<circle cx="4" cy="16" r="1.5" />
<circle cx="10" cy="16" r="1.5" />
</svg>
</div>
<div className="flex-1 min-w-0 flex items-start gap-3 p-4 pl-2">
<div className="shrink-0 w-14 h-14 rounded-xl bg-gradient-to-br from-gray-800 to-gray-900 border border-white/5 flex items-center justify-center overflow-hidden">
<img src={boss.image_url || '/default.png'} alt={boss.name} className="w-full h-full object-cover" />
</div>
<div className="flex-1 min-w-0">
<div className="flex items-baseline gap-2">
<h3 className="font-semibold truncate">{boss.name}</h3>
<span className="text-xs text-gray-500 shrink-0">최대 {boss.max_party_size}</span>
</div>
<div className="flex flex-wrap gap-1 mt-2">
{DIFFICULTIES.filter((d) => boss.difficulties?.some((bd) => bd.difficulty === d.key)).map((d) => {
const bd = boss.difficulties.find((x) => x.difficulty === d.key)
return (
<Tooltip key={d.key} text={`${d.label} · ${formatMeso(bd.crystal_price)}`}>
<span
className="text-[10px] font-medium px-1.5 py-0.5 rounded border"
style={getDifficultyBadgeStyle(d.key)}
>
{d.label}
</span>
</Tooltip>
)
})}
</div>
</div>
</div>
</div>
)
}
function SortableBossCard({ boss }) {
const { attributes, listeners, setNodeRef, transform, transition, isDragging, setActivatorNodeRef } = useSortable({
id: boss.id,
transition: { duration: 200, easing: 'cubic-bezier(0.25, 0.1, 0.25, 1)' },
})
const style = {
transform: CSS.Transform.toString(transform),
transition,
}
return (
<div
ref={setNodeRef}
style={style}
className={`relative ${isDragging ? 'opacity-30' : ''}`}
>
{/* 드래그 핸들 (좌측) */}
<button
type="button"
ref={setActivatorNodeRef}
{...attributes}
{...listeners}
className="absolute left-0 top-0 bottom-0 w-8 z-10 cursor-grab active:cursor-grabbing rounded-l-2xl hover:bg-white/5 transition touch-none"
aria-label="순서 변경"
/>
{/* 카드 본체 - Link */}
<Link to={`bosses/${boss.id}`} className="block group hover:[&_h3]:text-emerald-300 [&_h3]:transition">
<BossCardContent boss={boss} />
</Link>
</div>
)
}
export default function BossList() {
const queryClient = useQueryClient()
const { data: bosses = [], isLoading } = useQuery({
queryKey: ['admin', 'boss-crystal', 'bosses'],
queryFn: () => api('/api/admin/boss-crystal/bosses').catch(() => []),
})
const [items, setItems] = useState([])
const [activeId, setActiveId] = useState(null)
useEffect(() => { setItems(bosses) }, [bosses])
const sensors = useSensors(
useSensor(PointerSensor, { activationConstraint: { distance: 5 } }),
useSensor(KeyboardSensor, { coordinateGetter: sortableKeyboardCoordinates })
)
const reorderMutation = useMutation({
mutationFn: (ids) => api('/api/admin/boss-crystal/bosses/reorder', {
method: 'POST',
body: { ids },
}),
onError: (err) => {
alert(err.message)
queryClient.invalidateQueries({ queryKey: ['admin', 'boss-crystal', 'bosses'] })
},
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['boss-crystal'] })
},
})
const handleDragEnd = (event) => {
const { active, over } = event
setActiveId(null)
if (!over || active.id === over.id) return
const oldIdx = items.findIndex((b) => b.id === active.id)
const newIdx = items.findIndex((b) => b.id === over.id)
const next = arrayMove(items, oldIdx, newIdx)
setItems(next)
reorderMutation.mutate(next.map((b) => b.id))
}
const activeBoss = items.find((b) => b.id === activeId)
return (
<div className="space-y-6">
<div className="flex items-end justify-between gap-4 flex-wrap">
<div>
<h2 className="text-lg font-semibold">보스 결정 관리</h2>
<p className="text-sm text-gray-500 mt-0.5">보스 정보 난이도별 결정 가격을 관리합니다</p>
</div>
<Link
to="bosses/new"
className="flex items-center gap-1.5 rounded-lg bg-emerald-600 hover:bg-emerald-500 px-4 py-2 text-sm font-medium transition shadow-lg shadow-emerald-500/20"
>
<span className="text-base leading-none">+</span>
보스 추가
</Link>
</div>
{isLoading ? (
<div className="grid gap-3 sm:grid-cols-2 lg:grid-cols-3">
{Array.from({ length: 4 }).map((_, i) => (
<div key={i} className="h-32 rounded-2xl bg-white/[0.02] animate-pulse" />
))}
</div>
) : items.length === 0 ? (
<div className="rounded-2xl border border-dashed border-white/10 bg-white/[0.02] p-16 text-center">
<div className="text-5xl mb-3 opacity-30"></div>
<p className="text-gray-400 mb-4">등록된 보스가 없습니다</p>
<Link to="bosses/new" className="text-sm text-emerald-400 hover:text-emerald-300 transition">
보스 추가하기
</Link>
</div>
) : (
<DndContext
sensors={sensors}
collisionDetection={closestCenter}
onDragStart={(e) => setActiveId(e.active.id)}
onDragCancel={() => setActiveId(null)}
onDragEnd={handleDragEnd}
>
<SortableContext items={items.map((b) => b.id)} strategy={rectSortingStrategy}>
<div className="grid gap-3 sm:grid-cols-2 lg:grid-cols-3">
{items.map((boss) => (
<SortableBossCard key={boss.id} boss={boss} />
))}
</div>
</SortableContext>
<DragOverlay
dropAnimation={{
duration: 200,
easing: 'cubic-bezier(0.25, 0.1, 0.25, 1)',
}}
>
{activeBoss ? <BossCardContent boss={activeBoss} dragging /> : null}
</DragOverlay>
</DndContext>
)}
</div>
)
}