feat: home beautiful images
This commit is contained in:
@@ -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 }
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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(
|
||||||
|
|||||||
@@ -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");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user