feat: home beautiful images

This commit is contained in:
Julien Froidefond
2025-02-16 23:16:01 +01:00
parent 54115ac9cc
commit e8be6cb724
4 changed files with 83 additions and 22 deletions

View File

@@ -1,34 +1,25 @@
import { NextResponse } from "next/server"; import { NextRequest, NextResponse } from "next/server";
import { BookService } from "@/lib/services/book.service"; import { ImageService } from "@/lib/services/image.service";
export const dynamic = "force-dynamic"; export const dynamic = "force-dynamic";
export async function GET( export async function GET(
request: Request, request: NextRequest,
{ params }: { params: { bookId: string; pageNumber: string } } { params }: { params: { bookId: string; pageNumber: string } }
) { ) {
try { try {
// Convertir le numéro de page en nombre const { buffer, contentType } = await ImageService.getImage(
const pageNumber = parseInt(params.pageNumber); `books/${params.bookId}/pages/${params.pageNumber}`
if (isNaN(pageNumber) || pageNumber < 1) { );
return NextResponse.json({ error: "Numéro de page invalide" }, { status: 400 });
}
const response = await BookService.getPage(params.bookId, pageNumber);
const buffer = await response.arrayBuffer();
const headers = new Headers();
headers.set("Content-Type", response.headers.get("Content-Type") || "image/jpeg");
headers.set("Cache-Control", "public, max-age=31536000"); // Cache for 1 year
return new NextResponse(buffer, { return new NextResponse(buffer, {
status: 200, headers: {
headers, "Content-Type": contentType || "image/jpeg",
"Cache-Control": "public, max-age=31536000, immutable",
},
}); });
} catch (error) { } catch (error) {
console.error("API Book Page - Erreur:", error); console.error("Erreur lors de la récupération de la page du livre:", error);
return NextResponse.json( return new NextResponse("Erreur lors de la récupération de la page", { status: 500 });
{ error: "Erreur lors de la récupération de la page" },
{ status: 500 }
);
} }
} }

View File

@@ -0,0 +1,39 @@
import { NextRequest, NextResponse } from "next/server";
import { ImageService } from "@/lib/services/image.service";
import { SeriesService } from "@/lib/services/series.service";
export const dynamic = "force-dynamic";
export async function GET(request: NextRequest, { params }: { params: { seriesId: string } }) {
try {
// Récupérer l'ID du premier livre
const firstBookId = await SeriesService.getFirstBook(params.seriesId);
// Récupérer la première page du premier livre
const { buffer, contentType } = await ImageService.getImage(`books/${firstBookId}/pages/1`);
return new NextResponse(buffer, {
headers: {
"Content-Type": contentType || "image/jpeg",
"Cache-Control": "public, max-age=31536000, immutable",
},
});
} catch (error) {
console.error("Erreur lors de la récupération de la première page de la série:", error);
// En cas d'erreur, on essaie de récupérer le thumbnail comme fallback
try {
const { buffer, contentType } = await ImageService.getImage(
`series/${params.seriesId}/thumbnail`
);
return new NextResponse(buffer, {
headers: {
"Content-Type": contentType || "image/jpeg",
"Cache-Control": "public, max-age=31536000, immutable",
},
});
} catch (fallbackError) {
return new NextResponse("Erreur lors de la récupération de l'image", { status: 500 });
}
}
}

View File

@@ -111,7 +111,9 @@ function MediaCard({ item, onClick }: MediaCardProps) {
<> <>
<ImageLoader isLoading={imageLoading} /> <ImageLoader isLoading={imageLoading} />
<Image <Image
src={`/api/komga/images/${isSeries ? "series" : "books"}/${item.id}/thumbnail`} src={`/api/komga/images/${isSeries ? "series" : "books"}/${item.id}/${
isSeries ? "first-page" : "pages/1"
}`}
alt={`Couverture de ${title}`} alt={`Couverture de ${title}`}
fill fill
className={cn( className={cn(

View File

@@ -44,4 +44,33 @@ export class SeriesService extends BaseApiService {
return this.handleError(error, "Impossible de récupérer les tomes"); return this.handleError(error, "Impossible de récupérer les tomes");
} }
} }
static async getFirstBook(seriesId: string): Promise<string> {
try {
const config = await this.getKomgaConfig();
const url = this.buildUrl(config, `series/${seriesId}/books`);
const headers = this.getAuthHeaders(config);
return this.fetchWithCache<string>(
`series-first-book-${seriesId}`,
async () => {
const response = await fetch(`${url}?page=0&size=1`, { headers });
if (!response.ok) {
throw new Error(`Erreur HTTP: ${response.status}`);
}
const data = await response.json();
if (!data.content || data.content.length === 0) {
throw new Error("Aucun livre trouvé dans la série");
}
return data.content[0].id;
},
"SERIES"
);
} catch (error) {
console.error("Erreur lors de la récupération du premier livre:", error);
return this.handleError(error, "Impossible de récupérer le premier livre");
}
}
} }