Files
stripstream-librarian/apps/backoffice/app/components/BookCard.tsx
Froidefond Julien c2f1c7732e feat(ui): Lazy loading images + dark mode job highlight
- Ajout loading='lazy' sur les couvertures de livres (BookCard + detail)
- Correction couleur job-highlighted en dark mode (fond bleu foncé)
- Animation pulse adaptée pour le dark mode
2026-03-06 12:11:53 +01:00

72 lines
1.8 KiB
TypeScript

import Image from "next/image";
import Link from "next/link";
import { BookDto } from "../../lib/api";
interface BookCardProps {
book: BookDto;
getBookCoverUrl: (bookId: string) => string;
}
export function BookCard({ book, getBookCoverUrl }: BookCardProps) {
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>
<div className="book-info">
<h3 className="book-title" title={book.title}>
{book.title}
</h3>
{book.author && (
<p className="book-author">{book.author}</p>
)}
{book.series && (
<p className="book-series">
{book.series}
{book.volume && ` #${book.volume}`}
</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>}
</div>
</div>
</Link>
);
}
interface BooksGridProps {
books: BookDto[];
getBookCoverUrl: (bookId: string) => string;
}
export function BooksGrid({ books, getBookCoverUrl }: BooksGridProps) {
return (
<div className="books-grid">
{books.map((book) => (
<BookCard key={book.id} book={book} getBookCoverUrl={getBookCoverUrl} />
))}
</div>
);
}
interface EmptyStateProps {
message: string;
}
export function EmptyState({ message }: EmptyStateProps) {
return (
<div className="empty-state">
<p>{message}</p>
</div>
);
}