13 Commits

39 changed files with 1700 additions and 197 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

@ -5,8 +5,8 @@
"": { "": {
"name": "app.gaertan.art", "name": "app.gaertan.art",
"dependencies": { "dependencies": {
"@aws-sdk/client-s3": "^3.974.0", "@aws-sdk/client-s3": "^3.980.0",
"@aws-sdk/s3-request-presigner": "^3.974.0", "@aws-sdk/s3-request-presigner": "^3.980.0",
"@hookform/resolvers": "^5.2.2", "@hookform/resolvers": "^5.2.2",
"@prisma/adapter-pg": "^7.3.0", "@prisma/adapter-pg": "^7.3.0",
"@prisma/client": "^7.3.0", "@prisma/client": "^7.3.0",
@ -27,15 +27,16 @@
"lucide-react": "^0.561.0", "lucide-react": "^0.561.0",
"next": "16.1.6", "next": "16.1.6",
"next-themes": "^0.4.6", "next-themes": "^0.4.6",
"pg": "^8.17.2", "pg": "^8.18.0",
"react": "19.2.4", "react": "19.2.4",
"react-dom": "19.2.4", "react-dom": "19.2.4",
"react-hook-form": "^7.71.1", "react-hook-form": "^7.71.1",
"react-markdown": "^10.1.0", "react-markdown": "^10.1.0",
"simple-icons": "^16.6.0", "simple-icons": "^16.7.0",
"sonner": "^2.0.7", "sonner": "^2.0.7",
"tailwind-merge": "^3.4.0", "tailwind-merge": "^3.4.0",
"zod": "^4.3.6", "zod": "^4.3.6",
"zustand": "^5.0.6",
}, },
"devDependencies": { "devDependencies": {
"@biomejs/biome": "2.2.0", "@biomejs/biome": "2.2.0",
@ -944,6 +945,8 @@
"zod": ["zod@4.3.6", "", {}, "sha512-rftlrkhHZOcjDwkGlnUtZZkvaPHCsDATp4pGpuOOMDaTdDDXF91wuVDJoWoPsKX/3YPQ5fHuF3STjcYyKr+Qhg=="], "zod": ["zod@4.3.6", "", {}, "sha512-rftlrkhHZOcjDwkGlnUtZZkvaPHCsDATp4pGpuOOMDaTdDDXF91wuVDJoWoPsKX/3YPQ5fHuF3STjcYyKr+Qhg=="],
"zustand": ["zustand@5.0.11", "", { "peerDependencies": { "@types/react": ">=18.0.0", "immer": ">=9.0.6", "react": ">=18.0.0", "use-sync-external-store": ">=1.2.0" }, "optionalPeers": ["@types/react", "immer", "react", "use-sync-external-store"] }, "sha512-fdZY+dk7zn/vbWNCYmzZULHRrss0jx5pPFiOuMZ/5HJN6Yv3u+1Wswy/4MpZEkEGhtNH+pwxZB8OKgUBPzYAGg=="],
"zwitch": ["zwitch@2.0.4", "", {}, "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A=="], "zwitch": ["zwitch@2.0.4", "", {}, "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A=="],
"@aws-crypto/crc32/@aws-sdk/types": ["@aws-sdk/types@3.953.0", "", { "dependencies": { "@smithy/types": "^4.10.0", "tslib": "^2.6.2" } }, "sha512-M9Iwg9kTyqTErI0vOTVVpcnTHWzS3VplQppy8MuL02EE+mJ0BIwpWfsaAPQW+/XnVpdNpWZTsHcNE29f1+hR8g=="], "@aws-crypto/crc32/@aws-sdk/types": ["@aws-sdk/types@3.953.0", "", { "dependencies": { "@smithy/types": "^4.10.0", "tslib": "^2.6.2" } }, "sha512-M9Iwg9kTyqTErI0vOTVVpcnTHWzS3VplQppy8MuL02EE+mJ0BIwpWfsaAPQW+/XnVpdNpWZTsHcNE29f1+hR8g=="],

View File

@ -42,6 +42,7 @@
"simple-icons": "^16.7.0", "simple-icons": "^16.7.0",
"sonner": "^2.0.7", "sonner": "^2.0.7",
"tailwind-merge": "^3.4.0", "tailwind-merge": "^3.4.0",
"zustand": "^5.0.6",
"zod": "^4.3.6" "zod": "^4.3.6"
}, },
"devDependencies": { "devDependencies": {

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[]
@ -491,6 +524,15 @@ model TermsOfService {
version Int @default(autoincrement()) version Int @default(autoincrement())
} }
model About {
id String @id @default(cuid())
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
markdown String
version Int @default(autoincrement())
}
model User { model User {
id String @id id String @id
name String name String

View File

@ -51,6 +51,7 @@ export async function getAnimalStudiesPage(input: unknown): Promise<AnimalStudie
id: true, id: true,
name: true, name: true,
altText: true, altText: true,
nsfw: true,
sortKey: true, sortKey: true,
file: { select: { fileKey: true } }, file: { select: { fileKey: true } },
variants: { variants: {
@ -80,6 +81,7 @@ export async function getAnimalStudiesPage(input: unknown): Promise<AnimalStudie
id: r.id, id: r.id,
name: r.name, name: r.name,
altText: r.altText, altText: r.altText,
nsfw: r.nsfw ?? false,
fileKey: r.file.fileKey, fileKey: r.file.fileKey,
width: w, width: w,
height: h, height: h,

View File

@ -12,6 +12,7 @@ export type PortfolioArtworkItem = {
id: string; id: string;
name: string; name: string;
altText: string | null; altText: string | null;
nsfw: boolean;
sortKey: number | null; sortKey: number | null;
year: number | null; year: number | null;
@ -113,6 +114,7 @@ export async function getPortfolioArtworksPage(args: {
id: true, id: true,
name: true, name: true,
altText: true, altText: true,
nsfw: true,
year: true, year: true,
sortKey: true, sortKey: true,
file: { select: { fileKey: true } }, file: { select: { fileKey: true } },
@ -138,6 +140,7 @@ export async function getPortfolioArtworksPage(args: {
id: r.id, id: r.id,
name: r.name, name: r.name,
altText: r.altText ?? null, altText: r.altText ?? null,
nsfw: r.nsfw ?? false,
sortKey: r.sortKey ?? null, sortKey: r.sortKey ?? null,
year: r.year ?? null, year: r.year ?? null,
fileKey: r.file.fileKey, fileKey: r.file.fileKey,

View File

@ -0,0 +1,176 @@
"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;
nsfw: boolean;
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,
nsfw: 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,
nsfw: r.nsfw ?? false,
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

@ -0,0 +1,34 @@
import { prisma } from "@/lib/prisma";
import Image from "next/image";
import ReactMarkdown from "react-markdown";
export default async function AboutPage() {
const about = await prisma.about.findFirst({
orderBy: [{ version: "desc" }],
});
return (
<div className="mx-auto w-full max-w-6xl px-4 py-8">
<div className="markdown text-center">
<ReactMarkdown
components={{
img: ({ src, alt }) => {
if (!src || typeof src !== "string") return null;
return (
<Image
src={src}
alt={alt ?? ""}
width={1200}
height={800}
className="max-w-full h-auto rounded-md border border-border"
/>
);
},
}}
>
{about?.markdown}
</ReactMarkdown>
</div>
</div>
);
}

View File

@ -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);

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

@ -0,0 +1,13 @@
import UnderConstruction from "@/components/global/UnderConstruction";
export default function ArtfightPage() {
return (
<div className="mx-auto w-full max-w-6xl px-4 py-10">
<UnderConstruction
title="Artfight Gallery"
subtitle="This page is getting ready for its big debut."
note="Im curating attacks, revenges, and progress shots — check back soon."
/>
</div>
);
}

View File

@ -1,12 +1,13 @@
import ArtworkMetaCard from "@/components/artworks/ArtworkMetaCard"; import ArtworkMetaCard from "@/components/artworks/ArtworkMetaCard";
import ArtworkTimelapseViewer from "@/components/artworks/ArtworkTimelapseViewer"; import ArtworkTimelapseViewer from "@/components/artworks/ArtworkTimelapseViewer";
import { ContextBackButton } from "@/components/artworks/ContextBackButton"; import { ContextBackButton } from "@/components/artworks/ContextBackButton";
import NsfwConsentDialog from "@/components/nsfw/NsfwConsentDialog";
import NsfwImage from "@/components/nsfw/NsfwImage";
import NsfwLink from "@/components/nsfw/NsfwLink";
import { Button } from "@/components/ui/button"; import { Button } from "@/components/ui/button";
import { prisma } from "@/lib/prisma"; import { prisma } from "@/lib/prisma";
import { cn } from "@/lib/utils"; import { cn } from "@/lib/utils";
import { PlayCircle } from "lucide-react"; import { PlayCircle } from "lucide-react";
import Image from "next/image";
import Link from "next/link";
export default async function SingleArtworkPage({ export default async function SingleArtworkPage({
params, params,
@ -49,6 +50,7 @@ export default async function SingleArtworkPage({
return ( return (
<div className="px-4 sm:px-8 py-4"> <div className="px-4 sm:px-8 py-4">
<NsfwConsentDialog hasNsfw={Boolean(artwork.nsfw)} />
<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 hidden sm:block"> <div className="z-10 hidden sm:block">
<ContextBackButton /> <ContextBackButton />
@ -69,16 +71,17 @@ export default async function SingleArtworkPage({
className="relative w-full bg-muted items-center justify-center" className="relative w-full bg-muted items-center justify-center"
style={{ aspectRatio: "4 / 3" }} style={{ aspectRatio: "4 / 3" }}
> >
<Link href={`/raw/${artwork.id}`}> <NsfwLink href={`/raw/${artwork.id}`} nsfw={Boolean(artwork.nsfw)}>
<Image <NsfwImage
src={`/api/image/resized/${artwork.file.fileKey}.webp`} src={`/api/image/resized/${artwork.file.fileKey}.webp`}
alt={artwork.altText || "Artwork"} alt={artwork.altText || "Artwork"}
fill={!width || !height} fill={!width || !height}
width={width} width={width}
height={height} height={height}
nsfw={Boolean(artwork.nsfw)}
className={cn("object-cover transition duration-300")} className={cn("object-cover transition duration-300")}
/> />
</Link> </NsfwLink>
</div> </div>
</div> </div>
{artwork.timelapse?.enabled ? ( {artwork.timelapse?.enabled ? (

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

@ -0,0 +1,13 @@
import UnderConstruction from "@/components/global/UnderConstruction";
export default function MiniaturesPage() {
return (
<div className="mx-auto w-full max-w-6xl px-4 py-10">
<UnderConstruction
title="Warhammer Miniatures"
subtitle="This page is getting ready for its big debut."
note="Im curating attacks, revenges, and progress shots — check back soon."
/>
</div>
);
}

View File

@ -10,15 +10,23 @@ export default function Home() {
Welcome to my place! Welcome to my place!
</h1> </h1>
<p className="text-muted-foreground max-w-xl text-lg mb-6"> <p className="text-muted-foreground max-w-xl text-lg mb-6">
I&apos;m an illustrator, character designer, miniature painter, 3d modeller, makeup artist and much more and happy to show you things i&apos;ve created. I&apos;m an illustrator, character designer, miniature painter, 3d
modeller, makeup artist and much more and happy to show you things
i&apos;ve created.
</p> </p>
<p className="text-muted-foreground max-w-xl text-lg mb-6"> <p className="text-muted-foreground max-w-xl text-lg mb-6">
If you want to commission me<br />you can find all the information you need here:<br /> <Link href="/commissions" className="underline text-primary" >Commissions</Link> If you want to commission me
<br />
you can find all the information you need here:
<br />{" "}
<Link href="/commissions" className="underline text-primary">
Commissions
</Link>
</p> </p>
<div> <div>
<SocialLinks /> <SocialLinks />
</div> </div>
</div> </div>
</div > </div>
); );
} }

View File

@ -1,10 +1,10 @@
import { prisma } from '@/lib/prisma'; import { prisma } from "@/lib/prisma";
import ReactMarkdown from 'react-markdown'; import ReactMarkdown from "react-markdown";
export default async function TosPage() { export default async function TosPage() {
const tos = await prisma.termsOfService.findFirst({ const tos = await prisma.termsOfService.findFirst({
orderBy: [{ version: "desc" }], orderBy: [{ version: "desc" }],
}) });
return ( return (
<div className="mx-auto w-full max-w-6xl px-4 py-8"> <div className="mx-auto w-full max-w-6xl px-4 py-8">

45
src/app/error.tsx Normal file
View File

@ -0,0 +1,45 @@
"use client";
import Link from "next/link";
import { Button } from "@/components/ui/button";
export default function ErrorPage({
error,
reset,
}: {
error: Error & { digest?: string };
reset: () => void;
}) {
return (
<main className="relative min-h-screen overflow-hidden">
<div className="pointer-events-none absolute inset-0">
<div className="absolute -left-16 top-6 h-60 w-60 rounded-full bg-destructive/15 blur-3xl" />
<div className="absolute right-0 bottom-0 h-72 w-72 rounded-full bg-primary/20 blur-3xl" />
</div>
<div className="relative mx-auto flex min-h-screen w-full max-w-3xl flex-col items-center justify-center gap-6 px-6 py-16 text-center">
<div className="rounded-full border border-border/60 bg-card px-4 py-1 text-xs font-semibold uppercase tracking-[0.3em] text-muted-foreground">
Something went wrong
</div>
<h1 className="text-4xl font-semibold tracking-tight text-foreground sm:text-5xl">
We hit a snag.
</h1>
<p className="text-base text-muted-foreground">
The page failed to load. You can try again or head back home.
</p>
<div className="flex flex-wrap items-center justify-center gap-3">
<Button size="lg" onClick={reset}>
Try again
</Button>
<Button asChild size="lg" variant="outline">
<Link href="/">Go home</Link>
</Button>
</div>
<div className="rounded-2xl border border-border/60 bg-card/70 px-4 py-3 text-xs text-muted-foreground">
{error.digest ? `Error ID: ${error.digest}` : "Unexpected error"}
</div>
</div>
</main>
);
}

49
src/app/global-error.tsx Normal file
View File

@ -0,0 +1,49 @@
"use client";
import Link from "next/link";
import { Button } from "@/components/ui/button";
export default function GlobalError({
error,
reset,
}: {
error: Error & { digest?: string };
reset: () => void;
}) {
return (
<html lang="en">
<body className="bg-background text-foreground">
<main className="relative min-h-screen overflow-hidden">
<div className="pointer-events-none absolute inset-0">
<div className="absolute -left-24 top-0 h-64 w-64 rounded-full bg-destructive/15 blur-3xl" />
<div className="absolute right-0 bottom-0 h-80 w-80 rounded-full bg-primary/20 blur-3xl" />
</div>
<div className="relative mx-auto flex min-h-screen w-full max-w-3xl flex-col items-center justify-center gap-6 px-6 py-16 text-center">
<div className="rounded-full border border-border/60 bg-card px-4 py-1 text-xs font-semibold uppercase tracking-[0.3em] text-muted-foreground">
Critical error
</div>
<h1 className="text-4xl font-semibold tracking-tight text-foreground sm:text-5xl">
We could not recover.
</h1>
<p className="text-base text-muted-foreground">
A global error occurred. Try again or return home.
</p>
<div className="flex flex-wrap items-center justify-center gap-3">
<Button size="lg" onClick={reset}>
Retry
</Button>
<Button asChild size="lg" variant="outline">
<Link href="/">Go home</Link>
</Button>
</div>
<div className="rounded-2xl border border-border/60 bg-card/70 px-4 py-3 text-xs text-muted-foreground">
{error.digest ? `Error ID: ${error.digest}` : "Unexpected error"}
</div>
</div>
</main>
</body>
</html>
);
}

17
src/app/loading.tsx Normal file
View File

@ -0,0 +1,17 @@
export default function Loading() {
return (
<main className="relative flex min-h-screen items-center justify-center overflow-hidden">
<div className="pointer-events-none absolute inset-0">
<div className="absolute -left-24 -top-24 h-72 w-72 rounded-full bg-primary/20 blur-3xl" />
<div className="absolute right-0 bottom-0 h-72 w-72 rounded-full bg-accent/30 blur-3xl" />
</div>
<div className="relative flex flex-col items-center gap-4 rounded-3xl border border-border/60 bg-card/70 px-10 py-12 shadow-lg">
<div className="h-10 w-10 animate-spin rounded-full border-2 border-muted-foreground border-t-transparent" />
<div className="text-sm font-semibold uppercase tracking-[0.3em] text-muted-foreground">
Loading
</div>
<div className="text-xs text-muted-foreground">Preparing the gallery.</div>
</div>
</main>
);
}

48
src/app/not-found.tsx Normal file
View File

@ -0,0 +1,48 @@
import Link from "next/link";
import { GameOfLifeMini } from "@/components/global/GameOfLifeMini";
import { Button } from "@/components/ui/button";
export default function NotFound() {
return (
<main className="relative min-h-screen overflow-hidden">
<div className="pointer-events-none absolute inset-0">
<div className="absolute -left-32 -top-24 h-72 w-72 rounded-full bg-primary/15 blur-3xl" />
<div className="absolute right-0 top-10 h-64 w-64 rounded-full bg-accent/40 blur-3xl" />
<div className="absolute bottom-0 left-1/2 h-80 w-80 -translate-x-1/2 rounded-full bg-secondary/40 blur-3xl" />
</div>
<div className="relative mx-auto flex min-h-screen w-full max-w-6xl flex-col justify-center gap-10 px-6 py-16 lg:flex-row lg:items-center">
<section className="max-w-xl space-y-6">
<div className="inline-flex items-center gap-2 rounded-full border border-border/60 bg-card px-3 py-1 text-xs font-semibold uppercase tracking-[0.3em] text-muted-foreground">
Lost and Found
</div>
<h1 className="text-4xl font-semibold tracking-tight text-foreground sm:text-5xl">
Page not found.
</h1>
<p className="text-base text-muted-foreground">
This page wandered off. While we look for it, you can play a quick
Game of Life. Click cells to shape the pattern.
</p>
<div className="flex flex-wrap gap-3">
<Button asChild size="lg">
<Link href="/">Return home</Link>
</Button>
<Button asChild variant="outline" size="lg">
<Link href="/artworks">Browse artworks</Link>
</Button>
</div>
<div className="rounded-2xl border border-border/60 bg-card/70 p-4 text-sm text-muted-foreground shadow-sm">
Tip: The gliders are shy. Try clicking five cells in an L shape.
</div>
</section>
<section className="w-full max-w-xl">
<div className="rounded-3xl border border-border/60 bg-card/70 p-5 shadow-lg">
<GameOfLifeMini />
</div>
</section>
</div>
</main>
);
}

View File

@ -68,6 +68,11 @@ export default function AnimalStudiesGallery({
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

@ -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

@ -1,6 +1,9 @@
"use client"; "use client";
import NsfwBadge from "@/components/nsfw/NsfwBadge";
import NsfwConsentDialog from "@/components/nsfw/NsfwConsentDialog";
import { cn } from "@/lib/utils"; import { cn } from "@/lib/utils";
import { useNsfwStore } from "@/stores/nsfw-store";
import Image from "next/image"; import Image from "next/image";
import Link from "next/link"; import Link from "next/link";
import { import {
@ -24,6 +27,9 @@ export type JustifiedGalleryItem = {
/** Optional: dominant color for hover ring. */ /** Optional: dominant color for hover ring. */
dominantHex?: string | null; dominantHex?: string | null;
/** Optional: NSFW flag */
nsfw?: boolean | null;
}; };
type Props = { type Props = {
@ -44,6 +50,9 @@ 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; debug?: boolean;
className?: string; className?: string;
}; };
@ -85,12 +94,39 @@ export default function JustifiedGallery({
maxRowItems = 5, maxRowItems = 5,
maxRowItemsMobile = 3, maxRowItemsMobile = 3,
gap = 12, gap = 12,
gapNarrow,
gapNarrowMaxWidth = 720,
gapBreakpoints,
debug = false, debug = false,
className, className,
}: Props) { }: Props) {
const consent = useNsfwStore((s) => s.consent);
const allowNsfw = consent === "allow";
const hasNsfw = useMemo(() => items.some((i) => i.nsfw), [items]);
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(() => {
@ -143,7 +179,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.
@ -172,14 +208,15 @@ export default function JustifiedGallery({
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 = gap * (current.length - 1); const gaps = effectiveGap * (current.length - 1);
const widthWithoutGaps = Math.max(0, available - gaps); const widthWithoutGaps = Math.max(0, available - gaps);
const computedH = widthWithoutGaps / aspectSum; const computedH = widthWithoutGaps / aspectSum;
@ -227,7 +264,7 @@ export default function JustifiedGallery({
}, [ }, [
items, items,
containerWidth, containerWidth,
gap, effectiveGap,
targetRowHeight, targetRowHeight,
targetRowHeightMobile, targetRowHeightMobile,
maxRowHeight, maxRowHeight,
@ -248,6 +285,7 @@ export default function JustifiedGallery({
ref={containerRef} ref={containerRef}
className={cn("mx-auto w-full max-w-6xl", className)} className={cn("mx-auto w-full max-w-6xl", className)}
> >
<NsfwConsentDialog hasNsfw={hasNsfw} />
<div className="space-y-3"> <div className="space-y-3">
{rows.map((row, idx) => ( {rows.map((row, idx) => (
<div key={getRowKey(row)}> <div key={getRowKey(row)}>
@ -260,7 +298,7 @@ export default function JustifiedGallery({
? "justify-start" ? "justify-start"
: "justify-between", : "justify-between",
)} )}
style={{ columnGap: gap }} style={{ columnGap: effectiveGap }}
> >
{row.map((t) => ( {row.map((t) => (
<GalleryTile <GalleryTile
@ -269,13 +307,15 @@ export default function JustifiedGallery({
hrefBase={hrefBase} hrefBase={hrefBase}
hrefFrom={hrefFrom} hrefFrom={hrefFrom}
showCaption={showCaption} showCaption={showCaption}
allowNsfw={allowNsfw}
/> />
))} ))}
</div> </div>
{debug ? ( {debug ? (
<div className="text-xs text-muted-foreground font-mono"> <div className="text-xs text-muted-foreground font-mono">
{`row ${idx + 1} | h=${Math.round(row[0]?.h ?? 0)} | w=${Math.round( {`row ${idx + 1} | h=${Math.round(row[0]?.h ?? 0)} | w=${Math.round(
row.reduce((sum, t) => sum + t.w, 0) + gap * (row.length - 1), row.reduce((sum, t) => sum + t.w, 0) +
effectiveGap * (row.length - 1),
)} | items=${row.length} | `} )} | items=${row.length} | `}
{row {row
.map( .map(
@ -302,16 +342,20 @@ function GalleryTile({
hrefBase, hrefBase,
hrefFrom, hrefFrom,
showCaption, showCaption,
allowNsfw,
}: { }: {
tile: RowTile; tile: RowTile;
hrefBase: string; hrefBase: string;
hrefFrom: string; hrefFrom: string;
showCaption: boolean; showCaption: boolean;
allowNsfw: boolean;
}) { }) {
const { item, w, h } = tile; const { item, w, h } = tile;
const href = `${hrefBase}/${item.id}?from=${encodeURIComponent(hrefFrom)}`; const href = `${hrefBase}/${item.id}?from=${encodeURIComponent(hrefFrom)}`;
const src = `/api/image/gallery/${item.fileKey}.webp`; const src = `/api/image/gallery/${item.fileKey}.webp`;
const isNsfw = Boolean(item.nsfw);
const shouldBlur = isNsfw && !allowNsfw;
const style: CSSProperties & { "--dom"?: string } = {}; const style: CSSProperties & { "--dom"?: string } = {};
const dom = normalizeColor(item.dominantHex); const dom = normalizeColor(item.dominantHex);
@ -346,11 +390,26 @@ function GalleryTile({
alt={item.altText ?? item.name ?? "Artwork"} alt={item.altText ?? item.name ?? "Artwork"}
width={w} width={w}
height={h} height={h}
className="h-full w-full object-cover" className={cn(
"h-full w-full object-cover transition-[filter,transform] duration-200",
shouldBlur ? "blur-2xl scale-[1.02]" : "blur-0",
)}
// Tiles are thumbnail-ish; bias Next toward small resources. // Tiles are thumbnail-ish; bias Next toward small resources.
sizes="(max-width: 640px) 90vw, (max-width: 1024px) 50vw, 320px" sizes="(max-width: 640px) 90vw, (max-width: 1024px) 50vw, 320px"
/> />
{isNsfw ? (
shouldBlur ? null : (
<NsfwBadge className="absolute left-2 bottom-2 z-10" />
)
) : null}
{shouldBlur ? (
<div className="absolute inset-0 z-10 flex items-center justify-center bg-black/45">
<NsfwBadge className="scale-110 shadow-md" />
</div>
) : null}
{showCaption ? ( {showCaption ? (
<div className="pointer-events-none absolute inset-x-0 top-0 bg-black/60 p-3"> <div className="pointer-events-none absolute inset-x-0 top-0 bg-black/60 p-3">
<div className="text-sm font-medium text-white line-clamp-1"> <div className="text-sm font-medium text-white line-clamp-1">

View File

@ -0,0 +1,200 @@
"use client";
import type { MouseEvent } from "react";
import { useEffect, useMemo, useRef, useState } from "react";
import { Button } from "@/components/ui/button";
function createGrid(cols: number, rows: number) {
return Array.from({ length: rows }, () => Array.from({ length: cols }, () => 0));
}
function seedRandom(grid: number[][], density: number) {
for (let y = 0; y < grid.length; y += 1) {
for (let x = 0; x < grid[0].length; x += 1) {
grid[y][x] = Math.random() < density ? 1 : 0;
}
}
}
function stepGrid(grid: number[][]) {
const rows = grid.length;
const cols = grid[0]?.length ?? 0;
const next = createGrid(cols, rows);
for (let y = 0; y < rows; y += 1) {
const yPrev = (y - 1 + rows) % rows;
const yNext = (y + 1) % rows;
for (let x = 0; x < cols; x += 1) {
const xPrev = (x - 1 + cols) % cols;
const xNext = (x + 1) % cols;
const neighbors =
grid[yPrev][xPrev] +
grid[yPrev][x] +
grid[yPrev][xNext] +
grid[y][xPrev] +
grid[y][xNext] +
grid[yNext][xPrev] +
grid[yNext][x] +
grid[yNext][xNext];
if (grid[y][x]) {
next[y][x] = neighbors === 2 || neighbors === 3 ? 1 : 0;
} else {
next[y][x] = neighbors === 3 ? 1 : 0;
}
}
}
return next;
}
export function GameOfLifeMini() {
const canvasRef = useRef<HTMLCanvasElement | null>(null);
const [running, setRunning] = useState(true);
const [speed, setSpeed] = useState(6);
const cellSize = 10;
const width = 420;
const height = 260;
const { rows, cols } = useMemo(
() => ({
cols: Math.floor(width / cellSize),
rows: Math.floor(height / cellSize),
}),
[],
);
const gridRef = useRef<number[][]>(createGrid(cols, rows));
useEffect(() => {
gridRef.current = createGrid(cols, rows);
seedRandom(gridRef.current, 0.22);
}, [cols, rows]);
useEffect(() => {
const canvas = canvasRef.current;
if (!canvas) return;
const ctx = canvas.getContext("2d");
if (!ctx) return;
let raf = 0;
let last = performance.now();
const tickMs = 1000 / speed;
const draw = () => {
ctx.clearRect(0, 0, width, height);
ctx.fillStyle = "#141414";
ctx.fillRect(0, 0, width, height);
const grid = gridRef.current;
ctx.fillStyle = "#f2f2f2";
for (let y = 0; y < rows; y += 1) {
for (let x = 0; x < cols; x += 1) {
if (grid[y][x]) {
ctx.fillRect(
x * cellSize,
y * cellSize,
cellSize - 1,
cellSize - 1,
);
}
}
}
};
const loop = (now: number) => {
if (running && now - last >= tickMs) {
gridRef.current = stepGrid(gridRef.current);
last = now;
}
draw();
raf = requestAnimationFrame(loop);
};
raf = requestAnimationFrame(loop);
return () => cancelAnimationFrame(raf);
}, [running, speed, cols, rows]);
const handleToggleCell = (event: MouseEvent<HTMLCanvasElement>) => {
const rect = event.currentTarget.getBoundingClientRect();
const x = Math.floor((event.clientX - rect.left) / cellSize);
const y = Math.floor((event.clientY - rect.top) / cellSize);
if (x < 0 || y < 0 || y >= rows || x >= cols) return;
gridRef.current[y][x] = gridRef.current[y][x] ? 0 : 1;
};
return (
<div className="space-y-3">
<div className="flex items-center justify-between">
<div>
<div className="text-sm font-semibold uppercase tracking-wide text-muted-foreground">
Game of Life
</div>
<div className="text-xs text-muted-foreground">
Click to toggle cells. Space-time improvisation.
</div>
</div>
<div className="flex items-center gap-2">
<Button
size="sm"
variant={running ? "secondary" : "default"}
onClick={() => setRunning((prev) => !prev)}
>
{running ? "Pause" : "Play"}
</Button>
<Button
size="sm"
variant="outline"
onClick={() => {
gridRef.current = createGrid(cols, rows);
}}
>
Clear
</Button>
<Button
size="sm"
variant="outline"
onClick={() => {
gridRef.current = createGrid(cols, rows);
seedRandom(gridRef.current, 0.22);
}}
>
Randomize
</Button>
</div>
</div>
<div className="rounded-2xl border border-border/60 bg-background/60 p-3 shadow-sm">
<canvas
ref={canvasRef}
width={width}
height={height}
onClick={handleToggleCell}
className="h-65 w-full rounded-xl border border-border/40 bg-black"
/>
</div>
<div className="flex items-center gap-2 text-xs text-muted-foreground">
<span>Speed</span>
<div className="flex items-center gap-2">
<Button
size="sm"
variant="ghost"
onClick={() => setSpeed((prev) => Math.max(2, prev - 1))}
>
-
</Button>
<span className="min-w-6 text-center tabular-nums">{speed}</span>
<Button
size="sm"
variant="ghost"
onClick={() => setSpeed((prev) => Math.min(12, prev + 1))}
>
+
</Button>
</div>
</div>
</div>
);
}

View File

@ -1,14 +1,20 @@
import ModeToggle from "./ModeToggle"; import ModeToggle from "./ModeToggle";
import NsfwModeToggle from "./NsfwModeToggle";
import TopNav from "./TopNav"; import TopNav from "./TopNav";
export default function Header() { export default function Header() {
return ( return (
<div className="w-full"> <div className="w-full">
<div className="flex items-center justify-between px-4 md:px-8 py-2"> <div className="flex w-full items-center gap-4 px-4 md:px-8 py-2">
<div className="flex-1 min-w-0">
<TopNav /> <TopNav />
</div>
<div className="flex items-center gap-2 shrink-0">
<NsfwModeToggle />
<ModeToggle /> <ModeToggle />
</div> </div>
</div> </div>
</div>
); );
} }

View File

@ -0,0 +1,42 @@
"use client";
import { Eye, EyeOff } from "lucide-react";
import { Button } from "@/components/ui/button";
import {
DropdownMenu,
DropdownMenuContent,
DropdownMenuItem,
DropdownMenuTrigger,
} from "@/components/ui/dropdown-menu";
import { useNsfwStore } from "@/stores/nsfw-store";
export default function NsfwModeToggle() {
const consent = useNsfwStore((s) => s.consent);
const setConsent = useNsfwStore((s) => s.setConsent);
const enabled = consent === "allow";
return (
<DropdownMenu>
<DropdownMenuTrigger asChild>
<Button variant="outline" size="icon">
{enabled ? (
<Eye className="h-[1.2rem] w-[1.2rem]" />
) : (
<EyeOff className="h-[1.2rem] w-[1.2rem]" />
)}
<span className="sr-only">Toggle NSFW mode</span>
</Button>
</DropdownMenuTrigger>
<DropdownMenuContent align="end">
<DropdownMenuItem onClick={() => setConsent("allow")}>
Allow NSFW Content
</DropdownMenuItem>
<DropdownMenuItem onClick={() => setConsent("deny")}>
Hide NSFW Content
</DropdownMenuItem>
</DropdownMenuContent>
</DropdownMenu>
);
}

View File

@ -1,45 +1,222 @@
"use client" "use client"
import { NavigationMenu, NavigationMenuItem, NavigationMenuLink, NavigationMenuList, navigationMenuTriggerStyle } from "@/components/ui/navigation-menu"; import { NavigationMenu, NavigationMenuContent, NavigationMenuItem, NavigationMenuLink, NavigationMenuList, NavigationMenuTrigger, navigationMenuTriggerStyle } from "@/components/ui/navigation-menu";
import { Menu } from "lucide-react"; import { ChevronDown, Ellipsis, Menu } from "lucide-react";
import Link from "next/link"; import Link from "next/link";
import { useState } from "react"; import { useEffect, useLayoutEffect, useRef, useState } from "react";
import { Button } from "../ui/button"; import { Button } from "../ui/button";
import { Sheet, SheetContent, SheetHeader, SheetTitle, SheetTrigger } from "../ui/sheet"; import { Sheet, SheetContent, SheetHeader, SheetTitle, SheetTrigger } from "../ui/sheet";
const links = [ const links = [
{ href: "/", label: "Home" }, { type: "link" as const, href: "/", label: "Home" },
{ href: "/artworks", label: "Portfolio" }, { type: "link" as const, href: "/artworks", label: "Portfolio" },
{
type: "dropdown" as const,
label: "Categories",
items: [
{ href: "/artworks/animalstudies", label: "Animal Studies" }, { href: "/artworks/animalstudies", label: "Animal Studies" },
{ href: "/commissions", label: "Commissions" }, // { href: "/artworks/artfight", label: "Artfight" }
{ href: "/commissions/status", label: "Commission Status" }, ],
{ href: "/tos", label: "Terms of Service" }, },
// { href: "/portfolio/artfight", label: "Artfight" }, // { type: "link" as const, href: "/miniatures", label: "Miniatures" },
// { href: "/portfolio/minis", label: "Miniatures" }, { type: "link" as const, href: "/commissions", label: "Commissions" },
// { href: "/commissions", label: "Commissions" }, { type: "link" as const, href: "/commissions/status", label: "Commission Status" },
// { href: "/ych", label: "YCH / Custom offers" }, { type: "link" as const, href: "/tos", label: "Terms of Service" },
// { href: "/todo", label: "todo (temp)" }, { type: "link" as const, href: "/about", label: "About Me" },
] ]
export default function TopNav() { export default function TopNav() {
const [open, setOpen] = useState(false) const [open, setOpen] = useState(false)
const requiredLinks = [
links[0], // Home
links[1], // Portfolio
links[2], // Commissions
];
const flexibleLinks = links.filter((link) => !requiredLinks.includes(link));
const [visibleCount, setVisibleCount] = useState(flexibleLinks.length);
const listRef = useRef<HTMLUListElement | null>(null);
const containerRef = useRef<HTMLDivElement | null>(null);
const measureListRef = useRef<HTMLUListElement | null>(null);
const [containerWidth, setContainerWidth] = useState(0);
const visibleLinks = [...requiredLinks, ...flexibleLinks.slice(0, visibleCount)];
const overflowLinks = flexibleLinks.slice(visibleCount);
const showMore = overflowLinks.length > 0;
useEffect(() => {
const containerEl = containerRef.current;
if (!containerEl) return;
const update = () => {
setContainerWidth(containerEl.getBoundingClientRect().width);
};
const ro = new ResizeObserver(update);
ro.observe(containerEl);
window.addEventListener("resize", update);
update();
return () => {
ro.disconnect();
window.removeEventListener("resize", update);
};
}, []);
useLayoutEffect(() => {
const measureEl = measureListRef.current;
if (!measureEl || !containerWidth) return;
const items = Array.from(measureEl.children) as HTMLElement[];
if (!items.length) return;
const moreItem = items[items.length - 1];
const moreWidth = moreItem.getBoundingClientRect().width;
const itemWidths = items.slice(0, -1).map((el) => el.getBoundingClientRect().width);
const requiredIndexes = new Set([0, 1, 4]);
let requiredWidth = 0;
const flexibleWidths: number[] = [];
itemWidths.forEach((w, idx) => {
if (requiredIndexes.has(idx)) {
requiredWidth += w;
} else {
flexibleWidths.push(w);
}
});
const totalFlexibleWidth = flexibleWidths.reduce((a, b) => a + b, 0);
let nextVisibleCount = flexibleLinks.length;
const safetyPadding = 24;
if (requiredWidth + totalFlexibleWidth + safetyPadding > containerWidth) {
let available = containerWidth - requiredWidth - moreWidth - safetyPadding;
if (available < 0) available = 0;
let fit = 0;
let used = 0;
for (const w of flexibleWidths) {
if (used + w > available) break;
used += w;
fit += 1;
}
nextVisibleCount = fit;
}
if (nextVisibleCount !== visibleCount) {
setVisibleCount(nextVisibleCount);
}
}, [containerWidth, flexibleLinks.length, visibleCount]);
return ( return (
<div className="w-full flex items-center justify-between"> <div className="w-full flex items-center justify-between">
{/* Desktop Nav */} {/* Desktop Nav */}
<div className="hidden md:flex"> <div className="hidden md:flex flex-1 min-w-0">
<NavigationMenu> <NavigationMenu
<NavigationMenuList> viewport={false}
{links.map(({ href, label }) => ( delayDuration={0}
<NavigationMenuItem key={href}> skipDelayDuration={0}
className="w-full min-w-0 max-w-none"
>
<div ref={containerRef} className="w-full max-w-full min-w-0">
<NavigationMenuList
ref={listRef}
className="w-full flex-nowrap justify-start min-w-0"
>
{visibleLinks.map((item) => {
if (item.type === "dropdown") {
return (
<NavigationMenuItem key={item.label}>
<NavigationMenuTrigger className="hover:bg-hover data-[state=open]:bg-hover">
{item.label}
</NavigationMenuTrigger>
<NavigationMenuContent className="z-50">
<ul className="min-w-48">
{item.items.map(({ href, label }) => (
<li key={href}>
<NavigationMenuLink asChild className="w-full hover:bg-hover">
<Link href={href}>{label}</Link>
</NavigationMenuLink>
</li>
))}
</ul>
</NavigationMenuContent>
</NavigationMenuItem>
);
}
return (
<NavigationMenuItem key={item.href}>
<NavigationMenuLink <NavigationMenuLink
asChild asChild
className={`${navigationMenuTriggerStyle()} hover:bg-hover data-active:bg-hover focus:bg-hover active:bg-hover`} className={`${navigationMenuTriggerStyle()} hover:bg-hover data-active:bg-hover focus:bg-hover active:bg-hover`}
> >
<Link href={href}>{label}</Link> <Link href={item.href}>{item.label}</Link>
</NavigationMenuLink> </NavigationMenuLink>
</NavigationMenuItem> </NavigationMenuItem>
);
})}
{showMore ? (
<NavigationMenuItem>
<NavigationMenuTrigger className="hover:bg-hover data-[state=open]:bg-hover">
<Ellipsis className="h-4 w-4" />
</NavigationMenuTrigger>
<NavigationMenuContent className="z-50">
<ul className="min-w-48">
{overflowLinks.map((item) => {
if (item.type === "dropdown") {
return (
<li key={item.label}>
<div className="px-2 py-1 text-xs uppercase tracking-wide text-muted-foreground">
{item.label}
</div>
<ul className="pl-2">
{item.items.map(({ href, label }) => (
<li key={href}>
<NavigationMenuLink asChild className="w-full hover:bg-hover">
<Link href={href}>{label}</Link>
</NavigationMenuLink>
</li>
))} ))}
</ul>
</li>
);
}
return (
<li key={item.href}>
<NavigationMenuLink asChild className="w-full hover:bg-hover">
<Link href={item.href}>{item.label}</Link>
</NavigationMenuLink>
</li>
);
})}
</ul>
</NavigationMenuContent>
</NavigationMenuItem>
) : null}
</NavigationMenuList>
</div>
</NavigationMenu>
</div>
<div className="absolute left-0 top-0 -z-10 opacity-0 pointer-events-none">
<NavigationMenu viewport={false} delayDuration={0} skipDelayDuration={0}>
<NavigationMenuList ref={measureListRef} className="flex-nowrap">
{links.map((item) => (
<NavigationMenuItem key={`measure-${item.type === "dropdown" ? item.label : item.href}`}>
<div className={navigationMenuTriggerStyle()}>
{item.type === "dropdown" ? (
<span className="inline-flex items-center gap-1">
{item.label}
<ChevronDown className="h-3 w-3" />
</span>
) : (
item.label
)}
</div>
</NavigationMenuItem>
))}
<NavigationMenuItem>
<div className={navigationMenuTriggerStyle()}>
<Ellipsis className="h-4 w-4" />
</div>
</NavigationMenuItem>
</NavigationMenuList> </NavigationMenuList>
</NavigationMenu> </NavigationMenu>
</div> </div>
@ -56,7 +233,15 @@ export default function TopNav() {
<SheetTitle className="text-lg">Navigation</SheetTitle> <SheetTitle className="text-lg">Navigation</SheetTitle>
</SheetHeader> </SheetHeader>
<nav className="mt-4 flex flex-col gap-2"> <nav className="mt-4 flex flex-col gap-2">
{links.map(({ href, label }) => ( {links.map((item) => {
if (item.type === "dropdown") {
return (
<div key={item.label} className="px-2">
<div className="px-2 py-1 text-xs uppercase tracking-wide text-muted-foreground">
{item.label}
</div>
<div className="flex flex-col">
{item.items.map(({ href, label }) => (
<Link <Link
key={href} key={href}
href={href} href={href}
@ -67,6 +252,23 @@ export default function TopNav() {
{label} {label}
</Link> </Link>
))} ))}
</div>
</div>
);
}
return (
<Link
key={item.href}
href={item.href}
onClick={() => setOpen(false)}
className="block px-4 py-2 rounded-md text-sm font-medium transition-colors
focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring"
>
{item.label}
</Link>
);
})}
</nav> </nav>
</SheetContent> </SheetContent>
</Sheet> </Sheet>

View File

@ -0,0 +1,55 @@
import { cn } from "@/lib/utils";
import { Construction } from "lucide-react";
import Link from "next/link";
export default function UnderConstruction({
title = "Under Construction",
subtitle = "Artfight is getting its finishing touches.",
note = "Check back soon for the full gallery.",
actionHref = "/artworks",
actionLabel = "Back to Portfolio",
}: {
title?: string;
subtitle?: string;
note?: string;
actionHref?: string;
actionLabel?: string;
}) {
return (
<section className="relative overflow-hidden rounded-3xl border border-border bg-background/80 px-6 py-10 shadow-sm sm:px-10 sm:py-14">
<div className="pointer-events-none absolute inset-0 bg-[radial-gradient(circle_at_top,var(--color-muted)_0%,transparent_55%)] opacity-70" />
<div className="pointer-events-none absolute left-0 top-0 h-2 w-full bg-[repeating-linear-gradient(135deg,#7f1d1d_0px,#7f1d1d_8px,#ffffff_8px,#ffffff_16px)]" />
<div className="relative flex flex-col gap-8 lg:flex-row lg:items-center lg:justify-between">
<div className="max-w-xl space-y-4">
<div className="inline-flex items-center gap-2 rounded-full border border-border/70 bg-background/70 px-3 py-1 text-xs font-medium uppercase tracking-widest text-muted-foreground">
<Construction className="h-3.5 w-3.5 text-foreground/70" />
Under construction
</div>
<h1 className="text-3xl font-semibold tracking-tight sm:text-4xl">
{title}
</h1>
<p className="text-base text-muted-foreground sm:text-lg">
{subtitle}
</p>
<p className="text-sm text-muted-foreground/80">{note}</p>
<Link
href={actionHref}
className={cn(
"inline-flex items-center justify-center rounded-full border border-border bg-foreground px-4 py-2 text-sm font-medium text-background",
"transition hover:-translate-y-0.5 hover:bg-foreground/90"
)}
>
{actionLabel}
</Link>
</div>
<div className="relative mx-auto w-full max-w-md">
<div className="relative flex flex-col items-center justify-center gap-4 p-4">
<Construction className="h-20 w-20 text-red-800" />
</div>
</div>
</div>
</section>
);
}

View File

@ -0,0 +1,19 @@
"use client";
import { EyeOff } from "lucide-react";
import { cn } from "@/lib/utils";
export default function NsfwBadge({ className }: { className?: string }) {
return (
<div
className={cn(
"inline-flex items-center gap-1 rounded-full bg-black/70 px-2 py-1 text-xs font-semibold text-white shadow-sm",
className,
)}
>
<EyeOff className="h-3 w-3" />
NSFW
</div>
);
}

View File

@ -0,0 +1,75 @@
"use client";
import { Button } from "@/components/ui/button";
import {
Dialog,
DialogContent,
DialogDescription,
DialogFooter,
DialogHeader,
DialogTitle,
} from "@/components/ui/dialog";
import { EyeOff } from "lucide-react";
import { useEffect, useState } from "react";
import { useNsfwStore } from "@/stores/nsfw-store";
export default function NsfwConsentDialog({ hasNsfw }: { hasNsfw: boolean }) {
const consent = useNsfwStore((s) => s.consent);
const setConsent = useNsfwStore((s) => s.setConsent);
const [open, setOpen] = useState(false);
useEffect(() => {
if (!hasNsfw) {
setOpen(false);
return;
}
if (consent === "unset") setOpen(true);
else setOpen(false);
}, [hasNsfw, consent]);
if (!hasNsfw) return null;
return (
<Dialog
open={open}
onOpenChange={(next) => {
if (!next && consent === "unset") return;
setOpen(next);
}}
>
<DialogContent showCloseButton={false}>
<DialogHeader>
<DialogTitle className="flex items-center gap-2">
<EyeOff className="h-5 w-5 text-destructive" />
Sensitive Content
</DialogTitle>
<DialogDescription>
Sensitive artworks ahead. Confirm youre 18+ to view it. You can
change this anytime with the NSFW toggle in the header.
</DialogDescription>
</DialogHeader>
<DialogFooter>
<Button
variant="outline"
onClick={() => {
setConsent("deny");
setOpen(false);
}}
>
Keep blurred
</Button>
<Button
onClick={() => {
setConsent("allow");
setOpen(false);
}}
>
Im 18+ · Show
</Button>
</DialogFooter>
</DialogContent>
</Dialog>
);
}

View File

@ -0,0 +1,47 @@
"use client";
import Image, { type ImageProps } from "next/image";
import { cn } from "@/lib/utils";
import { useNsfwStore } from "@/stores/nsfw-store";
import NsfwBadge from "./NsfwBadge";
type NsfwImageProps = ImageProps & {
nsfw?: boolean;
wrapperClassName?: string;
};
export default function NsfwImage({
nsfw = false,
wrapperClassName,
className,
...props
}: NsfwImageProps) {
const consent = useNsfwStore((s) => s.consent);
const allowNsfw = consent === "allow";
const shouldBlur = nsfw && !allowNsfw;
return (
<div>
<Image
{...props}
className={cn(
"transition-[filter,transform] duration-200",
shouldBlur ? "blur-2xl scale-[1.02]" : "blur-0",
className,
)}
/>
{nsfw ? shouldBlur ? null : (
<NsfwBadge className="absolute left-2 bottom-2 z-10" />
) : null}
{shouldBlur ? (
<div className="absolute inset-0 z-10 flex items-center justify-center bg-black/45">
<NsfwBadge className="scale-110 shadow-md" />
</div>
) : null}
</div>
);
}

View File

@ -0,0 +1,38 @@
"use client";
import Link from "next/link";
import { cn } from "@/lib/utils";
import { useNsfwStore } from "@/stores/nsfw-store";
export default function NsfwLink({
href,
nsfw = false,
className,
children,
}: {
href: string;
nsfw?: boolean;
className?: string;
children: React.ReactNode;
}) {
const consent = useNsfwStore((s) => s.consent);
const allowNsfw = consent === "allow";
const blocked = nsfw && !allowNsfw;
if (blocked) {
return (
<div
className={cn("block cursor-not-allowed", className)}
aria-disabled="true"
>
{children}
</div>
);
}
return (
<Link href={href} className={cn("block", className)}>
{children}
</Link>
);
}

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,10 +86,31 @@ 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,
altText: it.altText, altText: it.altText,
nsfw: it.nsfw,
fileKey: it.fileKey, fileKey: it.fileKey,
width: it.thumbW, width: it.thumbW,
height: it.thumbH, height: it.thumbH,
@ -111,7 +138,12 @@ 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"
@ -123,6 +155,11 @@ export default function PortfolioGallery({
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,118 @@
"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(() => {
void resetKey;
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,
nsfw: it.nsfw,
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>
);
}

View File

@ -1,6 +1,6 @@
import * as React from "react"
import { Slot } from "@radix-ui/react-slot" import { Slot } from "@radix-ui/react-slot"
import { cva, type VariantProps } from "class-variance-authority" import { cva, type VariantProps } from "class-variance-authority"
import type * as React from "react"
import { cn } from "@/lib/utils" import { cn } from "@/lib/utils"

30
src/stores/nsfw-store.ts Normal file
View File

@ -0,0 +1,30 @@
"use client";
import { create } from "zustand";
import { createJSONStorage, persist } from "zustand/middleware";
export type NsfwConsent = "unset" | "allow" | "deny";
type NsfwStore = {
consent: NsfwConsent;
setConsent: (consent: NsfwConsent) => void;
allow: () => void;
deny: () => void;
reset: () => void;
};
export const useNsfwStore = create<NsfwStore>()(
persist(
(set) => ({
consent: "unset",
setConsent: (consent) => set({ consent }),
allow: () => set({ consent: "allow" }),
deny: () => set({ consent: "deny" }),
reset: () => set({ consent: "unset" }),
}),
{
name: "gaertan-nsfw-consent",
storage: createJSONStorage(() => localStorage),
},
),
);