레이아웃: - 풀스크린 모드 컨텍스트 (BossCrystal 페이지에서 푸터 숨김 + viewport 고정) - 캐릭터 패널: 자연 높이 + viewport 한도 + 내부 목록 스크롤 - 보스 패널: 헤더 고정 + 목록 내부 스크롤 - 커스텀 스크롤바 (전역) 캐릭터 패널: - framer-motion Reorder로 드래그앤드롭 정렬 - 가로 캐릭터 행 + 6x2 보스 그리드 + 난이도 영문 첫글자 뱃지 - 총 수익에 ResizeObserver 기반 자동 폰트 fit - 캐릭터 삭제 시 첫번째 자동 선택, 입력 재개 시 에러 메시지 자동 제거 기능: - 공개 보스/캐릭터 API 추가 - API 키 라이브 키로 변경 Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
33 lines
1 KiB
JavaScript
33 lines
1 KiB
JavaScript
import { Router } from 'express';
|
|
import { BossCrystalBoss, BossCrystalBossDifficulty } from '../models/index.js';
|
|
import { getPublicUrl } from '../lib/s3.js';
|
|
|
|
const router = Router();
|
|
|
|
// 공개 보스 목록
|
|
router.get('/bosses', async (_req, res) => {
|
|
try {
|
|
const bosses = await BossCrystalBoss.findAll({
|
|
order: [['sort_order', 'ASC'], ['id', 'ASC']],
|
|
include: [{ model: BossCrystalBossDifficulty, as: 'difficulties' }],
|
|
});
|
|
res.json(bosses.map((b) => {
|
|
const json = b.toJSON();
|
|
return {
|
|
id: json.id,
|
|
name: json.name,
|
|
image_url: json.image_path ? getPublicUrl(json.image_path) : null,
|
|
max_party_size: json.max_party_size,
|
|
difficulties: (json.difficulties || []).map((d) => ({
|
|
difficulty: d.difficulty,
|
|
crystal_price: Number(d.crystal_price),
|
|
})),
|
|
};
|
|
}));
|
|
} catch (err) {
|
|
console.error('보스 목록 조회 오류:', err.message);
|
|
res.status(500).json({ error: '보스 목록 조회 실패' });
|
|
}
|
|
});
|
|
|
|
export default router;
|