refactor: 모든 Admin 페이지 API 모듈화 완료

- AdminMemberEdit API 모듈 적용
- AdminScheduleForm API 모듈 적용
- AdminAlbumPhotos API 모듈 적용 (업로드는 SSE 유지)
- api/admin/albums.js에 getAlbumTeasers, deleteAlbumTeaser 추가

총 적용 완료:
- Public 페이지: 7개
- Admin 페이지: 10개 전체 완료
This commit is contained in:
caadiq 2026-01-09 22:26:37 +09:00
parent f006309ef4
commit 8124a1abe1
4 changed files with 49 additions and 97 deletions

View file

@ -48,3 +48,15 @@ export async function deleteAlbumPhoto(albumId, photoId) {
method: "DELETE",
});
}
// 앨범 티저 목록 조회
export async function getAlbumTeasers(albumId) {
return fetchAdminApi(`/api/admin/albums/${albumId}/teasers`);
}
// 앨범 티저 삭제
export async function deleteAlbumTeaser(albumId, teaserId) {
return fetchAdminApi(`/api/admin/albums/${albumId}/teasers/${teaserId}`, {
method: "DELETE",
});
}

View file

@ -8,6 +8,10 @@ import {
Tag, FolderOpen, Save
} from 'lucide-react';
import Toast from '../../../components/Toast';
import * as authApi from '../../../api/admin/auth';
import { getAlbum } from '../../../api/public/albums';
import { getMembers } from '../../../api/public/members';
import * as albumsApi from '../../../api/admin/albums';
function AdminAlbumPhotos() {
const { albumId } = useParams();
@ -145,67 +149,47 @@ function AdminAlbumPhotos() {
useEffect(() => {
//
const token = localStorage.getItem('adminToken');
const userData = localStorage.getItem('adminUser');
if (!token || !userData) {
if (!authApi.hasToken()) {
navigate('/admin');
return;
}
setUser(JSON.parse(userData));
setUser(authApi.getCurrentUser());
fetchAlbumData();
}, [navigate, albumId]);
const fetchAlbumData = async () => {
try {
const token = localStorage.getItem('adminToken');
//
const albumRes = await fetch(`/api/albums/${albumId}`);
if (!albumRes.ok) throw new Error('앨범을 찾을 수 없습니다');
const albumData = await albumRes.json();
const albumData = await getAlbum(albumId);
setAlbum(albumData);
//
const membersRes = await fetch('/api/members');
if (membersRes.ok) {
const membersData = await membersRes.json();
try {
const membersData = await getMembers();
setMembers(membersData);
}
} catch (e) { /* 무시 */ }
//
const photosRes = await fetch(`/api/admin/albums/${albumId}/photos`, {
headers: { 'Authorization': `Bearer ${token}` }
});
let photosData = [];
if (photosRes.ok) {
photosData = await photosRes.json();
try {
photosData = await albumsApi.getAlbumPhotos(albumId);
setPhotos(photosData);
}
} catch (e) { /* 무시 */ }
//
const teasersRes = await fetch(`/api/admin/albums/${albumId}/teasers`, {
headers: { 'Authorization': `Bearer ${token}` }
});
let teasersData = [];
if (teasersRes.ok) {
teasersData = await teasersRes.json();
try {
teasersData = await albumsApi.getAlbumTeasers(albumId);
setTeasers(teasersData);
}
} catch (e) { /* 무시 */ }
// ( + 1)
// ,
//
const maxPhotoOrder = photosData.length > 0
? Math.max(...photosData.map(p => p.sort_order || 0))
: 0;
const maxTeaserOrder = teasersData.length > 0
? Math.max(...teasersData.map(t => t.sort_order || 0))
: 0;
//
setStartNumber(maxPhotoOrder + 1);
setLoading(false);
} catch (error) {
console.error('앨범 로드 오류:', error);
@ -230,8 +214,7 @@ function AdminAlbumPhotos() {
}, [photoType, photos, teasers]);
const handleLogout = () => {
localStorage.removeItem('adminToken');
localStorage.removeItem('adminUser');
authApi.logout();
navigate('/admin');
};
@ -504,7 +487,6 @@ function AdminAlbumPhotos() {
// ( /)
const handleDelete = async () => {
setDeleting(true);
const token = localStorage.getItem('adminToken');
try {
// ID ID
@ -515,24 +497,12 @@ function AdminAlbumPhotos() {
//
for (const photoId of photoIds) {
const res = await fetch(`/api/admin/albums/${albumId}/photos/${photoId}`, {
method: 'DELETE',
headers: { 'Authorization': `Bearer ${token}` }
});
if (!res.ok) {
throw new Error('사진 삭제 실패');
}
await albumsApi.deleteAlbumPhoto(albumId, photoId);
}
//
for (const teaserId of teaserIds) {
const res = await fetch(`/api/admin/albums/${albumId}/teasers/${teaserId}`, {
method: 'DELETE',
headers: { 'Authorization': `Bearer ${token}` }
});
if (!res.ok) {
throw new Error('티저 삭제 실패');
}
await albumsApi.deleteAlbumTeaser(albumId, teaserId);
}
// UI

View file

@ -6,6 +6,8 @@ import {
Home, ChevronRight, ChevronLeft, ChevronDown, User, Instagram, Calendar, Briefcase
} from 'lucide-react';
import Toast from '../../../components/Toast';
import * as authApi from '../../../api/admin/auth';
import * as membersApi from '../../../api/admin/members';
//
function CustomDatePicker({ value, onChange }) {
@ -272,28 +274,18 @@ function AdminMemberEdit() {
useEffect(() => {
//
const token = localStorage.getItem('adminToken');
const userData = localStorage.getItem('adminUser');
if (!token || !userData) {
if (!authApi.hasToken()) {
navigate('/admin');
return;
}
setUser(JSON.parse(userData));
setUser(authApi.getCurrentUser());
fetchMember();
}, [navigate, name]);
const fetchMember = async () => {
try {
const token = localStorage.getItem('adminToken');
const res = await fetch(`/api/admin/members/${encodeURIComponent(name)}`, {
headers: { Authorization: `Bearer ${token}` }
});
if (!res.ok) throw new Error('멤버 조회 실패');
const data = await res.json();
const data = await membersApi.getMember(encodeURIComponent(name));
setFormData({
name: data.name || '',
birth_date: data.birth_date ? data.birth_date.split('T')[0] : '',
@ -311,8 +303,7 @@ function AdminMemberEdit() {
};
const handleLogout = () => {
localStorage.removeItem('adminToken');
localStorage.removeItem('adminUser');
authApi.logout();
navigate('/admin');
};
@ -331,7 +322,6 @@ function AdminMemberEdit() {
setSaving(true);
try {
const token = localStorage.getItem('adminToken');
const formDataToSend = new FormData();
formDataToSend.append('name', formData.name);
@ -344,14 +334,7 @@ function AdminMemberEdit() {
formDataToSend.append('image', imageFile);
}
const res = await fetch(`/api/admin/members/${encodeURIComponent(name)}`, {
method: 'PUT',
headers: { Authorization: `Bearer ${token}` },
body: formDataToSend
});
if (!res.ok) throw new Error('수정 실패');
await membersApi.updateMember(encodeURIComponent(name), formDataToSend);
setToast({ message: '멤버 정보가 수정되었습니다.', type: 'success' });
setTimeout(() => navigate('/admin/members'), 1000);
} catch (error) {

View file

@ -26,6 +26,10 @@ import {
} from "lucide-react";
import Toast from "../../../components/Toast";
import Lightbox from "../../../components/common/Lightbox";
import * as authApi from "../../../api/admin/auth";
import * as categoriesApi from "../../../api/admin/categories";
import * as schedulesApi from "../../../api/admin/schedules";
import { getMembers } from "../../../api/public/members";
// (AdminMemberEdit.jsx )
function CustomDatePicker({ value, onChange, placeholder = "날짜 선택" }) {
const [isOpen, setIsOpen] = useState(false);
@ -840,8 +844,7 @@ function AdminScheduleForm() {
//
const fetchCategories = async () => {
try {
const res = await fetch("/api/admin/schedule-categories");
const data = await res.json();
const data = await categoriesApi.getCategories();
setCategories(data);
//
if (data.length > 0 && !formData.category) {
@ -861,15 +864,12 @@ function AdminScheduleForm() {
}, [toast]);
useEffect(() => {
const token = localStorage.getItem("adminToken");
const userData = localStorage.getItem("adminUser");
if (!token || !userData) {
if (!authApi.hasToken()) {
navigate("/admin");
return;
}
setUser(JSON.parse(userData));
setUser(authApi.getCurrentUser());
fetchMembers();
fetchCategories();
@ -883,18 +883,7 @@ function AdminScheduleForm() {
const fetchSchedule = async () => {
setLoading(true);
try {
const token = localStorage.getItem("adminToken");
const res = await fetch(`/api/admin/schedules/${id}`, {
headers: {
Authorization: `Bearer ${token}`,
},
});
if (!res.ok) {
throw new Error("일정을 찾을 수 없습니다.");
}
const data = await res.json();
const data = await schedulesApi.getSchedule(id);
//
setFormData({
@ -957,8 +946,7 @@ function AdminScheduleForm() {
const fetchMembers = async () => {
try {
const res = await fetch("/api/members");
const data = await res.json();
const data = await getMembers();
setMembers(data.filter((m) => !m.is_former));
} catch (error) {
console.error("멤버 로드 오류:", error);
@ -966,8 +954,7 @@ function AdminScheduleForm() {
};
const handleLogout = () => {
localStorage.removeItem("adminToken");
localStorage.removeItem("adminUser");
authApi.logout();
navigate("/admin");
};