Merge branch 'subwork' into mainwork

This commit is contained in:
koreacomp5
2025-11-02 04:39:42 +09:00
5 changed files with 188 additions and 63 deletions

View File

@@ -1,36 +1,80 @@
"use client";
import useSWR from "swr";
import { useState } from "react";
import { useMemo, useState } from "react";
const fetcher = (url: string) => fetch(url).then((r) => r.json());
export default function AdminUsersPage() {
const [q, setQ] = useState("");
const { data, mutate } = useSWR<{ users: any[] }>(`/api/admin/users?q=${encodeURIComponent(q)}`, fetcher);
const [queryDraft, setQueryDraft] = useState("");
const [page, setPage] = useState(1);
const [pageSize, setPageSize] = useState(20);
const key = useMemo(() => (
`/api/admin/users?q=${encodeURIComponent(q)}&page=${page}&pageSize=${pageSize}`
),[q, page, pageSize]);
const { data, mutate } = useSWR<{ total: number; page: number; pageSize: number; users: any[] }>(key, fetcher);
const users = data?.users ?? [];
const total = data?.total ?? 0;
const totalPages = Math.max(1, Math.ceil(total / pageSize));
const handleSearch = () => {
const term = queryDraft.trim();
if (!term) { setQ(""); setPage(1); return; }
setQ(term);
setPage(1);
};
return (
<div>
<h1> </h1>
<div style={{ display: "flex", gap: 8, marginBottom: 12 }}>
<input placeholder="검색(nickname/phone/name)" value={q} onChange={(e) => setQ(e.target.value)} />
<div className="mb-3 rounded-xl border border-neutral-300 bg-white p-3 flex items-center gap-2">
<input
className="h-9 w-full max-w-[360px] px-3 rounded-md border border-neutral-300 bg-white text-sm"
placeholder="검색 (닉네임/전화/이름)"
value={queryDraft}
onChange={(e) => setQueryDraft(e.target.value)}
onKeyDown={(e) => { if (e.key === "Enter") handleSearch(); }}
/>
<button
className="h-9 px-4 rounded-md border border-neutral-300 bg-white text-sm hover:bg-neutral-100 disabled:opacity-50"
onClick={handleSearch}
disabled={false}
>
</button>
<span className="ml-auto text-xs text-neutral-600"> {total.toLocaleString()}</span>
</div>
<div className="rounded-xl border border-neutral-200 overflow-hidden">
<table className="w-full text-sm">
<thead className="bg-[#f6f4f4] border-b border-[#e6e6e6]">
<tr>
<th className="px-4 py-2 text-left text-[12px] text-[#8c8c8c]"></th>
<th className="px-4 py-2 text-left text-[12px] text-[#8c8c8c]"></th>
<th className="px-4 py-2 text-left text-[12px] text-[#8c8c8c]"></th>
<th className="px-4 py-2 text-right text-[12px] text-[#8c8c8c]"></th>
<th className="px-4 py-2 text-center text-[12px] text-[#8c8c8c]"></th>
<th className="px-4 py-2 text-center text-[12px] text-[#8c8c8c]"></th>
<th className="px-4 py-2 text-center text-[12px] text-[#8c8c8c]"></th>
<th className="px-4 py-2 text-left text-[12px] text-[#8c8c8c]"></th>
<th className="px-4 py-2 text-center text-[12px] text-[#8c8c8c]"></th>
</tr>
</thead>
<tbody className="divide-y divide-[#ececec] bg-white">
{users.map((u) => (
<Row key={u.userId} u={u} onChanged={mutate} />
))}
</tbody>
</table>
</div>
<div style={{ display: "flex", alignItems: "center", gap: 8, marginTop: 12 }}>
<button disabled={page <= 1} onClick={() => setPage((p) => Math.max(1, p - 1))}></button>
<span style={{ fontSize: 12 }}> {page} / {totalPages}</span>
<button disabled={page >= totalPages} onClick={() => setPage((p) => Math.min(totalPages, p + 1))}></button>
<span style={{ marginLeft: 12, fontSize: 12 }}> </span>
<select value={pageSize} onChange={(e) => { setPageSize(parseInt(e.target.value, 10)); setPage(1); }}>
{[10, 20, 50, 100].map((s) => (<option key={s} value={s}>{s}</option>))}
</select>
</div>
<table style={{ width: "100%", borderCollapse: "collapse" }}>
<thead>
<tr>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
{users.map((u) => (
<Row key={u.userId} u={u} onChanged={mutate} />
))}
</tbody>
</table>
</div>
);
}
@@ -45,25 +89,30 @@ function Row({ u, onChanged }: { u: any; onChanged: () => void }) {
}
const allRoles = ["admin", "editor", "user"] as const;
return (
<tr>
<td>{u.nickname}</td>
<td>{u.name}</td>
<td>{u.phone}</td>
<td>
<select value={status} onChange={(e) => setStatus(e.target.value)}>
<option value="active">active</option>
<option value="suspended">suspended</option>
<option value="withdrawn">withdrawn</option>
<tr className="hover:bg-neutral-50">
<td className="px-4 py-2 text-left">{u.nickname}</td>
<td className="px-4 py-2 text-left">{u.name}</td>
<td className="px-4 py-2 text-left">{u.phone}</td>
<td className="px-4 py-2 text-right tabular-nums">{(u.points ?? 0).toLocaleString()}</td>
<td className="px-4 py-2 text-center">{u.level ?? 1}</td>
<td className="px-4 py-2 text-center">{u.grade ?? 0}</td>
<td className="px-4 py-2 text-center">
<select className="h-8 px-2 border border-neutral-300 rounded-md bg-white text-sm" value={status} onChange={(e) => setStatus(e.target.value)}>
<option value="active"></option>
<option value="suspended"></option>
<option value="withdrawn"></option>
</select>
</td>
<td>
<td className="px-4 py-2">
{allRoles.map((r) => (
<label key={r} style={{ marginRight: 8 }}>
<label key={r} className="mr-2">
<input type="checkbox" checked={roles.includes(r)} onChange={(e) => setRoles((prev) => (e.target.checked ? Array.from(new Set([...prev, r])) : prev.filter((x) => x !== r)))} /> {r}
</label>
))}
</td>
<td><button onClick={save}></button></td>
<td className="px-4 py-2 text-center">
<button className="h-8 px-3 rounded-md bg-[#f94b37] text-white text-sm border border-[#d73b29] hover:opacity-95" onClick={save}></button>
</td>
</tr>
);
}