A modern web application to track your Plex watchlist and organize titles by airing status or media type.
- 🔐 Plex OAuth Authentication - Secure login with your Plex account
- 📺 Watchlist Management - View movies and shows from your Plex watchlist
- 📊 Smart Grouping - Titles organized by:
- Currently Airing
- Not Yet Aired
- Recently Ended
- Finished Airing
- 🎛️ View Controls - Floating controls for filters, sorting, and grouping:
- All, Movies, TV Shows, Anime, Anime Movies
- Airing date, title, or rating / popularity sorting
- Airing-date or media-type grouping
- 🌓 Theme Support - Light, Dark, and System themes
- 🔄 Auto-refresh - Manual and automatic watchlist updates
- 📱 Responsive Design - Beautiful UI for mobile, tablet, and desktop
- React 19
- TanStack Query (React Query) - Server state management
- Zustand - Client state management
- Tailwind CSS v4 - Styling
- Rsbuild - Build tool
- Biome - Linter and formatter
Install the dependencies:
bun installStart the dev server, and the app will be available at http://localhost:3000:
bun run devBuild the app for production:
bun run buildPreview the production build locally:
bun run previewRun the linter:
bun biome check src/Auto-fix linting issues:
bun biome check --write src/- Sign in with your Plex account using OAuth
- The app fetches your full watchlist from Plex
- Titles are automatically classified as Movies, TV Shows, Anime, or Anime Movies
- Enable auto-refresh to keep your watchlist up-to-date
- Switch between light and dark themes based on your preference
- Cached watchlist data renders immediately on app load, then currently airing shows get one background season refresh per authenticated token. This prevents the cache write from retriggering the same Plex
childrenrequests indefinitely. - Persisted watchlist cache lives in IndexedDB because the enriched watchlist is too large for Safari
localStorage; small auth/theme preferences still uselocalStorage. The watchlist hook waits for async IndexedDB hydration before fetching so a valid cache does not race with a network refresh. - The watchlist cache includes a schema version. Bump or invalidate it when cached record shape changes, especially fields used by filters such as Plex
Genremetadata. - Filter categories are exact and non-overlapping: anime uses Plex genre metadata when available, then falls back to the media type so every title has one category.
- The floating view controls use the round button as the only close affordance while open; outside clicks hit a transparent backdrop that closes the panel without clicking the watchlist underneath.
- The watchlist API is paginated; keep pagination intact so "All" really means the full watchlist.
- Use
bun run test,bun run lint, andbun run buildbefore shipping changes.