feat: allow admins to accept miis in queue

This commit is contained in:
trafficlunar 2026-04-01 12:01:30 +01:00
parent 147d005a14
commit 12c0205bf5
5 changed files with 60 additions and 10 deletions

View file

@ -8,6 +8,7 @@ import ControlCenter from "@/components/admin/control-center";
import RegenerateImagesButton from "@/components/admin/regenerate-images";
import UserManagement from "@/components/admin/user-management";
import Reports from "@/components/admin/reports";
import MiiList from "@/components/mii/list";
export const metadata: Metadata = {
title: "Admin - TomodachiShare",
@ -18,7 +19,11 @@ export const metadata: Metadata = {
},
};
export default async function AdminPage() {
interface Props {
searchParams: Promise<{ [key: string]: string | string[] | undefined }>;
}
export default async function AdminPage({ searchParams }: Props) {
const session = await auth();
if (!session || Number(session.user?.id) !== Number(process.env.NEXT_PUBLIC_ADMIN_USER_ID)) redirect("/404");
@ -66,6 +71,14 @@ export default async function AdminPage() {
</div>
<Reports />
{/* Queue */}
<div className="flex items-center gap-4 text-zinc-500 text-sm font-medium my-1">
<hr className="grow border-zinc-300" />
<span>Reports</span>
<hr className="grow border-zinc-300" />
</div>
<MiiList parentPage="admin" searchParams={await searchParams} />
</div>
);
}

View file

@ -0,0 +1,29 @@
import { NextRequest, NextResponse } from "next/server";
import { auth } from "@/lib/auth";
import { prisma } from "@/lib/prisma";
import { idSchema } from "@/lib/schemas";
export async function PATCH(request: NextRequest) {
const session = await auth();
if (!session) return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
if (Number(session.user?.id) !== Number(process.env.NEXT_PUBLIC_ADMIN_USER_ID)) return NextResponse.json({ error: "Forbidden" }, { status: 403 });
const searchParams = request.nextUrl.searchParams;
const parsedMiiId = idSchema.safeParse(searchParams.get("id"));
if (!parsedMiiId.success) return NextResponse.json({ error: parsedMiiId.error.issues[0].message }, { status: 400 });
const miiId = parsedMiiId.data;
await prisma.mii.update({
where: {
id: miiId,
},
data: {
in_queue: false,
},
});
return NextResponse.json({ success: true });
}

View file

@ -37,7 +37,7 @@ export default async function ProfileSettingsPage({ searchParams }: Props) {
</div>
<Suspense fallback={<Skeleton />}>
<MiiList inLikesPage searchParams={await searchParams} />
<MiiList parentPage="likes" searchParams={await searchParams} />
</Suspense>
</div>
);

View file

@ -15,10 +15,10 @@ import MiiGrid from "./mii-grid";
interface Props {
searchParams: { [key: string]: string | string[] | undefined };
userId?: number; // Profiles
inLikesPage?: boolean; // Self-explanatory
parentPage?: "likes" | "admin";
}
export default async function MiiList({ searchParams, userId, inLikesPage }: Props) {
export default async function MiiList({ searchParams, userId, parentPage }: Props) {
const session = await auth();
const parsed = searchSchema.safeParse(searchParams);
if (!parsed.success) return <h1>{parsed.error.issues[0].message}</h1>;
@ -28,7 +28,7 @@ export default async function MiiList({ searchParams, userId, inLikesPage }: Pro
// My Likes page
let miiIdsLiked: number[] | undefined = undefined;
if (inLikesPage && session?.user?.id) {
if (parentPage === "likes" && session?.user?.id) {
const likedMiis = await prisma.like.findMany({
where: { userId: Number(session.user.id) },
select: { miiId: true },
@ -37,9 +37,9 @@ export default async function MiiList({ searchParams, userId, inLikesPage }: Pro
}
const where: Prisma.MiiWhereInput = {
in_queue: false,
in_queue: parentPage === "admin",
// Only show liked miis on likes page
...(inLikesPage && miiIdsLiked && { id: { in: miiIdsLiked } }),
...(parentPage === "likes" && miiIdsLiked && { id: { in: miiIdsLiked } }),
// Searching
...(query && {
OR: [{ name: { contains: query, mode: "insensitive" } }, { tags: { has: query } }, { description: { contains: query, mode: "insensitive" } }],
@ -184,7 +184,7 @@ export default async function MiiList({ searchParams, userId, inLikesPage }: Pro
</div>
</div>
<MiiGrid miis={miis} userId={userId} />
<MiiGrid miis={miis} userId={userId} parentPage={parentPage} />
<Pagination lastPage={lastPage} />
</div>
);

View file

@ -13,11 +13,12 @@ import Carousel from "@/components/carousel";
interface Props {
miis: Prisma.MiiGetPayload<{ include: { user: { select: { id: true; name: true } }; _count: { select: { likedBy: true } } } }>[];
userId?: number;
parentPage?: string;
}
const fetcher = (url: string) => fetch(url).then((res) => res.json());
export default function MiiGrid({ miis, userId }: Props) {
export default function MiiGrid({ miis, userId, parentPage }: Props) {
const session = useSession();
const ids = miis.map((m) => m.id).join(",");
const { data } = useSWR<number[]>(session.data?.user && miis.length > 0 ? `/api/mii/has-liked?ids=${ids}` : null, fetcher, {
@ -79,6 +80,13 @@ export default function MiiGrid({ miis, userId }: Props) {
<DeleteMiiButton miiId={mii.id} miiName={mii.name} likes={mii._count.likedBy} />
</div>
)}
{parentPage === "admin" && (
<div className="flex gap-1 text-2xl justify-end text-zinc-400">
<button onClick={() => fetch(`/api/admin/accept-mii?id=${mii.id}`, { method: "PATCH" })} className="cursor-pointer">
<Icon icon="material-symbols:check-rounded" />
</button>
</div>
)}
</div>
</div>
</div>