feat(ui): Components refactoring with Tailwind - UI kit, icons, lazy loading images

- Created reusable UI components (Card, Button, Badge, Form, Icon)
- Added PageIcon and NavIcon components with consistent styling
- Refactored all pages to use new UI components
- Added non-blocking image loading with skeleton for book covers
- Created LibraryActions dropdown for library settings
- Added emojis to buttons for better UX
- Fixed Client Component issues with getBookCoverUrl
This commit is contained in:
2026-03-06 14:11:23 +01:00
parent 05a18c3c77
commit d001e29bbc
24 changed files with 1235 additions and 459 deletions

View File

@@ -1,42 +1,92 @@
"use client";
import { useState } from "react";
import Image from "next/image";
import Link from "next/link";
import { BookDto } from "../../lib/api";
interface BookCardProps {
book: BookDto;
getBookCoverUrl: (bookId: string) => string;
book: BookDto & { coverUrl?: string };
}
export function BookCard({ book, getBookCoverUrl }: BookCardProps) {
function BookImage({ src, alt }: { src: string; alt: string }) {
const [isLoaded, setIsLoaded] = useState(false);
return (
<Link href={`/books/${book.id}`} className="book-card">
<div className="book-cover">
<Image
src={getBookCoverUrl(book.id)}
alt={`Cover of ${book.title}`}
width={150}
height={220}
className="cover-image"
unoptimized
loading="lazy"
/>
<div className="relative aspect-[2/3] overflow-hidden bg-gradient-to-br from-line/50 to-line">
{/* Skeleton */}
<div
className={`absolute inset-0 bg-muted/10 animate-pulse transition-opacity duration-300 ${
isLoaded ? 'opacity-0 pointer-events-none' : 'opacity-100'
}`}
>
<div className="absolute inset-0 bg-gradient-to-r from-transparent via-muted/20 to-transparent shimmer" />
</div>
<div className="book-info">
<h3 className="book-title" title={book.title}>
{/* Image */}
<Image
src={src}
alt={alt}
fill
className={`object-cover group-hover:scale-105 transition-all duration-300 ${
isLoaded ? 'opacity-100' : 'opacity-0'
}`}
sizes="(max-width: 640px) 50vw, (max-width: 768px) 33vw, (max-width: 1024px) 25vw, 16vw"
onLoad={() => setIsLoaded(true)}
unoptimized
/>
</div>
);
}
export function BookCard({ book }: BookCardProps) {
const coverUrl = book.coverUrl || `/api/books/${book.id}/pages/1?format=webp&width=200`;
return (
<Link
href={`/books/${book.id}`}
className="group block bg-card rounded-xl border border-line shadow-soft hover:shadow-card hover:-translate-y-1 transition-all duration-200 overflow-hidden"
>
<BookImage
src={coverUrl}
alt={`Cover of ${book.title}`}
/>
{/* Book Info */}
<div className="p-4">
<h3
className="font-semibold text-foreground mb-1 line-clamp-2 min-h-[2.5rem]"
title={book.title}
>
{book.title}
</h3>
{book.author && (
<p className="book-author">{book.author}</p>
<p className="text-sm text-muted mb-1 truncate">{book.author}</p>
)}
{book.series && (
<p className="book-series">
<p className="text-xs text-muted/80 truncate mb-2">
{book.series}
{book.volume && ` #${book.volume}`}
{book.volume && <span className="text-primary font-medium"> #{book.volume}</span>}
</p>
)}
<div className="book-meta">
<span className={`book-kind ${book.kind}`}>{book.kind.toUpperCase()}</span>
{book.language && <span className="book-lang">{book.language.toUpperCase()}</span>}
{/* Meta Tags */}
<div className="flex items-center gap-2 mt-2">
<span className={`
px-2 py-0.5 text-[10px] font-bold uppercase tracking-wider rounded-full
${book.kind === 'cbz' ? 'bg-success-soft text-success' : ''}
${book.kind === 'cbr' ? 'bg-warning-soft text-warning' : ''}
${book.kind === 'pdf' ? 'bg-error-soft text-error' : ''}
`}>
{book.kind}
</span>
{book.language && (
<span className="px-2 py-0.5 text-[10px] font-medium uppercase tracking-wider rounded-full bg-primary-soft text-primary">
{book.language}
</span>
)}
</div>
</div>
</Link>
@@ -44,15 +94,14 @@ export function BookCard({ book, getBookCoverUrl }: BookCardProps) {
}
interface BooksGridProps {
books: BookDto[];
getBookCoverUrl: (bookId: string) => string;
books: (BookDto & { coverUrl?: string })[];
}
export function BooksGrid({ books, getBookCoverUrl }: BooksGridProps) {
export function BooksGrid({ books }: BooksGridProps) {
return (
<div className="books-grid">
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5 xl:grid-cols-6 gap-4">
{books.map((book) => (
<BookCard key={book.id} book={book} getBookCoverUrl={getBookCoverUrl} />
<BookCard key={book.id} book={book} />
))}
</div>
);
@@ -64,8 +113,13 @@ interface EmptyStateProps {
export function EmptyState({ message }: EmptyStateProps) {
return (
<div className="empty-state">
<p>{message}</p>
<div className="flex flex-col items-center justify-center py-16 text-center">
<div className="w-16 h-16 mb-4 text-muted/30">
<svg fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={1.5} d="M12 6.253v13m0-13C10.832 5.477 9.246 5 7.5 5S4.168 5.477 3 6.253v13C4.168 18.477 5.754 18 7.5 18s3.332.477 4.5 1.253m0-13C13.168 5.477 14.754 5 16.5 5c1.747 0 3.332.477 4.5 1.253v13C19.832 18.477 18.247 18 16.5 18c-1.746 0-3.332.477-4.5 1.253" />
</svg>
</div>
<p className="text-muted text-lg">{message}</p>
</div>
);
}