4cafa1fb0b19329f43aad8a6361a9207f849450a
Magikarp Collection
A Next.js web app to browse every English Magikarp Pokémon card with images, set symbols, search, set filter, and a personal checklist stored in your browser.
Features
- Checklist (variants-aware): Track base, holofoil, and reverse holofoil per card. Local persistence with migration from v1 to v2.
- Card images: Uses official images from
images.pokemontcg.io. - Set symbols: Displays set symbol and series/name.
- Search & filter: Search by name/number/rarity/set and filter by set.
- Totals include variants: The header and tab badges use variant totals (base + available variants) for collected/uncollected counts.
- Sort by release date: Cards are ordered by set release date (oldest first).
- Caching with TTL: Server route caches card data to
data/cards.jsonwith a 24h TTL and a manual refresh button. - Cached/Live indicator: Header shows whether data came from cache or a live fetch, with last updated timestamp.
- Non-blocking error toast: Refresh errors show a short-lived toast without disrupting the current view.
- Responsive UI: Tailwind CSS, mobile-friendly grid.
Tech Stack
- Next.js 15 (App Router)
- React 19
- TypeScript
- Tailwind CSS v4
- Axios for HTTP
Quick Start
1) Prerequisites
- Node.js 18+ and npm
2) Install dependencies
npm install
3) Environment variables
Get a free Pokémon TCG API key: https://dev.pokemontcg.io/
Create one of the following files in the project root and add your key:
.env.local(recommended for Next.js).env(also supported)
# .env.local
NEXT_PUBLIC_POKEMON_TCG_API_KEY=your_key_here
An example is provided in .env.local.example.
4) Run the dev server
npm run dev
By default: http://localhost:3000
If you see a server already running on port 3000, you can use another port:
PORT=3001 npm run dev
Scripts
npm run dev– start Next.js in developmentnpm run build– production buildnpm run start– start production servernpm run lint– lint project
Project Structure
magikarp-collection/
├─ src/
│ ├─ app/
│ │ ├─ api/
│ │ │ ├─ magikarp/route.ts # API route with 24h TTL cache (data/cards.json)
│ │ │ └─ checklist/route.ts # Server-backed checklist persistence (optional)
│ │ ├─ layout.tsx # Root layout and global wrapper
│ │ ├─ page.tsx # Main page: search, set filter, grid, checklist
│ │ └─ globals.css # Tailwind entry + global styles
│ ├─ components/
│ │ ├─ CardGrid.tsx # Grid for card list
│ │ ├─ CardItem.tsx # Card tile with image, rarity, set badge, checklist toggle
│ │ ├─ Header.tsx # Title, search, stats
│ │ ├─ SetBadge.tsx # Shows set symbol and name
│ │ └─ SetFilter.tsx # Dropdown to filter by set
│ ├─ lib/
│ │ ├─ api.ts # Axios client for Pokémon TCG API
│ │ ├─ checklist.ts # Checklist utils (local + server-backed)
│ │ └─ currency.ts # Price formatting + exchange rate hook
│ └─ types/
│ └─ pokemon.ts # TypeScript types for cards/sets
├─ data/ # Local cache for cards/checklist JSON (gitignored)
├─ next.config.ts # Remote image domains, turbopack settings
├─ tsconfig.json # TS config with path alias '@/*'
├─ postcss.config.cjs # Tailwind v4 PostCSS plugin config
├─ Dockerfile # Multi-stage Docker build (standalone output)
├─ docker-compose.yml # Compose with APP_PORT and data volume
├─ .dockerignore # Reduce build context
├─ package.json
└─ .env.local.example # Example env file
Data Source
- Pokémon TCG API v2: https://api.pokemontcg.io/
- Endpoint used:
GET /v2/cardswith queryq=name:magikarp - API key header:
X-Api-Key: <your key>(automatically added whenNEXT_PUBLIC_POKEMON_TCG_API_KEYis present)
- Endpoint used:
How It Works
- The client loads data from
GET /api/magikarp(src/app/api/magikarp/route.ts). - The route proxies to Pokémon TCG API using
fetchMagikarpCards()(src/lib/api.ts), caches the full response indata/cards.json, and serves cached data when fresh (<24h) unless?refresh=1is provided. - Cards render in
CardGrid/CardItem, ordered by set release date. - Checklist persists locally (v2 supports variant-level tracking). A server-backed route exists at
api/checklistif you enable it.
Docker
Build and run using Docker/Compose. The container listens on port 3000 internally; pick a host port via APP_PORT.
Build and run
# Default host port 3000
docker compose up --build
# Custom host port
APP_PORT=8080 docker compose up --build
Open: http://localhost:3000 (or your chosen APP_PORT).
Volumes and data
- The
./datadirectory is mounted to/app/datain the container so cached JSON persists between restarts. data/cards.jsonanddata/checklist.jsonare gitignored by default.
Deployment
- Vercel: Zero-config for Next.js. Add
NEXT_PUBLIC_POKEMON_TCG_API_KEYin project Environment Variables. - Netlify/Other: Build command
npm run build, publish.nextwith a Next adapter or use Next on Node; make sure environment vars are set.
Troubleshooting
- "Cannot find module '@tailwindcss/postcss'": install the Tailwind v4 PostCSS plugin and ensure CommonJS config.
npm i -D @tailwindcss/postcss # Ensure postcss.config.cjs exists with: # module.exports = { plugins: { '@tailwindcss/postcss': {} } } - Images not loading: confirm
next.config.tsallowsimages.pokemontcg.ioand restart the dev server. - No cards shown: ensure your API key is valid and in
.env.local(or.env). Restart the dev server after changes.
License
This project is provided as-is for personal collection tracking.
Description
Languages
TypeScript
96.9%
Dockerfile
2.5%
CSS
0.4%
JavaScript
0.2%