mirror of
https://github.com/Radiquum/AniX.git
synced 2025-04-06 00:04:39 +00:00
156 lines
5.4 KiB
TypeScript
156 lines
5.4 KiB
TypeScript
"use client";
|
|
import useSWRInfinite from "swr/infinite";
|
|
import useSWR from "swr";
|
|
import { Spinner } from "#/components/Spinner/Spinner";
|
|
import { useState, useEffect } from "react";
|
|
import { useScrollPosition } from "#/hooks/useScrollPosition";
|
|
import { useUserStore } from "../store/auth";
|
|
import { ENDPOINTS } from "#/api/config";
|
|
import { useRouter } from "next/navigation";
|
|
import { ReleaseSection } from "#/components/ReleaseSection/ReleaseSection";
|
|
|
|
import { CollectionInfoBasics } from "#/components/CollectionInfo/CollectionInfo.Basics";
|
|
import { InfoLists } from "#/components/InfoLists/InfoLists";
|
|
import { CollectionInfoControls } from "#/components/CollectionInfo/CollectionInfoControls";
|
|
import { CommentsMain } from "#/components/Comments/Comments.Main";
|
|
|
|
const fetcher = async (url: string) => {
|
|
const res = await fetch(url);
|
|
|
|
if (!res.ok) {
|
|
const error = new Error(
|
|
`An error occurred while fetching the data. status: ${res.status}`
|
|
);
|
|
error.message = await res.json();
|
|
throw error;
|
|
}
|
|
|
|
return res.json();
|
|
};
|
|
|
|
export const ViewCollectionPage = (props: { id: number }) => {
|
|
const userStore = useUserStore();
|
|
const [isLoadingEnd, setIsLoadingEnd] = useState(false);
|
|
const router = useRouter();
|
|
|
|
function useFetchCollectionInfo(type: "info" | "comments") {
|
|
let url: string;
|
|
|
|
if (type == "info") {
|
|
url = `${ENDPOINTS.collection.base}/${props.id}`;
|
|
} else if (type == "comments") {
|
|
url = `${ENDPOINTS.collection.base}/comment/all/${props.id}/0?sort=3`;
|
|
}
|
|
|
|
if (userStore.token) {
|
|
url += `${type != "info" ? "&" : "?"}token=${userStore.token}`;
|
|
}
|
|
|
|
const { data, isLoading } = useSWR(url, fetcher);
|
|
return [data, isLoading];
|
|
}
|
|
const getKey = (pageIndex: number, previousPageData: any) => {
|
|
if (previousPageData && !previousPageData.content.length) return null;
|
|
let url: string = `${ENDPOINTS.collection.base}/${props.id}/releases/${pageIndex}`;
|
|
if (userStore.token) {
|
|
url += `?token=${userStore.token}`;
|
|
}
|
|
return url;
|
|
};
|
|
|
|
const [collectionInfo, collectionInfoIsLoading] =
|
|
useFetchCollectionInfo("info");
|
|
const [collectionComments, collectionCommentsIsLoading] =
|
|
useFetchCollectionInfo("comments");
|
|
|
|
const { data, error, isLoading, size, setSize } = useSWRInfinite(
|
|
getKey,
|
|
fetcher,
|
|
{ initialSize: 2 }
|
|
);
|
|
|
|
const [content, setContent] = useState(null);
|
|
useEffect(() => {
|
|
if (data) {
|
|
let allReleases = [];
|
|
for (let i = 0; i < data.length; i++) {
|
|
allReleases.push(...data[i].content);
|
|
}
|
|
setContent(allReleases);
|
|
setIsLoadingEnd(true);
|
|
}
|
|
}, [data]);
|
|
|
|
const scrollPosition = useScrollPosition();
|
|
useEffect(() => {
|
|
if (scrollPosition >= 98 && scrollPosition <= 99) {
|
|
setSize(size + 1);
|
|
}
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [scrollPosition]);
|
|
|
|
return (
|
|
<>
|
|
{collectionInfoIsLoading ? (
|
|
<div className="flex items-center justify-center w-full h-screen">
|
|
<Spinner />
|
|
</div>
|
|
) : (
|
|
collectionInfo && (
|
|
<>
|
|
<div className="flex flex-col flex-wrap gap-2 px-2 pb-2 sm:flex-row">
|
|
<CollectionInfoBasics
|
|
image={collectionInfo.collection.image}
|
|
title={collectionInfo.collection.title}
|
|
description={collectionInfo.collection.description}
|
|
authorAvatar={collectionInfo.collection.creator.avatar}
|
|
authorLogin={collectionInfo.collection.creator.login}
|
|
authorId={collectionInfo.collection.creator.id}
|
|
creationDate={collectionInfo.collection.creation_date}
|
|
updateDate={collectionInfo.collection.last_update_date}
|
|
/>
|
|
<div className="flex flex-col gap-2 w-full max-w-full lg:max-w-[48%]">
|
|
{collectionComments && !collectionCommentsIsLoading && (
|
|
<CommentsMain
|
|
release_id={props.id}
|
|
token={userStore.token}
|
|
comments={collectionComments.content.slice(0, 2)}
|
|
type="collection"
|
|
/>
|
|
)}
|
|
{userStore.token && !isLoading && (
|
|
<>
|
|
<InfoLists
|
|
completed={collectionInfo.completed_count}
|
|
planned={collectionInfo.plan_count}
|
|
abandoned={collectionInfo.dropped_count}
|
|
delayed={collectionInfo.hold_on_count}
|
|
watching={collectionInfo.watching_count}
|
|
total={data[0].total_count}
|
|
/>
|
|
<CollectionInfoControls
|
|
isFavorite={collectionInfo.collection.is_favorite}
|
|
id={collectionInfo.collection.id}
|
|
authorId={collectionInfo.collection.creator.id}
|
|
isPrivate={collectionInfo.collection.is_private}
|
|
/>
|
|
</>
|
|
)}
|
|
</div>
|
|
</div>
|
|
{isLoading || !content || !isLoadingEnd ? (
|
|
<div className="flex items-center justify-center w-full h-screen">
|
|
<Spinner />
|
|
</div>
|
|
) : (
|
|
<ReleaseSection
|
|
sectionTitle={"Релизов в коллекции: " + data[0].total_count}
|
|
content={content}
|
|
/>
|
|
)}
|
|
</>
|
|
)
|
|
)}
|
|
</>
|
|
);
|
|
};
|