- page.tsx 2817 LOC → 107 LOC (탭 라우팅 + CacheFlushButton만)
- _panels/ChannelsPanel.tsx (222), VideosPanel.tsx (1282),
RestaurantsPanel.tsx (675), UsersPanel.tsx (383), DaemonPanel.tsx (231)
- localStorage.getItem("tasteby_token") 10곳 → getAdminToken() (lib/admin-utils)
- 패널 내부 로직/state 그대로 (점진적 접근, 회귀 위험 최소화)
후속 분리:
- (신규) SSE 파싱 6곳을 consumeSseStream으로 통일
설계서: docs/design/329-admin-split/README.md
Refs: #329 (Developer)
677 lines
31 KiB
TypeScript
677 lines
31 KiB
TypeScript
"use client";
|
||
import { getAdminToken } from "@/lib/admin-utils";
|
||
|
||
import { useCallback, useEffect, useState } from "react";
|
||
import { api } from "@/lib/api";
|
||
import type { Restaurant, VideoLink } from "@/lib/api";
|
||
|
||
// #329 — admin/page.tsx에서 추출
|
||
export function RestaurantsPanel({ isAdmin }: { isAdmin: boolean }) {
|
||
const [restaurants, setRestaurants] = useState<Restaurant[]>([]);
|
||
const [loading, setLoading] = useState(false);
|
||
const [page, setPage] = useState(0);
|
||
const [nameSearch, setNameSearch] = useState("");
|
||
const [selected, setSelected] = useState<Restaurant | null>(null);
|
||
const [editForm, setEditForm] = useState<Record<string, string>>({});
|
||
const [saving, setSaving] = useState(false);
|
||
const [videos, setVideos] = useState<VideoLink[]>([]);
|
||
const [tablingSearching, setTablingSearching] = useState(false);
|
||
const [bulkTabling, setBulkTabling] = useState(false);
|
||
const [bulkTablingProgress, setBulkTablingProgress] = useState({ current: 0, total: 0, name: "", linked: 0, notFound: 0 });
|
||
const [catchtableSearching, setCatchtableSearching] = useState(false);
|
||
const [bulkCatchtable, setBulkCatchtable] = useState(false);
|
||
const [bulkCatchtableProgress, setBulkCatchtableProgress] = useState({ current: 0, total: 0, name: "", linked: 0, notFound: 0 });
|
||
type RestSortKey = "name" | "region" | "cuisine_type" | "price_range" | "rating" | "business_status";
|
||
const [sortKey, setSortKey] = useState<RestSortKey>("name");
|
||
const [sortAsc, setSortAsc] = useState(true);
|
||
const perPage = 20;
|
||
|
||
const handleSort = (key: RestSortKey) => {
|
||
if (sortKey === key) setSortAsc(!sortAsc);
|
||
else { setSortKey(key); setSortAsc(true); }
|
||
};
|
||
const sortIcon = (key: RestSortKey) => sortKey !== key ? " ↕" : sortAsc ? " ↑" : " ↓";
|
||
|
||
const load = useCallback(() => {
|
||
setLoading(true);
|
||
api
|
||
.getRestaurants({ limit: 500 })
|
||
.then(setRestaurants)
|
||
.catch(console.error)
|
||
.finally(() => setLoading(false));
|
||
}, []);
|
||
|
||
useEffect(() => { load(); }, [load]);
|
||
|
||
// #322/#323 LLM 검증 UI
|
||
const [verifyPending, setVerifyPending] = useState<number | null>(null);
|
||
const [verifying, setVerifying] = useState(false);
|
||
const [verifyResult, setVerifyResult] = useState<string | null>(null);
|
||
|
||
const loadVerifyPending = useCallback(() => {
|
||
api.getVerifyPending().then((r) => setVerifyPending(r.pending)).catch(() => setVerifyPending(null));
|
||
}, []);
|
||
useEffect(() => { loadVerifyPending(); }, [loadVerifyPending]);
|
||
|
||
const handleVerifyAll = async () => {
|
||
if (!isAdmin) return;
|
||
if (!confirm(`미검증 식당 ${verifyPending ?? "?"}건을 LLM으로 일괄 검증합니다.\n잘못 등록된 데이터/프랜차이즈를 자동으로 숨김 처리합니다.\n진행할까요?`)) return;
|
||
setVerifying(true);
|
||
setVerifyResult(null);
|
||
try {
|
||
const r = await api.verifyAll(10);
|
||
setVerifyResult(`${r.processed}건 검증 완료`);
|
||
loadVerifyPending();
|
||
load();
|
||
} catch (e) {
|
||
setVerifyResult(`실패: ${e instanceof Error ? e.message : String(e)}`);
|
||
} finally {
|
||
setVerifying(false);
|
||
}
|
||
};
|
||
|
||
const handleToggleHidden = async (r: Restaurant) => {
|
||
if (!isAdmin) return;
|
||
const becomingHidden = !r.hidden;
|
||
const reason = becomingHidden ? (prompt("숨김 사유(선택)", "manual") ?? "manual") : "";
|
||
try {
|
||
await api.setRestaurantHidden(r.id, becomingHidden, reason || "manual");
|
||
load();
|
||
} catch (e) {
|
||
alert(`실패: ${e instanceof Error ? e.message : String(e)}`);
|
||
}
|
||
};
|
||
|
||
const filtered = restaurants.filter((r) => {
|
||
if (nameSearch && !r.name.toLowerCase().includes(nameSearch.toLowerCase())) return false;
|
||
return true;
|
||
});
|
||
|
||
const sorted = [...filtered].sort((a, b) => {
|
||
let av: string | number = a[sortKey] ?? "";
|
||
let bv: string | number = b[sortKey] ?? "";
|
||
if (sortKey === "rating") {
|
||
av = a.rating ?? 0;
|
||
bv = b.rating ?? 0;
|
||
}
|
||
const cmp = av < bv ? -1 : av > bv ? 1 : 0;
|
||
return sortAsc ? cmp : -cmp;
|
||
});
|
||
|
||
const totalPages = Math.max(1, Math.ceil(sorted.length / perPage));
|
||
const paged = sorted.slice(page * perPage, (page + 1) * perPage);
|
||
|
||
const handleSelect = (r: Restaurant) => {
|
||
if (selected?.id === r.id) {
|
||
setSelected(null);
|
||
setVideos([]);
|
||
return;
|
||
}
|
||
setSelected(r);
|
||
api.getRestaurantVideos(r.id).then(setVideos).catch(() => setVideos([]));
|
||
setEditForm({
|
||
name: r.name || "",
|
||
address: r.address || "",
|
||
region: r.region || "",
|
||
cuisine_type: r.cuisine_type || "",
|
||
price_range: r.price_range || "",
|
||
phone: r.phone || "",
|
||
website: r.website || "",
|
||
latitude: r.latitude != null ? String(r.latitude) : "",
|
||
longitude: r.longitude != null ? String(r.longitude) : "",
|
||
});
|
||
};
|
||
|
||
const handleSave = async () => {
|
||
if (!selected) return;
|
||
setSaving(true);
|
||
try {
|
||
const data: Record<string, unknown> = { ...editForm };
|
||
if (editForm.latitude) data.latitude = parseFloat(editForm.latitude);
|
||
else data.latitude = null;
|
||
if (editForm.longitude) data.longitude = parseFloat(editForm.longitude);
|
||
else data.longitude = null;
|
||
await api.updateRestaurant(selected.id, data as Partial<Restaurant>);
|
||
load();
|
||
setSelected(null);
|
||
} catch (e) {
|
||
alert("저장 실패: " + (e instanceof Error ? e.message : String(e)));
|
||
} finally {
|
||
setSaving(false);
|
||
}
|
||
};
|
||
|
||
const handleDelete = async () => {
|
||
if (!selected) return;
|
||
if (!confirm(`"${selected.name}" 식당을 삭제하시겠습니까?\n연결된 영상 매핑, 리뷰, 벡터도 함께 삭제됩니다.`)) return;
|
||
try {
|
||
await api.deleteRestaurant(selected.id);
|
||
setSelected(null);
|
||
load();
|
||
} catch {
|
||
alert("삭제 실패");
|
||
}
|
||
};
|
||
|
||
return (
|
||
<div>
|
||
<div className="flex items-center gap-3 mb-4">
|
||
<div className="flex">
|
||
<input
|
||
type="text"
|
||
placeholder="식당 이름 검색..."
|
||
value={nameSearch}
|
||
onChange={(e) => { setNameSearch(e.target.value); setPage(0); }}
|
||
onKeyDown={(e) => e.key === "Escape" && setNameSearch("")}
|
||
className="border border-r-0 rounded-l px-3 py-2 text-sm w-48 bg-surface text-gray-900"
|
||
/>
|
||
{nameSearch ? (
|
||
<button
|
||
onClick={() => { setNameSearch(""); setPage(0); }}
|
||
className="border rounded-r px-2 py-2 text-sm text-gray-400 hover:text-gray-600 hover:bg-gray-100"
|
||
>
|
||
✕
|
||
</button>
|
||
) : (
|
||
<button
|
||
onClick={() => load()}
|
||
className="border rounded-r px-2 py-2 text-sm text-gray-400 hover:text-gray-600 hover:bg-gray-100"
|
||
title="새로고침"
|
||
>
|
||
↻
|
||
</button>
|
||
)}
|
||
</div>
|
||
{isAdmin && (<>
|
||
{/* #322/#323 — LLM 검증 */}
|
||
<div className="flex items-center gap-2 text-xs text-gray-600 dark:text-gray-400">
|
||
<span>미검증 {verifyPending ?? "?"}건</span>
|
||
<button
|
||
onClick={handleVerifyAll}
|
||
disabled={verifying || verifyPending === 0}
|
||
className="px-3 py-1 text-xs rounded bg-amber-500 hover:bg-amber-600 text-white disabled:opacity-50 disabled:cursor-not-allowed"
|
||
>
|
||
{verifying ? "검증 중..." : "LLM 검증"}
|
||
</button>
|
||
{verifyResult && <span className="text-amber-600">{verifyResult}</span>}
|
||
</div>
|
||
<button
|
||
onClick={async () => {
|
||
const pending = await fetch(`/api/restaurants/tabling-pending`, {
|
||
headers: { Authorization: `Bearer ${getAdminToken()}` },
|
||
}).then(r => r.json());
|
||
if (pending.count === 0) { alert("테이블링 미연결 식당이 없습니다"); return; }
|
||
if (!confirm(`테이블링 미연결 식당 ${pending.count}개를 벌크 검색합니다.\n식당당 5~15초 소요됩니다. 진행할까요?`)) return;
|
||
setBulkTabling(true);
|
||
setBulkTablingProgress({ current: 0, total: pending.count, name: "", linked: 0, notFound: 0 });
|
||
try {
|
||
const res = await fetch("/api/restaurants/bulk-tabling", {
|
||
method: "POST",
|
||
headers: { Authorization: `Bearer ${getAdminToken()}` },
|
||
});
|
||
const reader = res.body!.getReader();
|
||
const decoder = new TextDecoder();
|
||
let buf = "";
|
||
while (true) {
|
||
const { done, value } = await reader.read();
|
||
if (done) break;
|
||
buf += decoder.decode(value, { stream: true });
|
||
const lines = buf.split("\n");
|
||
buf = lines.pop() || "";
|
||
for (const line of lines) {
|
||
const m = line.match(/^data:(.+)$/);
|
||
if (!m) continue;
|
||
const evt = JSON.parse(m[1]);
|
||
if (evt.type === "processing" || evt.type === "done" || evt.type === "notfound" || evt.type === "error") {
|
||
setBulkTablingProgress(p => ({
|
||
...p, current: evt.current, total: evt.total || p.total, name: evt.name,
|
||
linked: evt.type === "done" ? p.linked + 1 : p.linked,
|
||
notFound: (evt.type === "notfound" || evt.type === "error") ? p.notFound + 1 : p.notFound,
|
||
}));
|
||
} else if (evt.type === "complete") {
|
||
alert(`완료! 연결: ${evt.linked}개, 미발견: ${evt.notFound}개`);
|
||
}
|
||
}
|
||
}
|
||
} catch (e) { alert("벌크 테이블링 실패: " + (e instanceof Error ? e.message : String(e))); }
|
||
finally { setBulkTabling(false); load(); }
|
||
}}
|
||
disabled={bulkTabling}
|
||
className="px-3 py-1.5 text-xs bg-brand-500 text-white rounded hover:bg-brand-600 disabled:opacity-50"
|
||
>
|
||
{bulkTabling ? `테이블링 검색 중 (${bulkTablingProgress.current}/${bulkTablingProgress.total})` : "벌크 테이블링 연결"}
|
||
</button>
|
||
<button
|
||
onClick={async () => {
|
||
if (!confirm("테이블링 매핑을 전부 초기화하시겠습니까?")) return;
|
||
try {
|
||
await fetch("/api/restaurants/reset-tabling", {
|
||
method: "DELETE",
|
||
headers: { Authorization: `Bearer ${getAdminToken()}` },
|
||
});
|
||
alert("테이블링 매핑 초기화 완료");
|
||
load();
|
||
} catch (e) { alert("실패: " + e); }
|
||
}}
|
||
className="px-3 py-1.5 text-xs bg-red-50 text-red-600 border border-red-200 rounded hover:bg-red-100"
|
||
>
|
||
테이블링 초기화
|
||
</button>
|
||
<button
|
||
onClick={async () => {
|
||
if (!confirm("캐치테이블 매핑을 전부 초기화하시겠습니까?")) return;
|
||
try {
|
||
await fetch("/api/restaurants/reset-catchtable", {
|
||
method: "DELETE",
|
||
headers: { Authorization: `Bearer ${getAdminToken()}` },
|
||
});
|
||
alert("캐치테이블 매핑 초기화 완료");
|
||
load();
|
||
} catch (e) { alert("실패: " + e); }
|
||
}}
|
||
className="px-3 py-1.5 text-xs bg-red-50 text-red-600 border border-red-200 rounded hover:bg-red-100"
|
||
>
|
||
캐치테이블 초기화
|
||
</button>
|
||
<button
|
||
onClick={async () => {
|
||
const pending = await fetch(`/api/restaurants/catchtable-pending`, {
|
||
headers: { Authorization: `Bearer ${getAdminToken()}` },
|
||
}).then(r => r.json());
|
||
if (pending.count === 0) { alert("캐치테이블 미연결 식당이 없습니다"); return; }
|
||
if (!confirm(`캐치테이블 미연결 식당 ${pending.count}개를 벌크 검색합니다.\n식당당 5~15초 소요됩니다. 진행할까요?`)) return;
|
||
setBulkCatchtable(true);
|
||
setBulkCatchtableProgress({ current: 0, total: pending.count, name: "", linked: 0, notFound: 0 });
|
||
try {
|
||
const res = await fetch("/api/restaurants/bulk-catchtable", {
|
||
method: "POST",
|
||
headers: { Authorization: `Bearer ${getAdminToken()}` },
|
||
});
|
||
const reader = res.body!.getReader();
|
||
const decoder = new TextDecoder();
|
||
let buf = "";
|
||
while (true) {
|
||
const { done, value } = await reader.read();
|
||
if (done) break;
|
||
buf += decoder.decode(value, { stream: true });
|
||
const lines = buf.split("\n");
|
||
buf = lines.pop() || "";
|
||
for (const line of lines) {
|
||
const m = line.match(/^data:(.+)$/);
|
||
if (!m) continue;
|
||
const evt = JSON.parse(m[1]);
|
||
if (evt.type === "processing" || evt.type === "done" || evt.type === "notfound" || evt.type === "error") {
|
||
setBulkCatchtableProgress(p => ({
|
||
...p, current: evt.current, total: evt.total || p.total, name: evt.name,
|
||
linked: evt.type === "done" ? p.linked + 1 : p.linked,
|
||
notFound: (evt.type === "notfound" || evt.type === "error") ? p.notFound + 1 : p.notFound,
|
||
}));
|
||
} else if (evt.type === "complete") {
|
||
alert(`완료! 연결: ${evt.linked}개, 미발견: ${evt.notFound}개`);
|
||
}
|
||
}
|
||
}
|
||
} catch (e) { alert("벌크 캐치테이블 실패: " + (e instanceof Error ? e.message : String(e))); }
|
||
finally { setBulkCatchtable(false); load(); }
|
||
}}
|
||
disabled={bulkCatchtable}
|
||
className="px-3 py-1.5 text-xs bg-violet-500 text-white rounded hover:bg-violet-600 disabled:opacity-50"
|
||
>
|
||
{bulkCatchtable ? `캐치테이블 검색 중 (${bulkCatchtableProgress.current}/${bulkCatchtableProgress.total})` : "벌크 캐치테이블 연결"}
|
||
</button>
|
||
</>)}
|
||
<span className="text-sm text-gray-400 ml-auto">
|
||
{nameSearch ? `${sorted.length} / ` : ""}총 {restaurants.length}개 식당
|
||
</span>
|
||
</div>
|
||
{bulkTabling && bulkTablingProgress.name && (
|
||
<div className="bg-brand-50 rounded p-3 mb-4 text-sm">
|
||
<div className="flex justify-between mb-1">
|
||
<span>{bulkTablingProgress.current}/{bulkTablingProgress.total} - {bulkTablingProgress.name}</span>
|
||
<span className="text-xs text-gray-500">연결: {bulkTablingProgress.linked} / 미발견: {bulkTablingProgress.notFound}</span>
|
||
</div>
|
||
<div className="w-full bg-brand-200 rounded-full h-1.5">
|
||
<div className="bg-brand-500 h-1.5 rounded-full transition-all" style={{ width: `${(bulkTablingProgress.current / bulkTablingProgress.total) * 100}%` }} />
|
||
</div>
|
||
</div>
|
||
)}
|
||
{bulkCatchtable && bulkCatchtableProgress.name && (
|
||
<div className="bg-violet-50 rounded p-3 mb-4 text-sm">
|
||
<div className="flex justify-between mb-1">
|
||
<span>{bulkCatchtableProgress.current}/{bulkCatchtableProgress.total} - {bulkCatchtableProgress.name}</span>
|
||
<span className="text-xs text-gray-500">연결: {bulkCatchtableProgress.linked} / 미발견: {bulkCatchtableProgress.notFound}</span>
|
||
</div>
|
||
<div className="w-full bg-violet-200 rounded-full h-1.5">
|
||
<div className="bg-violet-500 h-1.5 rounded-full transition-all" style={{ width: `${(bulkCatchtableProgress.current / bulkCatchtableProgress.total) * 100}%` }} />
|
||
</div>
|
||
</div>
|
||
)}
|
||
|
||
<div className="bg-surface rounded-lg shadow overflow-auto">
|
||
<table className="w-full text-sm">
|
||
<thead className="bg-brand-50 border-b border-brand-100 text-brand-800 text-sm font-semibold">
|
||
<tr>
|
||
<th className="text-left px-4 py-3 cursor-pointer select-none hover:bg-gray-100" onClick={() => handleSort("name")}>이름{sortIcon("name")}</th>
|
||
<th className="text-left px-4 py-3 cursor-pointer select-none hover:bg-gray-100" onClick={() => handleSort("region")}>지역{sortIcon("region")}</th>
|
||
<th className="text-left px-4 py-3 cursor-pointer select-none hover:bg-gray-100" onClick={() => handleSort("cuisine_type")}>음식 종류{sortIcon("cuisine_type")}</th>
|
||
<th className="text-left px-4 py-3 cursor-pointer select-none hover:bg-gray-100" onClick={() => handleSort("price_range")}>가격대{sortIcon("price_range")}</th>
|
||
<th className="text-center px-4 py-3 cursor-pointer select-none hover:bg-gray-100" onClick={() => handleSort("rating")}>평점{sortIcon("rating")}</th>
|
||
<th className="text-center px-4 py-3 cursor-pointer select-none hover:bg-gray-100" onClick={() => handleSort("business_status")}>상태{sortIcon("business_status")}</th>
|
||
<th className="text-center px-4 py-3">검증</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
{paged.map((r) => (
|
||
<tr
|
||
key={r.id}
|
||
onClick={() => handleSelect(r)}
|
||
className={`border-b cursor-pointer hover:bg-brand-50/50 ${selected?.id === r.id ? "bg-brand-50" : ""}`}
|
||
>
|
||
<td className="px-4 py-3 font-medium">{r.name}</td>
|
||
<td className="px-4 py-3 text-gray-600 text-xs">{r.region || "-"}</td>
|
||
<td className="px-4 py-3 text-gray-600 text-xs">{r.cuisine_type || "-"}</td>
|
||
<td className="px-4 py-3 text-gray-600 text-xs">{r.price_range || "-"}</td>
|
||
<td className="px-4 py-3 text-center text-xs">
|
||
{r.rating ? (
|
||
<span><span className="text-yellow-500">★</span> {r.rating}</span>
|
||
) : "-"}
|
||
</td>
|
||
<td className="px-4 py-3 text-center">
|
||
{r.business_status === "CLOSED_PERMANENTLY" ? (
|
||
<span className="px-1.5 py-0.5 bg-red-100 text-red-700 rounded text-[10px] font-semibold">폐업</span>
|
||
) : r.business_status === "CLOSED_TEMPORARILY" ? (
|
||
<span className="px-1.5 py-0.5 bg-yellow-100 text-yellow-700 rounded text-[10px] font-semibold">임시휴업</span>
|
||
) : (
|
||
<span className="text-xs text-gray-400">-</span>
|
||
)}
|
||
</td>
|
||
<td className="px-4 py-3 text-center" onClick={(e) => e.stopPropagation()}>
|
||
{r.hidden ? (
|
||
<button
|
||
onClick={() => handleToggleHidden(r)}
|
||
disabled={!isAdmin}
|
||
title={r.hidden_reason || "manual"}
|
||
className="px-1.5 py-0.5 bg-red-100 text-red-700 rounded text-[10px] font-semibold hover:bg-red-200 disabled:opacity-50"
|
||
>
|
||
숨김 {r.hidden_reason ? `(${r.hidden_reason.slice(0, 12)})` : ""}
|
||
</button>
|
||
) : r.verified_at ? (
|
||
<button
|
||
onClick={() => handleToggleHidden(r)}
|
||
disabled={!isAdmin}
|
||
title="검증 통과 — 클릭하면 숨김"
|
||
className="px-1.5 py-0.5 bg-green-100 text-green-700 rounded text-[10px] font-semibold hover:bg-green-200 disabled:opacity-50"
|
||
>
|
||
OK
|
||
</button>
|
||
) : (
|
||
<span className="text-xs text-gray-400">미검증</span>
|
||
)}
|
||
</td>
|
||
</tr>
|
||
))}
|
||
{!loading && filtered.length === 0 && (
|
||
<tr>
|
||
<td colSpan={7} className="px-4 py-8 text-center text-gray-400">
|
||
식당 데이터가 없습니다
|
||
</td>
|
||
</tr>
|
||
)}
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
|
||
{totalPages > 1 && (
|
||
<div className="flex items-center justify-center gap-2 mt-4">
|
||
<button onClick={() => setPage(0)} disabled={page === 0} className="px-3 py-1 rounded border text-sm disabled:opacity-30 hover:bg-gray-100">«</button>
|
||
<button onClick={() => setPage(page - 1)} disabled={page === 0} className="px-3 py-1 rounded border text-sm disabled:opacity-30 hover:bg-gray-100">‹</button>
|
||
<span className="text-sm text-gray-600 px-2">{page + 1} / {totalPages}</span>
|
||
<button onClick={() => setPage(page + 1)} disabled={page >= totalPages - 1} className="px-3 py-1 rounded border text-sm disabled:opacity-30 hover:bg-gray-100">›</button>
|
||
<button onClick={() => setPage(totalPages - 1)} disabled={page >= totalPages - 1} className="px-3 py-1 rounded border text-sm disabled:opacity-30 hover:bg-gray-100">»</button>
|
||
</div>
|
||
)}
|
||
|
||
{/* 식당 상세/수정 패널 */}
|
||
{selected && (
|
||
<div className="mt-6 bg-surface rounded-lg shadow p-4">
|
||
<div className="flex items-center justify-between mb-4">
|
||
<h3 className="font-semibold text-base">{selected.name}</h3>
|
||
<button onClick={() => setSelected(null)} className="text-gray-400 hover:text-gray-600 text-xl leading-none">x</button>
|
||
</div>
|
||
<div className="grid grid-cols-1 md:grid-cols-2 gap-3">
|
||
{[
|
||
{ key: "name", label: "이름" },
|
||
{ key: "address", label: "주소" },
|
||
{ key: "region", label: "지역" },
|
||
{ key: "cuisine_type", label: "음식 종류" },
|
||
{ key: "price_range", label: "가격대" },
|
||
{ key: "phone", label: "전화" },
|
||
{ key: "website", label: "웹사이트" },
|
||
{ key: "latitude", label: "위도" },
|
||
{ key: "longitude", label: "경도" },
|
||
].map(({ key, label }) => (
|
||
<div key={key}>
|
||
<label className="text-xs text-gray-500">{label}</label>
|
||
<input
|
||
value={editForm[key] || ""}
|
||
onChange={(e) => setEditForm((f) => ({ ...f, [key]: e.target.value }))}
|
||
className="w-full border rounded px-2 py-1.5 text-sm bg-surface text-gray-900"
|
||
disabled={!isAdmin}
|
||
/>
|
||
</div>
|
||
))}
|
||
</div>
|
||
{selected.business_status && (
|
||
<p className="mt-3 text-xs text-gray-500">
|
||
Google 상태: <span className="font-medium">{selected.business_status}</span>
|
||
{selected.rating && <> / ★ {selected.rating} ({selected.rating_count?.toLocaleString()})</>}
|
||
</p>
|
||
)}
|
||
{selected.google_place_id && (
|
||
<p className="mt-1">
|
||
<a
|
||
href={`https://www.google.com/maps/place/?q=place_id:${selected.google_place_id}`}
|
||
target="_blank"
|
||
rel="noopener noreferrer"
|
||
className="text-brand-600 hover:underline text-xs"
|
||
>
|
||
Google Maps에서 보기
|
||
</a>
|
||
</p>
|
||
)}
|
||
{/* 테이블링 연결 */}
|
||
{isAdmin && (
|
||
<div className="mt-4 border-t pt-3">
|
||
<div className="flex items-center gap-2 mb-2">
|
||
<h4 className="text-xs font-semibold text-gray-500">테이블링</h4>
|
||
{selected.tabling_url === "NONE" ? (
|
||
<span className="text-xs text-gray-400">검색완료-없음</span>
|
||
) : selected.tabling_url ? (
|
||
<a href={selected.tabling_url} target="_blank" rel="noopener noreferrer"
|
||
className="text-brand-600 hover:underline text-xs">{selected.tabling_url}</a>
|
||
) : (
|
||
<span className="text-xs text-gray-400">미연결</span>
|
||
)}
|
||
<button
|
||
onClick={async () => {
|
||
setTablingSearching(true);
|
||
try {
|
||
const results = await api.searchTabling(selected.id);
|
||
if (results.length === 0) {
|
||
alert("테이블링에서 검색 결과가 없습니다");
|
||
} else {
|
||
const best = results[0];
|
||
if (confirm(`"${best.title}"\n${best.url}\n\n이 테이블링 페이지를 연결할까요?`)) {
|
||
await api.setTablingUrl(selected.id, best.url);
|
||
setSelected({ ...selected, tabling_url: best.url });
|
||
load();
|
||
}
|
||
}
|
||
} catch (e) { alert("검색 실패: " + (e instanceof Error ? e.message : String(e))); }
|
||
finally { setTablingSearching(false); }
|
||
}}
|
||
disabled={tablingSearching}
|
||
className="px-2 py-0.5 text-[11px] bg-brand-500 text-white rounded hover:bg-brand-600 disabled:opacity-50"
|
||
>
|
||
{tablingSearching ? "검색 중..." : "테이블링 검색"}
|
||
</button>
|
||
{selected.tabling_url && (
|
||
<button
|
||
onClick={async () => {
|
||
await api.setTablingUrl(selected.id, "");
|
||
setSelected({ ...selected, tabling_url: null });
|
||
load();
|
||
}}
|
||
className="px-2 py-0.5 text-[11px] text-red-500 border border-red-200 rounded hover:bg-red-50"
|
||
>
|
||
연결 해제
|
||
</button>
|
||
)}
|
||
</div>
|
||
</div>
|
||
)}
|
||
{/* 캐치테이블 연결 */}
|
||
{isAdmin && (
|
||
<div className="mt-4 border-t pt-3">
|
||
<div className="flex items-center gap-2 mb-2">
|
||
<h4 className="text-xs font-semibold text-gray-500">캐치테이블</h4>
|
||
{selected.catchtable_url === "NONE" ? (
|
||
<span className="text-xs text-gray-400">검색완료-없음</span>
|
||
) : selected.catchtable_url ? (
|
||
<a href={selected.catchtable_url} target="_blank" rel="noopener noreferrer"
|
||
className="text-brand-600 hover:underline text-xs">{selected.catchtable_url}</a>
|
||
) : (
|
||
<span className="text-xs text-gray-400">미연결</span>
|
||
)}
|
||
<button
|
||
onClick={async () => {
|
||
setCatchtableSearching(true);
|
||
try {
|
||
const results = await api.searchCatchtable(selected.id);
|
||
if (results.length === 0) {
|
||
alert("캐치테이블에서 검색 결과가 없습니다");
|
||
} else {
|
||
const best = results[0];
|
||
if (confirm(`"${best.title}"\n${best.url}\n\n이 캐치테이블 페이지를 연결할까요?`)) {
|
||
await api.setCatchtableUrl(selected.id, best.url);
|
||
setSelected({ ...selected, catchtable_url: best.url });
|
||
load();
|
||
}
|
||
}
|
||
} catch (e) { alert("검색 실패: " + (e instanceof Error ? e.message : String(e))); }
|
||
finally { setCatchtableSearching(false); }
|
||
}}
|
||
disabled={catchtableSearching}
|
||
className="px-2 py-0.5 text-[11px] bg-violet-500 text-white rounded hover:bg-violet-600 disabled:opacity-50"
|
||
>
|
||
{catchtableSearching ? "검색 중..." : "캐치테이블 검색"}
|
||
</button>
|
||
{selected.catchtable_url && (
|
||
<button
|
||
onClick={async () => {
|
||
await api.setCatchtableUrl(selected.id, "");
|
||
setSelected({ ...selected, catchtable_url: null });
|
||
load();
|
||
}}
|
||
className="px-2 py-0.5 text-[11px] text-red-500 border border-red-200 rounded hover:bg-red-50"
|
||
>
|
||
연결 해제
|
||
</button>
|
||
)}
|
||
</div>
|
||
</div>
|
||
)}
|
||
|
||
{videos.length > 0 && (
|
||
<div className="mt-4 border-t pt-3">
|
||
<h4 className="text-xs font-semibold text-gray-500 mb-2">연결된 영상 ({videos.length})</h4>
|
||
<div className="space-y-1.5 max-h-32 overflow-y-auto">
|
||
{videos.map((v) => (
|
||
<div key={v.video_id} className="flex items-center gap-2 text-xs">
|
||
<span className="px-1.5 py-0.5 bg-red-50 text-red-600 rounded text-[10px] font-medium shrink-0">
|
||
{v.channel_name}
|
||
</span>
|
||
<a href={v.url} target="_blank" rel="noopener noreferrer" className="text-brand-600 hover:underline truncate">
|
||
{v.title}
|
||
</a>
|
||
<span className="text-gray-400 shrink-0">{v.published_at?.slice(0, 10)}</span>
|
||
</div>
|
||
))}
|
||
</div>
|
||
</div>
|
||
)}
|
||
|
||
<div className="flex gap-2 mt-4">
|
||
{isAdmin && <button
|
||
onClick={handleSave}
|
||
disabled={saving}
|
||
className="px-4 py-2 text-sm bg-brand-600 text-white rounded hover:bg-brand-700 disabled:opacity-50"
|
||
>
|
||
{saving ? "저장 중..." : "저장"}
|
||
</button>}
|
||
<button
|
||
onClick={() => setSelected(null)}
|
||
className="px-4 py-2 text-sm border rounded text-gray-600 hover:bg-gray-100"
|
||
>
|
||
{isAdmin ? "취소" : "닫기"}
|
||
</button>
|
||
{isAdmin && <button
|
||
onClick={handleDelete}
|
||
className="px-4 py-2 text-sm text-red-500 border border-red-200 rounded hover:bg-red-50 ml-auto"
|
||
>
|
||
식당 삭제
|
||
</button>}
|
||
</div>
|
||
</div>
|
||
)}
|
||
</div>
|
||
);
|
||
}
|
||
|
||
|
||
/* ─── 유저 관리 ─── */
|
||
interface AdminUser {
|
||
id: string;
|
||
email: string | null;
|
||
nickname: string | null;
|
||
avatar_url: string | null;
|
||
is_admin: boolean;
|
||
provider: string | null;
|
||
created_at: string | null;
|
||
favorite_count: number;
|
||
review_count: number;
|
||
memo_count: number;
|
||
}
|
||
|
||
interface UserFavorite {
|
||
id: string;
|
||
name: string;
|
||
address: string | null;
|
||
region: string | null;
|
||
cuisine_type: string | null;
|
||
rating: number | null;
|
||
business_status: string | null;
|
||
created_at: string | null;
|
||
}
|
||
|
||
interface UserReview {
|
||
id: string;
|
||
restaurant_id: string;
|
||
rating: number;
|
||
review_text: string | null;
|
||
visited_at: string | null;
|
||
created_at: string | null;
|
||
restaurant_name: string | null;
|
||
}
|
||
|
||
interface UserMemo {
|
||
id: string;
|
||
restaurant_id: string;
|
||
rating: number | null;
|
||
memo_text: string | null;
|
||
visited_at: string | null;
|
||
created_at: string;
|
||
restaurant_name: string | null;
|
||
}
|
||
|