feat: add bookshelf home page with animated book spine components
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
102
src/app/page.tsx
102
src/app/page.tsx
@@ -1,101 +1,7 @@
|
||||
import Image from "next/image";
|
||||
import { getAllGames } from "@/lib/content";
|
||||
import BookShelf from "@/components/BookShelf";
|
||||
|
||||
export default function Home() {
|
||||
return (
|
||||
<div className="grid grid-rows-[20px_1fr_20px] items-center justify-items-center min-h-screen p-8 pb-20 gap-16 sm:p-20 font-[family-name:var(--font-geist-sans)]">
|
||||
<main className="flex flex-col gap-8 row-start-2 items-center sm:items-start">
|
||||
<Image
|
||||
className="dark:invert"
|
||||
src="https://nextjs.org/icons/next.svg"
|
||||
alt="Next.js logo"
|
||||
width={180}
|
||||
height={38}
|
||||
priority
|
||||
/>
|
||||
<ol className="list-inside list-decimal text-sm text-center sm:text-left font-[family-name:var(--font-geist-mono)]">
|
||||
<li className="mb-2">
|
||||
Get started by editing{" "}
|
||||
<code className="bg-black/[.05] dark:bg-white/[.06] px-1 py-0.5 rounded font-semibold">
|
||||
src/app/page.tsx
|
||||
</code>
|
||||
.
|
||||
</li>
|
||||
<li>Save and see your changes instantly.</li>
|
||||
</ol>
|
||||
|
||||
<div className="flex gap-4 items-center flex-col sm:flex-row">
|
||||
<a
|
||||
className="rounded-full border border-solid border-transparent transition-colors flex items-center justify-center bg-foreground text-background gap-2 hover:bg-[#383838] dark:hover:bg-[#ccc] text-sm sm:text-base h-10 sm:h-12 px-4 sm:px-5"
|
||||
href="https://vercel.com/new?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app"
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
>
|
||||
<Image
|
||||
className="dark:invert"
|
||||
src="https://nextjs.org/icons/vercel.svg"
|
||||
alt="Vercel logomark"
|
||||
width={20}
|
||||
height={20}
|
||||
/>
|
||||
Deploy now
|
||||
</a>
|
||||
<a
|
||||
className="rounded-full border border-solid border-black/[.08] dark:border-white/[.145] transition-colors flex items-center justify-center hover:bg-[#f2f2f2] dark:hover:bg-[#1a1a1a] hover:border-transparent text-sm sm:text-base h-10 sm:h-12 px-4 sm:px-5 sm:min-w-44"
|
||||
href="https://nextjs.org/docs?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app"
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
>
|
||||
Read our docs
|
||||
</a>
|
||||
</div>
|
||||
</main>
|
||||
<footer className="row-start-3 flex gap-6 flex-wrap items-center justify-center">
|
||||
<a
|
||||
className="flex items-center gap-2 hover:underline hover:underline-offset-4"
|
||||
href="https://nextjs.org/learn?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app"
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
>
|
||||
<Image
|
||||
aria-hidden
|
||||
src="https://nextjs.org/icons/file.svg"
|
||||
alt="File icon"
|
||||
width={16}
|
||||
height={16}
|
||||
/>
|
||||
Learn
|
||||
</a>
|
||||
<a
|
||||
className="flex items-center gap-2 hover:underline hover:underline-offset-4"
|
||||
href="https://vercel.com/templates?framework=next.js&utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app"
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
>
|
||||
<Image
|
||||
aria-hidden
|
||||
src="https://nextjs.org/icons/window.svg"
|
||||
alt="Window icon"
|
||||
width={16}
|
||||
height={16}
|
||||
/>
|
||||
Examples
|
||||
</a>
|
||||
<a
|
||||
className="flex items-center gap-2 hover:underline hover:underline-offset-4"
|
||||
href="https://nextjs.org?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app"
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
>
|
||||
<Image
|
||||
aria-hidden
|
||||
src="https://nextjs.org/icons/globe.svg"
|
||||
alt="Globe icon"
|
||||
width={16}
|
||||
height={16}
|
||||
/>
|
||||
Go to nextjs.org →
|
||||
</a>
|
||||
</footer>
|
||||
</div>
|
||||
);
|
||||
const games = getAllGames();
|
||||
return <BookShelf games={games} />;
|
||||
}
|
||||
|
||||
38
src/components/BookShelf.tsx
Normal file
38
src/components/BookShelf.tsx
Normal file
@@ -0,0 +1,38 @@
|
||||
import { GameMeta } from "@/types";
|
||||
import BookSpine from "./BookSpine";
|
||||
|
||||
export default function BookShelf({ games }: { games: GameMeta[] }) {
|
||||
return (
|
||||
<div className="min-h-screen flex flex-col items-center justify-center px-4 py-16">
|
||||
{/* Title */}
|
||||
<div className="text-center mb-16">
|
||||
<h1 className="font-playfair text-4xl md:text-6xl font-bold text-parchment mb-4">
|
||||
Cronicas de los Reinos
|
||||
</h1>
|
||||
<p className="font-lora text-stone-400 text-lg italic max-w-lg mx-auto">
|
||||
Las historias epicas de los mundos que nos unieron
|
||||
</p>
|
||||
<div className="w-32 h-0.5 bg-amber-700/50 mx-auto mt-6" />
|
||||
</div>
|
||||
|
||||
{/* Shelf */}
|
||||
<div className="relative">
|
||||
{/* Books row */}
|
||||
<div className="flex gap-6 md:gap-8 items-end justify-center flex-wrap px-8 pb-4">
|
||||
{games.map((game) => (
|
||||
<BookSpine key={game.slug} game={game} />
|
||||
))}
|
||||
</div>
|
||||
|
||||
{/* Shelf surface */}
|
||||
<div className="h-4 bg-gradient-to-b from-wood-medium to-wood-dark rounded-sm shadow-[0_4px_12px_rgba(0,0,0,0.5)]" />
|
||||
<div className="h-2 bg-wood-dark/50 rounded-b-sm" />
|
||||
</div>
|
||||
|
||||
{/* Footer */}
|
||||
<p className="mt-16 text-stone-600 text-sm font-lora">
|
||||
Selecciona un tomo para comenzar a leer
|
||||
</p>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
47
src/components/BookSpine.tsx
Normal file
47
src/components/BookSpine.tsx
Normal file
@@ -0,0 +1,47 @@
|
||||
import Link from "next/link";
|
||||
import { GameMeta } from "@/types";
|
||||
|
||||
export default function BookSpine({ game }: { game: GameMeta }) {
|
||||
return (
|
||||
<Link href={`/${game.slug}`} className="group block">
|
||||
<div
|
||||
className="relative w-44 h-64 md:w-48 md:h-72 rounded-r-md shadow-lg cursor-pointer transition-all duration-300 group-hover:-translate-y-3 group-hover:rotate-[-2deg] group-hover:shadow-2xl"
|
||||
style={{
|
||||
background: `linear-gradient(135deg, ${game.color} 0%, ${game.colorDark} 100%)`,
|
||||
}}
|
||||
>
|
||||
{/* Book spine edge */}
|
||||
<div
|
||||
className="absolute left-0 top-0 bottom-0 w-3 rounded-l-sm"
|
||||
style={{ backgroundColor: game.colorDark }}
|
||||
/>
|
||||
|
||||
{/* Book cover content */}
|
||||
<div className="flex flex-col items-center justify-center h-full px-4 text-center">
|
||||
<div
|
||||
className="w-12 h-0.5 mb-3"
|
||||
style={{ backgroundColor: game.accentColor }}
|
||||
/>
|
||||
<h2
|
||||
className="font-playfair text-lg md:text-xl font-bold leading-tight text-white"
|
||||
>
|
||||
{game.title}
|
||||
</h2>
|
||||
<p
|
||||
className="text-xs mt-2 opacity-75 font-lora italic"
|
||||
style={{ color: game.accentColor }}
|
||||
>
|
||||
{game.subtitle}
|
||||
</p>
|
||||
<div
|
||||
className="w-12 h-0.5 mt-3"
|
||||
style={{ backgroundColor: game.accentColor }}
|
||||
/>
|
||||
</div>
|
||||
|
||||
{/* Shine effect on hover */}
|
||||
<div className="absolute inset-0 bg-gradient-to-r from-white/0 via-white/10 to-white/0 opacity-0 group-hover:opacity-100 transition-opacity duration-300 rounded-r-md" />
|
||||
</div>
|
||||
</Link>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user