- /api/character/symbols 엔드포인트: Nexon API의 symbol-equipment를 (type, region, level, growth, force) 구조로 정제 후 반환 - 프론트: useQueries로 각 캐릭터 심볼 자동 로드, 새로고침마다 갱신, syncCharacterSymbols로 store의 progress에 병합 - equipped 판정을 store 기반으로 전환 - 남은 심볼/필요 메소/체납 메소 실제 계산, 만렙 시 '-' 표시 - 성장치 라벨 현재 레벨 기준 표시, 만렙 시 MAX/amber 색상 + 퍼센트 숨김 - 일퀘/주간퀘/추가 심볼 비활성화 및 완료 토글 숨김 (만렙) - 하단 요약 누적 체납/남은 필요 메소 실제 합산, 라벨 색상 통일 - 메소 값 호버 시 '억/만' 한글 축약 툴팁 - Select 비활성 상태에서 금지 커서 제거 Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
73 lines
2.5 KiB
JavaScript
73 lines
2.5 KiB
JavaScript
import { Router } from 'express';
|
|
import axios from 'axios';
|
|
|
|
const router = Router();
|
|
const NEXON_API_BASE = 'https://open.api.nexon.com';
|
|
|
|
// 캐릭터 닉네임으로 정보 조회
|
|
router.get('/search', async (req, res) => {
|
|
const { name } = req.query;
|
|
if (!name?.trim()) return res.status(400).json({ error: '캐릭터 닉네임을 입력해주세요' });
|
|
|
|
try {
|
|
// 1) ocid 조회
|
|
const { data: idData } = await axios.get(`${NEXON_API_BASE}/maplestory/v1/id`, {
|
|
params: { character_name: name.trim() },
|
|
headers: { 'x-nxopen-api-key': process.env.NEXON_API_KEY },
|
|
});
|
|
|
|
// 2) basic 조회
|
|
const { data: basic } = await axios.get(`${NEXON_API_BASE}/maplestory/v1/character/basic`, {
|
|
params: { ocid: idData.ocid },
|
|
headers: { 'x-nxopen-api-key': process.env.NEXON_API_KEY },
|
|
});
|
|
|
|
res.json({
|
|
ocid: idData.ocid,
|
|
character_name: basic.character_name,
|
|
world_name: basic.world_name,
|
|
job_name: basic.character_class,
|
|
character_level: basic.character_level,
|
|
character_image: basic.character_image,
|
|
});
|
|
} catch (err) {
|
|
if (err.response?.status === 400) {
|
|
return res.status(404).json({ error: '존재하지 않는 캐릭터입니다' });
|
|
}
|
|
console.error('캐릭터 조회 오류:', err.response?.data || err.message);
|
|
res.status(500).json({ error: '캐릭터 조회 실패' });
|
|
}
|
|
});
|
|
|
|
// OCID로 장착 심볼 조회
|
|
router.get('/symbols', async (req, res) => {
|
|
const { ocid } = req.query;
|
|
if (!ocid) return res.status(400).json({ error: 'ocid가 필요합니다' });
|
|
|
|
try {
|
|
const { data } = await axios.get(`${NEXON_API_BASE}/maplestory/v1/character/symbol-equipment`, {
|
|
params: { ocid },
|
|
headers: { 'x-nxopen-api-key': process.env.NEXON_API_KEY },
|
|
});
|
|
|
|
const parsed = (data.symbol || []).map((s) => {
|
|
const [prefix, region] = (s.symbol_name || '').split(' : ').map((t) => t.trim());
|
|
const type = prefix?.replace(/심볼$/, '').trim(); // '아케인심볼' → '아케인'
|
|
return {
|
|
type,
|
|
region,
|
|
level: Number(s.symbol_level) || 0,
|
|
force: Number(s.symbol_force) || 0,
|
|
growth_count: Number(s.symbol_growth_count) || 0,
|
|
require_growth_count: Number(s.symbol_require_growth_count) || 0,
|
|
};
|
|
});
|
|
|
|
res.json({ ocid, character_class: data.character_class, symbols: parsed });
|
|
} catch (err) {
|
|
console.error('심볼 조회 오류:', err.response?.data || err.message);
|
|
res.status(500).json({ error: '심볼 조회 실패' });
|
|
}
|
|
});
|
|
|
|
export default router;
|