11 Commits

17 changed files with 814 additions and 215 deletions

View File

@ -18,6 +18,12 @@ RUN bunx prisma generate
# Uncomment the following line in case you want to disable telemetry during the build. # Uncomment the following line in case you want to disable telemetry during the build.
ENV NEXT_TELEMETRY_DISABLED=1 ENV NEXT_TELEMETRY_DISABLED=1
ARG GIT_SHA=unknown
ARG APP_VERSION=0.0.0
ARG DEPLOY_ENV=production
ENV NEXT_PUBLIC_GIT_SHA=$GIT_SHA \
NEXT_PUBLIC_APP_VERSION=$APP_VERSION \
NEXT_PUBLIC_DEPLOY_ENV=$DEPLOY_ENV
ARG DATABASE_URL ARG DATABASE_URL
ENV DATABASE_URL=$DATABASE_URL ENV DATABASE_URL=$DATABASE_URL
@ -33,6 +39,12 @@ ENV NEXT_TELEMETRY_DISABLED=1
ENV NODE_ENV=production \ ENV NODE_ENV=production \
PORT=3000 \ PORT=3000 \
HOSTNAME="0.0.0.0" HOSTNAME="0.0.0.0"
ARG GIT_SHA=unknown
ARG APP_VERSION=0.0.0
ARG DEPLOY_ENV=production
ENV NEXT_PUBLIC_GIT_SHA=$GIT_SHA \
NEXT_PUBLIC_APP_VERSION=$APP_VERSION \
NEXT_PUBLIC_DEPLOY_ENV=$DEPLOY_ENV
RUN groupadd --system --gid 1001 nodejs && \ RUN groupadd --system --gid 1001 nodejs && \
useradd --system --uid 1001 --no-log-init -g nodejs nextjs useradd --system --uid 1001 --no-log-init -g nodejs nextjs

View File

@ -14,7 +14,6 @@ FROM base AS builder
WORKDIR /app WORKDIR /app
COPY --from=deps /app/node_modules ./node_modules COPY --from=deps /app/node_modules ./node_modules
COPY . . COPY . .
# RUN bunx prisma migrate deploy
RUN bunx prisma generate RUN bunx prisma generate
# Uncomment the following line in case you want to disable telemetry during the build. # Uncomment the following line in case you want to disable telemetry during the build.

View File

@ -53,7 +53,7 @@ model Artwork {
albums Album[] albums Album[]
categories ArtCategory[] categories ArtCategory[]
colors ArtworkColor[] colors ArtworkColor[]
tags ArtTag[] tags Tag[] @relation("ArtworkTags")
variants FileVariant[] variants FileVariant[]
@@index([colorStatus]) @@index([colorStatus])
@ -101,43 +101,7 @@ model ArtCategory {
description String? description String?
artworks Artwork[] artworks Artwork[]
tags ArtTag[] tagLinks TagCategory[]
}
model ArtTag {
id String @id @default(cuid())
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
sortIndex Int @default(0)
name String @unique
slug String @unique
isParent Boolean @default(false)
showOnAnimalPage Boolean @default(false)
description String?
aliases ArtTagAlias[]
artworks Artwork[]
categories ArtCategory[]
parentId String?
parent ArtTag? @relation("TagHierarchy", fields: [parentId], references: [id], onDelete: SetNull)
children ArtTag[] @relation("TagHierarchy")
}
model ArtTagAlias {
id String @id @default(cuid())
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
alias String @unique
tagId String
tag ArtTag @relation(fields: [tagId], references: [id], onDelete: Cascade)
@@unique([tagId, alias])
@@index([alias])
} }
model Color { model Color {
@ -248,6 +212,72 @@ model FileVariant {
@@unique([artworkId, type]) @@unique([artworkId, type])
} }
model Tag {
id String @id @default(cuid())
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
sortIndex Int @default(0)
name String @unique
slug String @unique
isVisible Boolean @default(true)
description String?
aliases TagAlias[]
categoryLinks TagCategory[]
categoryParents TagCategory[] @relation("TagCategoryParent")
artworks Artwork[] @relation("ArtworkTags")
commissionTypes CommissionType[] @relation("CommissionTypeTags")
commissionCustomCards CommissionCustomCard[] @relation("CommissionCustomCardTags")
miniatures Miniature[] @relation("MiniatureTags")
}
model TagAlias {
id String @id @default(cuid())
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
alias String @unique
tagId String
tag Tag @relation(fields: [tagId], references: [id], onDelete: Cascade)
@@unique([tagId, alias])
@@index([alias])
}
model TagCategory {
id String @id @default(cuid())
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
tagId String
categoryId String
isParent Boolean @default(false)
showOnAnimalPage Boolean @default(false)
parentTagId String?
tag Tag @relation(fields: [tagId], references: [id], onDelete: Cascade)
category ArtCategory @relation(fields: [categoryId], references: [id], onDelete: Cascade)
parentTag Tag? @relation("TagCategoryParent", fields: [parentTagId], references: [id], onDelete: SetNull)
@@unique([tagId, categoryId])
@@index([categoryId])
@@index([tagId])
@@index([parentTagId])
@@index([categoryId, parentTagId])
}
model Miniature {
id String @id @default(cuid())
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
tags Tag[] @relation("MiniatureTags")
}
model Commission { model Commission {
id String @id @default(cuid()) id String @id @default(cuid())
createdAt DateTime @default(now()) createdAt DateTime @default(now())
@ -265,6 +295,8 @@ model CommissionType {
description String? description String?
tags Tag[] @relation("CommissionTypeTags")
options CommissionTypeOption[] options CommissionTypeOption[]
extras CommissionTypeExtra[] extras CommissionTypeExtra[]
customInputs CommissionTypeCustomInput[] customInputs CommissionTypeCustomInput[]
@ -284,6 +316,7 @@ model CommissionCustomCard {
isVisible Boolean @default(true) isVisible Boolean @default(true)
isSpecialOffer Boolean @default(false) isSpecialOffer Boolean @default(false)
tags Tag[] @relation("CommissionCustomCardTags")
options CommissionCustomCardOption[] options CommissionCustomCardOption[]
extras CommissionCustomCardExtra[] extras CommissionCustomCardExtra[]
requests CommissionRequest[] requests CommissionRequest[]

View File

@ -0,0 +1,173 @@
"use server";
import type { Prisma } from "@/generated/prisma/browser";
import { prisma } from "@/lib/prisma";
export type Cursor = {
afterSortKey: number | null;
afterId: string;
} | null;
export type TaggedArtworkItem = {
id: string;
name: string;
altText: string | null;
sortKey: number | null;
year: number | null;
fileKey: string;
thumbW: number;
thumbH: number;
dominantHex: string;
};
type VariantPick = { type: string; width: number; height: number };
function pickVariant(variants: VariantPick[], type: string) {
return variants.find((v) => v.type === type) ?? null;
}
export async function getTaggedArtworksPage(args: {
take?: number;
cursor?: Cursor;
tagSlugs: string[];
onlyPublished?: boolean;
}): Promise<{
items: TaggedArtworkItem[];
nextCursor: Cursor;
total: number;
}> {
const { take = 60, cursor = null, tagSlugs, onlyPublished = true } = args;
const filteredSlugs = tagSlugs.map((s) => s.trim()).filter(Boolean);
if (filteredSlugs.length === 0) {
return { items: [], nextCursor: null, total: 0 };
}
const baseWhere: Prisma.ArtworkWhereInput = {
...(onlyPublished ? { published: true } : {}),
tags: { some: { slug: { in: filteredSlugs } } },
variants: { some: { type: "thumbnail" } },
};
const total = await prisma.artwork.count({ where: baseWhere });
const select = {
id: true,
name: true,
altText: true,
year: true,
sortKey: true,
file: { select: { fileKey: true } },
variants: {
where: { type: "thumbnail" },
select: { type: true, width: true, height: true },
take: 1,
},
colors: {
where: { type: "Vibrant" },
select: { color: { select: { hex: true } } },
take: 1,
},
} satisfies Prisma.ArtworkSelect;
type ArtworkRow = Prisma.ArtworkGetPayload<{ select: typeof select }>;
const mapRow = (r: ArtworkRow): TaggedArtworkItem | null => {
const thumb = pickVariant(r.variants, "thumbnail");
if (!thumb?.width || !thumb?.height) return null;
return {
id: r.id,
name: r.name,
altText: r.altText ?? null,
sortKey: r.sortKey ?? null,
year: r.year ?? null,
fileKey: r.file.fileKey,
thumbW: thumb.width,
thumbH: thumb.height,
dominantHex: r.colors[0]?.color?.hex ?? "#999999",
};
};
let items: TaggedArtworkItem[] = [];
let nextCursor: Cursor = null;
const inNullSegment = cursor?.afterSortKey === null;
if (!inNullSegment) {
const whereA: Prisma.ArtworkWhereInput = {
AND: [baseWhere, { sortKey: { not: null } }],
};
if (cursor?.afterSortKey != null) {
const sk = Number(cursor.afterSortKey);
whereA.OR = [
{ sortKey: { gt: sk } },
{ AND: [{ sortKey: sk }, { id: { gt: cursor.afterId } }] },
];
}
const rowsA = await prisma.artwork.findMany({
where: whereA,
orderBy: [{ sortKey: "asc" }, { id: "asc" }],
take: Math.min(take, 200),
select,
});
items = rowsA.map(mapRow).filter((x): x is TaggedArtworkItem => x !== null);
if (items.length >= take) {
const last = items.at(-1);
if (!last || last.sortKey == null) {
return { items, nextCursor: null, total };
}
nextCursor = { afterSortKey: last.sortKey, afterId: last.id };
return { items, nextCursor, total };
}
const remaining = take - items.length;
const whereB: Prisma.ArtworkWhereInput = {
AND: [baseWhere, { sortKey: null }],
};
const rowsB = await prisma.artwork.findMany({
where: whereB,
orderBy: [{ id: "asc" }],
take: Math.min(remaining, 200),
select,
});
const more = rowsB.map(mapRow).filter((x): x is TaggedArtworkItem => x !== null);
items = items.concat(more);
const last = items[items.length - 1];
nextCursor =
items.length < take || !last
? null
: { afterSortKey: last.sortKey ?? null, afterId: last.id };
return { items, nextCursor, total };
}
const whereB: Prisma.ArtworkWhereInput = {
AND: [baseWhere, { sortKey: null }],
...(cursor ? { id: { gt: cursor.afterId } } : {}),
};
const rowsB = await prisma.artwork.findMany({
where: whereB,
orderBy: [{ id: "asc" }],
take: Math.min(take, 200),
select,
});
items = rowsB.map(mapRow).filter((x): x is TaggedArtworkItem => x !== null);
const last = items[items.length - 1];
nextCursor =
items.length < take || !last
? null
: { afterSortKey: null, afterId: last.id };
return { items, nextCursor, total };
}

View File

@ -1,4 +1,4 @@
import { ArrowLeftIcon } from "lucide-react"; import { ArrowLeftIcon, ChevronRightIcon } from "lucide-react";
import Link from "next/link"; import Link from "next/link";
import { import {
@ -30,14 +30,22 @@ function sortArtworks(a: SimpleArtwork, b: SimpleArtwork) {
} }
export default async function AnimalListPage() { export default async function AnimalListPage() {
const tags = await prisma.artTag.findMany({ const tags = await prisma.tag.findMany({
where: { showOnAnimalPage: true }, where: {
isVisible: true,
categoryLinks: {
some: { category: { name: "Animal Studies" }, showOnAnimalPage: true },
},
},
select: { select: {
id: true, id: true,
name: true, name: true,
slug: true, slug: true,
sortIndex: true, sortIndex: true,
parentId: true, categoryLinks: {
where: { category: { name: "Animal Studies" } },
select: { parentTagId: true },
},
artworks: { artworks: {
where: { where: {
published: true, published: true,
@ -50,10 +58,15 @@ export default async function AnimalListPage() {
orderBy: [{ sortIndex: "asc" }, { name: "asc" }], orderBy: [{ sortIndex: "asc" }, { name: "asc" }],
}); });
const byId = new Map(tags.map((t) => [t.id, t])); const tagsWithParents = tags.map((t) => ({
const childrenByParentId = new Map<string, typeof tags>(); ...t,
parentId: t.categoryLinks[0]?.parentTagId ?? null,
}));
for (const t of tags) { const byId = new Map(tagsWithParents.map((t) => [t.id, t]));
const childrenByParentId = new Map<string, typeof tagsWithParents>();
for (const t of tagsWithParents) {
if (!t.parentId) continue; if (!t.parentId) continue;
const arr = childrenByParentId.get(t.parentId) ?? []; const arr = childrenByParentId.get(t.parentId) ?? [];
arr.push(t); arr.push(t);
@ -64,12 +77,12 @@ export default async function AnimalListPage() {
childrenByParentId.set(pid, arr.slice().sort(sortBySortIndexName)); childrenByParentId.set(pid, arr.slice().sort(sortBySortIndexName));
} }
const parents = tags const parents = tagsWithParents
.filter((t) => t.parentId === null) .filter((t) => t.parentId === null)
.slice() .slice()
.sort(sortBySortIndexName); .sort(sortBySortIndexName);
const orphans = tags const orphans = tagsWithParents
.filter((t) => t.parentId !== null && !byId.has(t.parentId)) .filter((t) => t.parentId !== null && !byId.has(t.parentId))
.slice() .slice()
.sort(sortBySortIndexName); .sort(sortBySortIndexName);
@ -82,16 +95,17 @@ export default async function AnimalListPage() {
} }
return ( return (
<ul className="space-y-1.5"> <ul className="space-y-1">
{list.map((a) => ( {list.map((a) => (
<li key={a.id}> <li key={a.id}>
<Link <Link
href={`/artworks/single/${a.id}?from=animal-index`} href={`/artworks/single/${a.id}?from=animal-index`}
className=" className="
inline-flex items-center gap-2 group flex w-full items-center gap-2
rounded-md px-2 py-1 rounded-md px-2 py-1.5
text-sm font-medium text-sm font-medium
hover:bg-muted hover:bg-muted/60
focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2
" "
> >
<span <span
@ -102,7 +116,7 @@ export default async function AnimalListPage() {
group-hover:translate-x-0.5 group-hover:translate-x-0.5
" "
> >
<ChevronRightIcon className="h-4 w-4" />
</span> </span>
<span className="leading-snug">{a.name}</span> <span className="leading-snug">{a.name}</span>
@ -149,7 +163,7 @@ export default async function AnimalListPage() {
</div> </div>
</header> </header>
<div className="space-y-6"> <div className="space-y-6 sm:space-y-4">
<Card> <Card>
<CardHeader className="space-y-1"> <CardHeader className="space-y-1">
<CardTitle className="text-base"> <CardTitle className="text-base">
@ -177,15 +191,17 @@ export default async function AnimalListPage() {
const isStandalone = children.length === 0; const isStandalone = children.length === 0;
return ( return (
<AccordionItem key={p.id} value={p.id} className="border-b"> <AccordionItem key={p.id} value={p.id} className="py-1 sm:py-1">
<AccordionTrigger <AccordionTrigger
className=" className="
py-4 py-4 sm:py-3
rounded-md px-2 -mx-2 rounded-md px-2 -mx-2
bg-hover text-hover-foreground dark:bg-hover dark:text-hover-foreground
transition-colors transition-colors
hover:bg-muted/60 hover:bg-hover/80 dark:hover:bg-hover/80
font-semibold
focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2
data-[state=open]:bg-muted/40 data-[state=open]:bg-muted/90 dark:data-[state=open]:bg-muted/90
" "
> >
<div className="flex w-full items-center justify-between pr-2"> <div className="flex w-full items-center justify-between pr-2">
@ -211,41 +227,44 @@ export default async function AnimalListPage() {
</div> </div>
</AccordionTrigger> </AccordionTrigger>
<AccordionContent className="pb-5"> <AccordionContent className="pb-4 sm:pb-4">
{isStandalone ? ( {isStandalone ? (
<div className="rounded-md border p-4"> <div className="space-y-2">
<div className="mb-3 flex items-center justify-between"> <div className="flex items-center justify-between text-sm font-medium">
<div className="text-sm font-medium">Artworks</div> <span>Artworks</span>
<Badge variant="outline">{p.artworks.length}</Badge> <Badge variant="outline">{p.artworks.length}</Badge>
</div> </div>
<div>
<ArtworkList items={p.artworks} /> <ArtworkList items={p.artworks} />
</div> </div>
</div>
) : ( ) : (
<div className="space-y-4"> <div className="space-y-4 sm:space-y-3">
{p.artworks.length > 0 ? ( {p.artworks.length > 0 ? (
<> <>
<div className="rounded-md border p-4"> <div className="space-y-2">
<div className="mb-3 flex items-center justify-between"> <div className="flex items-center justify-between text-sm font-medium">
<div className="text-sm font-medium">{/* Directly tagged */}</div> <span>Direct artworks</span>
<Badge variant="outline">{p.artworks.length}</Badge> <Badge variant="outline">{p.artworks.length}</Badge>
</div> </div>
<div>
<ArtworkList items={p.artworks} /> <ArtworkList items={p.artworks} />
</div> </div>
</div>
<Separator /> <Separator />
</> </>
) : null} ) : null}
<div className="grid grid-cols-1 gap-3 sm:grid-cols-2"> <div className="grid grid-cols-1 gap-4 sm:gap-2 sm:grid-cols-2">
{children.map((c) => ( {children.map((c) => (
<div key={c.id} className="rounded-md border p-4"> <div key={c.id} className="space-y-2 pt-3">
<div className="mb-3 flex items-center justify-between"> <div className="flex items-center justify-between text-sm font-medium">
<div className="min-w-0"> <div className="min-w-0">
<div className="truncate text-sm font-medium">{c.name}</div> <div className="truncate">{c.name}</div>
</div> </div>
<Badge variant="outline">{c.artworks.length}</Badge> <Badge variant="outline">{c.artworks.length}</Badge>
</div> </div>
<ArtworkList items={c.artworks} /> <ArtworkList items={c.artworks} />
</div> </div>
))} ))}
@ -269,12 +288,12 @@ export default async function AnimalListPage() {
Tags whose parent is not visible (or not configured for the animal page). Tags whose parent is not visible (or not configured for the animal page).
</p> */} </p> */}
</CardHeader> </CardHeader>
<CardContent className="space-y-3"> <CardContent className="space-y-4 sm:space-y-2">
<div className="grid grid-cols-1 gap-3 sm:grid-cols-2"> <div className="grid grid-cols-1 gap-4 sm:gap-2 sm:grid-cols-2">
{orphans.map((t) => ( {orphans.map((t) => (
<div key={t.id} className="rounded-md border p-4"> <div key={t.id} className="space-y-2 pt-3">
<div className="mb-3 flex items-center justify-between"> <div className="flex items-center justify-between text-sm font-medium">
<div className="truncate text-sm font-medium">{t.name}</div> <div className="truncate">{t.name}</div>
<Badge variant="outline">{t.artworks.length}</Badge> <Badge variant="outline">{t.artworks.length}</Badge>
</div> </div>
<ArtworkList items={t.artworks} /> <ArtworkList items={t.artworks} />

View File

@ -17,17 +17,28 @@ function parseTagsParam(tags: string | string[] | undefined): string[] {
function expandSelectedWithChildren( function expandSelectedWithChildren(
selectedSlugs: string[], selectedSlugs: string[],
tagsForFilter: Array<{ tagsForFilter: Array<{
id: string;
slug: string; slug: string;
children: Array<{ slug: string }>; parentId: string | null;
}>, }>,
) { ) {
const bySlug = new Map(tagsForFilter.map((t) => [t.slug, t])); const bySlug = new Map(tagsForFilter.map((t) => [t.slug, t]));
const childrenByParentId = new Map<string, typeof tagsForFilter>();
for (const t of tagsForFilter) {
if (!t.parentId) continue;
const arr = childrenByParentId.get(t.parentId) ?? [];
arr.push(t);
childrenByParentId.set(t.parentId, arr);
}
const out = new Set(selectedSlugs); const out = new Set(selectedSlugs);
for (const slug of selectedSlugs) { for (const slug of selectedSlugs) {
const t = bySlug.get(slug); const t = bySlug.get(slug);
if (!t) continue; if (!t) continue;
for (const c of t.children ?? []) out.add(c.slug); const children = childrenByParentId.get(t.id) ?? [];
for (const c of children) out.add(c.slug);
} }
return Array.from(out); return Array.from(out);
@ -41,24 +52,31 @@ export default async function AnimalStudiesPage({
const { tags } = await searchParams; const { tags } = await searchParams;
const selectedTagSlugs = parseTagsParam(tags); const selectedTagSlugs = parseTagsParam(tags);
const tagsForFilter = await prisma.artTag.findMany({ const tagLinks = await prisma.tagCategory.findMany({
where: { showOnAnimalPage: true }, where: {
showOnAnimalPage: true,
category: { name: "Animal Studies" },
tag: { isVisible: true },
},
select: { select: {
id: true, parentTagId: true,
name: true, tag: {
slug: true, select: { id: true, name: true, slug: true, sortIndex: true },
sortIndex: true,
parentId: true,
parent: { select: { id: true, name: true, slug: true, sortIndex: true } },
children: {
where: { showOnAnimalPage: true },
select: { id: true, name: true, slug: true, sortIndex: true, parentId: true },
orderBy: [{ sortIndex: "asc" }, { name: "asc" }],
}, },
}, },
orderBy: [{ sortIndex: "asc" }, { name: "asc" }], orderBy: [{ tag: { sortIndex: "asc" } }, { tag: { name: "asc" } }],
}); });
const tagsForFilter = tagLinks.map((link) => ({
id: link.tag.id,
name: link.tag.name,
slug: link.tag.slug,
sortIndex: link.tag.sortIndex,
parentId: link.parentTagId,
parent: null,
children: [],
}));
const expandedTagSlugs = expandSelectedWithChildren(selectedTagSlugs, tagsForFilter); const expandedTagSlugs = expandSelectedWithChildren(selectedTagSlugs, tagsForFilter);
return ( return (

View File

@ -8,7 +8,12 @@ import { PlayCircle } from "lucide-react";
import Image from "next/image"; import Image from "next/image";
import Link from "next/link"; import Link from "next/link";
export default async function SingleArtworkPage({ params }: { params: { id: string }; searchParams: Record<string, string | string[] | undefined>; }) { export default async function SingleArtworkPage({
params,
}: {
params: { id: string };
searchParams: Record<string, string | string[] | undefined>;
}) {
const { id } = await params; const { id } = await params;
const artwork = await prisma.artwork.findUnique({ const artwork = await prisma.artwork.findUnique({
where: { where: {
@ -24,34 +29,44 @@ export default async function SingleArtworkPage({ params }: { params: { id: stri
tags: true, tags: true,
variants: true, variants: true,
timelapse: { where: { enabled: true } }, timelapse: { where: { enabled: true } },
} },
}) });
if (!artwork) return <div>Artwork with this ID could not be found</div> if (!artwork) return <div>Artwork with this ID could not be found</div>;
const { width, height } = artwork.variants.find((v) => v.type === "resized") ?? { width: 0, height: 0 } const { width, height } = artwork.variants.find(
(v) => v.type === "resized",
) ?? { width: 0, height: 0 };
const colors = const colors =
artwork.colors?.map((c) => c.color?.hex).filter((hex): hex is string => Boolean(hex)) ?? [] artwork.colors
?.map((c) => c.color?.hex)
.filter((hex): hex is string => Boolean(hex)) ?? [];
const gradientColors = colors.length const gradientColors = colors.length
? colors.join(", ") ? colors.join(", ")
: "rgba(0,0,0,0.1), rgba(0,0,0,0.03)" : "rgba(0,0,0,0.1), rgba(0,0,0,0.03)";
return ( return (
<div className="px-8 py-4"> <div className="px-4 sm:px-8 py-4">
<div className="relative w-full min-h-10 flex items-center mb-4"> <div className="relative w-full min-h-10 flex items-center mb-4">
<div className="z-10"><ContextBackButton /></div> <div className="z-10 hidden sm:block">
<ContextBackButton />
</div>
{artwork.name ? ( {artwork.name ? (
<div className="pointer-events-none absolute left-1/2 -translate-x-1/2 text-center"> <div className="w-full text-center sm:pointer-events-none sm:absolute sm:left-1/2 sm:-translate-x-1/2">
<div className="pointer-events-auto"><h1 className="text-2xl font-bold mb-4 py-4">{artwork.name}</h1></div> <div className="sm:pointer-events-auto">
<h1 className="text-xl sm:text-2xl font-bold mb-2 sm:mb-4 py-2 sm:py-4 px-2 sm:px-0 wrap-break-word">
{artwork.name}
</h1>
</div>
</div> </div>
) : null} ) : null}
</div> </div>
<div className="flex flex-col items-center gap-4"> <div className="flex flex-col items-center gap-4">
<div className="group rounded-lg border overflow-hidden hover:shadow-lg transition-shadow bg-background relative"> <div className="group rounded-lg border overflow-hidden hover:shadow-lg transition-shadow bg-background relative">
<div className="relative w-full bg-muted items-center justify-center" <div
className="relative w-full bg-muted items-center justify-center"
style={{ aspectRatio: "4 / 3" }} style={{ aspectRatio: "4 / 3" }}
> >
<Link href={`/raw/${artwork.id}`}> <Link href={`/raw/${artwork.id}`}>
@ -94,6 +109,9 @@ export default async function SingleArtworkPage({ params }: { params: { id: stri
tags={artwork.tags} tags={artwork.tags}
/> />
</div> </div>
<div className="w-full flex justify-center sm:hidden">
<ContextBackButton className="mx-auto flex justify-center" />
</div>
</div> </div>
</div> </div>
); );

View File

@ -0,0 +1,48 @@
import TaggedGallery from "@/components/portfolio/TaggedGallery";
import { prisma } from "@/lib/prisma";
function parseTagsParam(tags: string | string[] | undefined): string[] {
if (!tags) return [];
const raw = Array.isArray(tags) ? tags.join(",") : tags;
return raw
.split(",")
.map((s) => s.trim())
.filter(Boolean);
}
export default async function TaggedPortfolioPage({
searchParams,
}: {
searchParams: { tags?: string | string[] };
}) {
const { tags } = await searchParams;
const selectedTagSlugs = parseTagsParam(tags);
const tagsSelected = selectedTagSlugs.length
? await prisma.tag.findMany({
where: { slug: { in: selectedTagSlugs } },
select: { id: true, name: true, slug: true },
orderBy: [{ sortIndex: "asc" }, { name: "asc" }],
})
: [];
return (
<div className="mx-auto w-full max-w-6xl px-4 py-8">
<header className="mb-6 flex flex-col gap-4 sm:flex-row sm:items-end sm:justify-between">
<div className="space-y-1">
<h1 className="text-2xl font-semibold tracking-tight sm:text-3xl">
{tagsSelected.length ? (
<div className="flex flex-wrap gap-2">
List of artworks tagged with:
{tagsSelected.map((t) => (
<span key={t.name.toLowerCase()}> {t.name.toLowerCase()}</span>
))}
</div>
) : "No tags selected"}
</h1>
</div>
</header>
<TaggedGallery tagSlugs={selectedTagSlugs} />
</div>
);
}

View File

@ -19,7 +19,11 @@ export default async function CommissionsPage() {
include: { include: {
options: { include: { option: true }, orderBy: { sortIndex: "asc" } }, options: { include: { option: true }, orderBy: { sortIndex: "asc" } },
extras: { include: { extra: true }, orderBy: { sortIndex: "asc" } }, extras: { include: { extra: true }, orderBy: { sortIndex: "asc" } },
customInputs: { include: { customInput: true }, orderBy: { sortIndex: "asc" } }, customInputs: {
include: { customInput: true },
orderBy: { sortIndex: "asc" },
},
tags: true,
}, },
orderBy: [{ sortIndex: "asc" }, { name: "asc" }], orderBy: [{ sortIndex: "asc" }, { name: "asc" }],
}), }),
@ -28,6 +32,7 @@ export default async function CommissionsPage() {
include: { include: {
options: { include: { option: true }, orderBy: { sortIndex: "asc" } }, options: { include: { option: true }, orderBy: { sortIndex: "asc" } },
extras: { include: { extra: true }, orderBy: { sortIndex: "asc" } }, extras: { include: { extra: true }, orderBy: { sortIndex: "asc" } },
tags: true,
}, },
orderBy: [{ sortIndex: "asc" }, { name: "asc" }], orderBy: [{ sortIndex: "asc" }, { name: "asc" }],
}), }),
@ -49,7 +54,7 @@ export default async function CommissionsPage() {
{guidelines?.exampleImageUrl ? ( {guidelines?.exampleImageUrl ? (
<Dialog> <Dialog>
<DialogTrigger asChild> <DialogTrigger asChild>
<Button variant="secondary">View example</Button> <Button variant="secondary">View type example</Button>
</DialogTrigger> </DialogTrigger>
<DialogContent className="flex w-auto! max-w-[95vw]! flex-col p-4 sm:p-6"> <DialogContent className="flex w-auto! max-w-[95vw]! flex-col p-4 sm:p-6">
<DialogHeader className="sr-only"> <DialogHeader className="sr-only">
@ -80,7 +85,10 @@ export default async function CommissionsPage() {
<CommissionGuidelines /> <CommissionGuidelines />
</div> </div>
<hr /> <hr />
<h2 id="commission-request-form" className="text-2xl font-semibold scroll-mt-24"> <h2
id="commission-request-form"
className="text-2xl font-semibold scroll-mt-24"
>
Request a Commission Request a Commission
</h2> </h2>
<CommissionOrderForm types={commissions} customCards={customCards} /> <CommissionOrderForm types={commissions} customCards={customCards} />

View File

@ -2,9 +2,12 @@ import { Badge } from "@/components/ui/badge";
import { prisma } from "@/lib/prisma"; import { prisma } from "@/lib/prisma";
const statusStyles: Record<string, string> = { const statusStyles: Record<string, string> = {
ACCEPTED: "bg-sky-500/15 text-sky-300 border-sky-500/30", ACCEPTED:
INPROGRESS: "bg-amber-500/15 text-amber-300 border-amber-500/30", "bg-sky-500/20 text-sky-700 border-sky-500/40 dark:bg-sky-500/15 dark:text-sky-300 dark:border-sky-500/30",
COMPLETED: "bg-emerald-500/15 text-emerald-300 border-emerald-500/30", INPROGRESS:
"bg-amber-500/20 text-amber-700 border-amber-500/40 dark:bg-amber-500/15 dark:text-amber-300 dark:border-amber-500/30",
COMPLETED:
"bg-emerald-500/20 text-emerald-700 border-emerald-500/40 dark:bg-emerald-500/15 dark:text-emerald-300 dark:border-emerald-500/30",
}; };
const statusLabels: Record<string, string> = { const statusLabels: Record<string, string> = {

View File

@ -105,6 +105,8 @@
--border: oklch(0.3289 0.0092 268.3843); --border: oklch(0.3289 0.0092 268.3843);
--input: oklch(0.3289 0.0092 268.3843); --input: oklch(0.3289 0.0092 268.3843);
--ring: oklch(0.6132 0.2294 291.7437); --ring: oklch(0.6132 0.2294 291.7437);
--hover: oklch(0.34 0.02 270);
--hover-foreground: var(--foreground);
--chart-1: oklch(0.8003 0.1821 151.7110); --chart-1: oklch(0.8003 0.1821 151.7110);
--chart-2: oklch(0.6132 0.2294 291.7437); --chart-2: oklch(0.6132 0.2294 291.7437);
--chart-3: oklch(0.8077 0.1035 19.5706); --chart-3: oklch(0.8077 0.1035 19.5706);

View File

@ -10,7 +10,7 @@ const FROM_TO_PATH: Record<string, string> = {
"animal-index": "/artworks/animalstudies/index" "animal-index": "/artworks/animalstudies/index"
}; };
export function ContextBackButton() { export function ContextBackButton({ className }: { className?: string }) {
const router = useRouter(); const router = useRouter();
const sp = useSearchParams(); const sp = useSearchParams();
const from = sp.get("from") ?? ""; const from = sp.get("from") ?? "";
@ -19,7 +19,7 @@ export function ContextBackButton() {
if (!target) return null; if (!target) return null;
return ( return (
<div className="w-full max-w-xl"> <div className={["w-full max-w-xl", className].filter(Boolean).join(" ")}>
<Link <Link
href={target} href={target}
className={[ className={[

View File

@ -1,52 +1,43 @@
"use client" "use client";
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card" import { Button } from "@/components/ui/button";
import type { CommissionExtra, CommissionOption, CommissionType, CommissionTypeExtra, CommissionTypeOption } from "@/generated/prisma/client" import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
import type {
CommissionExtra,
CommissionOption,
CommissionType,
CommissionTypeExtra,
CommissionTypeOption,
Tag,
} from "@/generated/prisma/client";
import Link from "next/link";
type CommissionTypeWithItems = CommissionType & { type CommissionTypeWithItems = CommissionType & {
options: (CommissionTypeOption & { options: (CommissionTypeOption & {
option: CommissionOption | null option: CommissionOption | null;
})[] })[];
extras: (CommissionTypeExtra & { extras: (CommissionTypeExtra & {
extra: CommissionExtra | null extra: CommissionExtra | null;
})[] })[];
} tags: Tag[];
};
export function CommissionCard({ commission }: { commission: CommissionTypeWithItems }) {
// const [open, setOpen] = useState(false)
export function CommissionCard({
commission,
}: {
commission: CommissionTypeWithItems;
}) {
return ( return (
<div className="flex flex-col h-full"> <div className="flex flex-col h-full">
<Card className="flex flex-col flex-1"> <Card className="flex flex-col flex-1">
<CardHeader> <CardHeader>
<CardTitle className="text-xl font-bold">{commission.name}</CardTitle> <CardTitle className="text-xl font-bold">{commission.name}</CardTitle>
<p className="text-muted-foreground text-sm">{commission.description}</p> <p className="text-muted-foreground text-sm">
{commission.description}
</p>
</CardHeader> </CardHeader>
<CardContent className="flex flex-col justify-start gap-4"> <CardContent className="flex flex-col flex-1 justify-start gap-4">
{/* {examples && examples.length > 0 && (
<Collapsible open={open} onOpenChange={setOpen}>
<CollapsibleTrigger className="text-sm underline text-muted-foreground">
{open ? "Hide Examples" : "See Examples"}
</CollapsibleTrigger>
<CollapsibleContent asChild>
<div className="overflow-hidden transition-all data-[state=closed]:max-h-0 data-[state=open]:max-h-[300px]">
<div className="flex gap-2 mt-2 overflow-x-auto">
{examples.map((src, idx) => (
<Image
key={src + idx}
src={src}
width={100}
height={100}
alt={`${type.name} example ${idx + 1}`}
className="h-24 w-auto rounded border"
/>
))}
</div>
</div>
</CollapsibleContent>
</Collapsible>
)} */}
<div> <div>
<h4 className="font-semibold">Options</h4> <h4 className="font-semibold">Options</h4>
<ul className="pl-4 list-disc"> <ul className="pl-4 list-disc">
@ -66,7 +57,9 @@ export function CommissionCard({ commission }: { commission: CommissionTypeWithI
</div> </div>
<div> <div>
{commission.extras.length > 0 && <h4 className="font-semibold">Extras</h4>} {commission.extras.length > 0 && (
<h4 className="font-semibold">Extras</h4>
)}
<ul className="pl-4 list-disc"> <ul className="pl-4 list-disc">
{commission.extras.map((extra) => ( {commission.extras.map((extra) => (
<li key={extra.id}> <li key={extra.id}>
@ -82,16 +75,21 @@ export function CommissionCard({ commission }: { commission: CommissionTypeWithI
))} ))}
</ul> </ul>
</div> </div>
{/* <div className="flex flex-wrap gap-2">
{commission.extras.map((extra) => (
<Badge variant="outline" key={extra.id}>
{extra.extra?.name}
</Badge>
))}
</div> */}
</CardContent> </CardContent>
{commission.tags.length > 0 ? (
<div className="mt-auto px-6 pb-6">
<Link
href={`/portfolio/tagged?tags=${encodeURIComponent(
commission.tags.map((t) => t.slug).join(","),
)}`}
>
<Button variant="secondary" className="w-full">
View example artworks
</Button>
</Link>
</div>
) : null}
</Card> </Card>
</div> </div>
) );
} }

View File

@ -1,5 +1,6 @@
"use client"; "use client";
import { Button } from "@/components/ui/button";
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"; import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
import { import {
Dialog, Dialog,
@ -8,8 +9,10 @@ import {
DialogTitle, DialogTitle,
DialogTrigger, DialogTrigger,
} from "@/components/ui/dialog"; } from "@/components/ui/dialog";
import type { Tag } from "@/generated/prisma/client";
import { cn } from "@/lib/utils"; import { cn } from "@/lib/utils";
import Image from "next/image"; import Image from "next/image";
import Link from "next/link";
type CustomCardOption = { type CustomCardOption = {
id: string; id: string;
@ -33,6 +36,7 @@ export type CommissionCustomCardWithItems = {
description: string | null; description: string | null;
referenceImageUrl: string | null; referenceImageUrl: string | null;
isSpecialOffer: boolean; isSpecialOffer: boolean;
tags: Tag[];
options: CustomCardOption[]; options: CustomCardOption[];
extras: CustomCardExtra[]; extras: CustomCardExtra[];
}; };
@ -50,7 +54,7 @@ export function CommissionCustomCard({
"flex flex-col h-full relative shadow-sm", "flex flex-col h-full relative shadow-sm",
card.isSpecialOffer card.isSpecialOffer
? "border-2 border-primary/50" ? "border-2 border-primary/50"
: "border-border" : "border-border",
)} )}
> >
{card.isSpecialOffer ? ( {card.isSpecialOffer ? (
@ -143,6 +147,19 @@ export function CommissionCustomCard({
</ul> </ul>
</div> </div>
</CardContent> </CardContent>
{card.tags.length > 0 ? (
<div className="mt-auto px-6 pb-6">
<Link
href={`/portfolio/tagged?tags=${encodeURIComponent(
card.tags.map((t) => t.slug).join(","),
)}`}
>
<Button variant="secondary" className="w-full">
View example artworks
</Button>
</Link>
</div>
) : null}
</Card> </Card>
</div> </div>
</div> </div>

View File

@ -44,6 +44,10 @@ type Props = {
maxRowItems?: number; // desktop maxRowItems?: number; // desktop
maxRowItemsMobile?: number; // <640px maxRowItemsMobile?: number; // <640px
gap?: number; // px gap?: number; // px
gapNarrow?: number; // px for narrower containers
gapNarrowMaxWidth?: number; // px breakpoint for gapNarrow
gapBreakpoints?: Array<{ maxWidth: number; gap: number }>;
debug?: boolean;
className?: string; className?: string;
}; };
@ -84,11 +88,33 @@ export default function JustifiedGallery({
maxRowItems = 5, maxRowItems = 5,
maxRowItemsMobile = 3, maxRowItemsMobile = 3,
gap = 12, gap = 12,
gapNarrow,
gapNarrowMaxWidth = 720,
gapBreakpoints,
debug = false,
className, className,
}: Props) { }: Props) {
const containerRef = useRef<HTMLDivElement | null>(null); const containerRef = useRef<HTMLDivElement | null>(null);
const sentinelRef = useRef<HTMLDivElement | null>(null); const sentinelRef = useRef<HTMLDivElement | null>(null);
const [containerWidth, setContainerWidth] = useState(0); const [containerWidth, setContainerWidth] = useState(0);
const effectiveGap = (() => {
if (gapBreakpoints && containerWidth > 0) {
const sorted = [...gapBreakpoints].sort((a, b) => a.maxWidth - b.maxWidth);
for (const bp of sorted) {
if (containerWidth <= bp.maxWidth) return bp.gap;
}
}
if (
gapNarrow != null &&
containerWidth > 0 &&
containerWidth <= gapNarrowMaxWidth
) {
return gapNarrow;
}
return gap;
})();
// Measure container width (responsive) // Measure container width (responsive)
useEffect(() => { useEffect(() => {
@ -125,7 +151,12 @@ export default function JustifiedGallery({
const isMobile = containerWidth < 640; const isMobile = containerWidth < 640;
const targetH = isMobile ? targetRowHeightMobile : targetRowHeight; const targetH = isMobile ? targetRowHeightMobile : targetRowHeight;
const maxItems = isMobile ? maxRowItemsMobile : maxRowItems; const maxItems = (() => {
if (containerWidth < 480) return Math.min(2, maxRowItemsMobile);
if (containerWidth < 720) return Math.min(3, maxRowItems);
if (containerWidth < 1024) return Math.min(4, maxRowItems);
return maxRowItems;
})();
const rowTiles: RowTile[][] = []; const rowTiles: RowTile[][] = [];
let current: Array<{ item: JustifiedGalleryItem; aspect: number }> = []; let current: Array<{ item: JustifiedGalleryItem; aspect: number }> = [];
@ -136,7 +167,7 @@ export default function JustifiedGallery({
const flush = () => { const flush = () => {
if (current.length === 0) return; if (current.length === 0) return;
const gaps = gap * (current.length - 1); const gaps = effectiveGap * (current.length - 1);
const widthWithoutGaps = Math.max(0, available - gaps); const widthWithoutGaps = Math.max(0, available - gaps);
// Compute row height so it exactly fills the row width. // Compute row height so it exactly fills the row width.
@ -155,21 +186,64 @@ export default function JustifiedGallery({
aspectSum = 0; aspectSum = 0;
}; };
for (const it of items) { const workingItems = items.slice();
for (let i = 0; i < workingItems.length; i += 1) {
const it = workingItems[i];
const a = aspectOf(it); const a = aspectOf(it);
current.push({ item: it, aspect: a }); current.push({ item: it, aspect: a });
aspectSum += a; aspectSum += a;
// Estimate the row width if we were to keep targetH // Estimate the row width if we were to keep targetH
const estimatedWidth = aspectSum * targetH + gap * (current.length - 1); const estimatedWidth =
aspectSum * targetH + effectiveGap * (current.length - 1);
// If we've filled the row (or reached max items) and have at least 2 tiles, flush. // If we've filled the row (or reached max items) and have at least 2 tiles, flush.
if ( if (
(estimatedWidth >= available || current.length >= maxItems) && (estimatedWidth >= available || current.length >= maxItems) &&
current.length > 1 current.length > 1
) { ) {
const gaps = effectiveGap * (current.length - 1);
const widthWithoutGaps = Math.max(0, available - gaps);
const computedH = widthWithoutGaps / aspectSum;
// If the row would be shorter than maxRowHeight, reduce items and flush.
if (computedH < maxRowHeight && current.length > 1) {
const last = current.pop();
if (last) {
aspectSum -= last.aspect;
}
const limit = widthWithoutGaps / maxRowHeight;
let swapped = false;
for (let look = 1; look <= 2; look += 1) {
const idx = i + look;
if (idx >= workingItems.length) break;
const candidate = workingItems[idx];
const candidateAspect = aspectOf(candidate);
if (aspectSum + candidateAspect <= limit) {
workingItems[idx] = last?.item ?? candidate;
current.push({ item: candidate, aspect: candidateAspect });
aspectSum += candidateAspect;
swapped = true;
break;
}
}
flush(); flush();
if (!swapped && last) {
current = [last];
aspectSum = last.aspect;
} else {
current = [];
aspectSum = 0;
}
} else {
flush();
}
} }
} }
@ -178,7 +252,7 @@ export default function JustifiedGallery({
}, [ }, [
items, items,
containerWidth, containerWidth,
gap, effectiveGap,
targetRowHeight, targetRowHeight,
targetRowHeightMobile, targetRowHeightMobile,
maxRowHeight, maxRowHeight,
@ -192,17 +266,26 @@ export default function JustifiedGallery({
return `${first}-${last}-${row.length}`; return `${first}-${last}-${row.length}`;
}, []); }, []);
const isSmallScreen = containerWidth < 640;
return ( return (
<div <div
ref={containerRef} ref={containerRef}
className={cn("mx-auto w-full max-w-6xl", className)} className={cn("mx-auto w-full max-w-6xl", className)}
> >
<div className="space-y-3"> <div className="space-y-3">
{rows.map((row) => ( {rows.map((row, idx) => (
<div key={getRowKey(row)}>
<div <div
key={getRowKey(row)} className={cn(
className="flex justify-center" "flex",
style={{ gap }} row.length === 1 && (isSmallScreen || idx !== rows.length - 1)
? "justify-center"
: idx === rows.length - 1
? "justify-start"
: "justify-between",
)}
style={{ columnGap: effectiveGap }}
> >
{row.map((t) => ( {row.map((t) => (
<GalleryTile <GalleryTile
@ -214,6 +297,21 @@ export default function JustifiedGallery({
/> />
))} ))}
</div> </div>
{debug ? (
<div className="text-xs text-muted-foreground font-mono">
{`row ${idx + 1} | h=${Math.round(row[0]?.h ?? 0)} | w=${Math.round(
row.reduce((sum, t) => sum + t.w, 0) +
effectiveGap * (row.length - 1),
)} | items=${row.length} | `}
{row
.map(
(t) =>
`${t.item.id}:${Math.round(t.w)}x${Math.round(t.h)} (src ${t.item.width}x${t.item.height})`,
)
.join(" | ")}
</div>
) : null}
</div>
))} ))}
</div> </div>

View File

@ -31,6 +31,12 @@ export default function PortfolioGallery({
const [items, setItems] = useState<PortfolioArtworkItem[]>([]); const [items, setItems] = useState<PortfolioArtworkItem[]>([]);
const [done, setDone] = useState(false); const [done, setDone] = useState(false);
const [loading, setLoading] = useState(false); const [loading, setLoading] = useState(false);
const showScreenDebug = false;
const [screen, setScreen] = useState<{
width: number;
height: number;
dpr: number;
} | null>(null);
const inFlight = useRef(false); const inFlight = useRef(false);
const doneRef = useRef(false); const doneRef = useRef(false);
@ -80,6 +86,26 @@ export default function PortfolioGallery({
void loadMore(); void loadMore();
}, [loadMore]); }, [loadMore]);
useEffect(() => {
if (!showScreenDebug) return;
const update = () => {
setScreen({
width: window.innerWidth,
height: window.innerHeight,
dpr: window.devicePixelRatio || 1,
});
};
update();
window.addEventListener("resize", update);
window.addEventListener("orientationchange", update);
return () => {
window.removeEventListener("resize", update);
window.removeEventListener("orientationchange", update);
};
}, []);
const galleryItems: JustifiedGalleryItem[] = items.map((it) => ({ const galleryItems: JustifiedGalleryItem[] = items.map((it) => ({
id: it.id, id: it.id,
name: it.name, name: it.name,
@ -90,17 +116,17 @@ export default function PortfolioGallery({
dominantHex: it.dominantHex, dominantHex: it.dominantHex,
})); }));
useEffect(() => { // useEffect(() => {
if (items.length === 0) return; // if (items.length === 0) return;
// Debug: inspect dominantHex values coming from the server. // // Debug: inspect dominantHex values coming from the server.
console.log( // console.log(
"[PortfolioGallery] dominantHex sample", // "[PortfolioGallery] dominantHex sample",
items.slice(0, 5).map((it) => ({ // items.slice(0, 5).map((it) => ({
id: it.id, // id: it.id,
dominantHex: it.dominantHex, // dominantHex: it.dominantHex,
})) // }))
); // );
}, [items]); // }, [items]);
if (!loading && done && galleryItems.length === 0) { if (!loading && done && galleryItems.length === 0) {
return ( return (
@ -111,17 +137,28 @@ export default function PortfolioGallery({
} }
return ( return (
<div className="w-full"> <div className="relative w-full">
{showScreenDebug && screen ? (
<div className="pointer-events-none absolute right-2 top-2 z-10 rounded border border-border/60 bg-background/80 px-2 py-1 text-[11px] font-mono text-muted-foreground shadow-sm">
Screen {screen.width} × {screen.height} · {screen.dpr}x
</div>
) : null}
<JustifiedGallery <JustifiedGallery
items={galleryItems} items={galleryItems}
hrefFrom="portfolio" hrefFrom="portfolio"
showCaption={false} showCaption={false}
debug={false}
targetRowHeight={160} targetRowHeight={160}
targetRowHeightMobile={160} targetRowHeightMobile={160}
maxRowHeight={300} maxRowHeight={300}
maxRowItems={5} maxRowItems={5}
maxRowItemsMobile={1} maxRowItemsMobile={1}
gap={12} gap={12}
gapBreakpoints={[
{ maxWidth: 685, gap: 6 },
{ maxWidth: 910, gap: 8 },
{ maxWidth: 1130, gap: 10 },
]}
onLoadMore={done ? undefined : () => void loadMore()} onLoadMore={done ? undefined : () => void loadMore()}
hasMore={!done} hasMore={!done}
isLoadingMore={loading} isLoadingMore={loading}

View File

@ -0,0 +1,116 @@
"use client";
import type {
Cursor,
TaggedArtworkItem,
} from "@/actions/portfolio/getTaggedArtworksPage";
import { getTaggedArtworksPage } from "@/actions/portfolio/getTaggedArtworksPage";
import JustifiedGallery, {
type JustifiedGalleryItem,
} from "@/components/gallery/JustifiedGallery";
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
export default function TaggedGallery({ tagSlugs }: { tagSlugs: string[] }) {
const normalizedSlugs = useMemo(
() => tagSlugs.map((s) => s.trim()).filter(Boolean),
[tagSlugs],
);
const resetKey = useMemo(
() => normalizedSlugs.slice().sort().join(","),
[normalizedSlugs],
);
const [items, setItems] = useState<TaggedArtworkItem[]>([]);
const [done, setDone] = useState(false);
const [loading, setLoading] = useState(false);
const inFlight = useRef(false);
const doneRef = useRef(false);
doneRef.current = done;
const cursorRef = useRef<Cursor>(null);
useEffect(() => {
setItems([]);
setDone(false);
doneRef.current = false;
inFlight.current = false;
cursorRef.current = null;
}, [resetKey]);
const loadMore = useCallback(async () => {
if (inFlight.current || doneRef.current || normalizedSlugs.length === 0)
return 0;
inFlight.current = true;
setLoading(true);
try {
const data = await getTaggedArtworksPage({
take: 60,
cursor: cursorRef.current,
tagSlugs: normalizedSlugs,
onlyPublished: true,
});
setItems((prev) => {
const seen = new Set(prev.map((x) => x.id));
const next = data.items.filter((x) => !seen.has(x.id));
return prev.concat(next);
});
cursorRef.current = data.nextCursor;
if (!data.nextCursor) setDone(true);
return data.items.length;
} finally {
setLoading(false);
inFlight.current = false;
}
}, [normalizedSlugs]);
useEffect(() => {
void loadMore();
}, [loadMore]);
const galleryItems: JustifiedGalleryItem[] = items.map((it) => ({
id: it.id,
name: it.name,
altText: it.altText,
fileKey: it.fileKey,
width: it.thumbW,
height: it.thumbH,
dominantHex: it.dominantHex,
}));
if (!loading && done && galleryItems.length === 0) {
return (
<p className="text-muted-foreground text-center py-20">
No artworks to display
</p>
);
}
return (
<div className="w-full">
<JustifiedGallery
items={galleryItems}
hrefFrom="portfolio"
showCaption={false}
debug={false}
targetRowHeight={160}
targetRowHeightMobile={160}
maxRowHeight={300}
maxRowItems={5}
maxRowItemsMobile={1}
gap={12}
gapBreakpoints={[
{ maxWidth: 685, gap: 6 },
{ maxWidth: 910, gap: 8 },
{ maxWidth: 1130, gap: 10 },
]}
onLoadMore={done ? undefined : () => void loadMore()}
hasMore={!done}
isLoadingMore={loading}
/>
</div>
);
}