mirror of
https://github.com/Radiquum/AniX.git
synced 2025-04-06 00:04:39 +00:00
feat: add collection favorite button and delete button if own collection
fix: private collection loading fix: wrong lists bar percentages
This commit is contained in:
parent
9a5d1eb6bd
commit
82e38f02b4
6 changed files with 131 additions and 35 deletions
|
@ -13,13 +13,15 @@ export async function generateMetadata(
|
||||||
const previousOG = (await parent).openGraph;
|
const previousOG = (await parent).openGraph;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
title: "коллекция - " + collection.collection.title,
|
title: collection.collection
|
||||||
description: collection.collection.description,
|
? "коллекция - " + collection.collection.title
|
||||||
|
: "Приватная коллекция",
|
||||||
|
description: collection.collection && collection.collection.description,
|
||||||
openGraph: {
|
openGraph: {
|
||||||
...previousOG,
|
...previousOG,
|
||||||
images: [
|
images: [
|
||||||
{
|
{
|
||||||
url: collection.collection.image, // Must be an absolute URL
|
url: collection.collection && collection.collection.image, // Must be an absolute URL
|
||||||
width: 600,
|
width: 600,
|
||||||
height: 800,
|
height: 800,
|
||||||
},
|
},
|
||||||
|
|
|
@ -14,7 +14,7 @@ export const CollectionInfoBasics = (props: {
|
||||||
updateDate: number;
|
updateDate: number;
|
||||||
}) => {
|
}) => {
|
||||||
return (
|
return (
|
||||||
<Card className="w-full max-w-full lg:max-w-[50%]">
|
<Card className="flex-1 w-full">
|
||||||
<div className="flex flex-col items-end justify-between sm:items-center sm:flex-row">
|
<div className="flex flex-col items-end justify-between sm:items-center sm:flex-row">
|
||||||
<div className="flex flex-col gap-1">
|
<div className="flex flex-col gap-1">
|
||||||
<p>создана: {unixToDate(props.creationDate, "full")}</p>
|
<p>создана: {unixToDate(props.creationDate, "full")}</p>
|
||||||
|
|
72
app/components/CollectionInfo/CollectionInfoControls.tsx
Normal file
72
app/components/CollectionInfo/CollectionInfoControls.tsx
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
"use client";
|
||||||
|
import { Card, Button } from "flowbite-react";
|
||||||
|
import { useState } from "react";
|
||||||
|
import { useUserStore } from "#/store/auth";
|
||||||
|
import { ENDPOINTS } from "#/api/config";
|
||||||
|
import { useRouter } from "next/navigation";
|
||||||
|
|
||||||
|
export const CollectionInfoControls = (props: {
|
||||||
|
isFavorite: boolean;
|
||||||
|
id: number;
|
||||||
|
authorId: number;
|
||||||
|
isPrivate: boolean;
|
||||||
|
}) => {
|
||||||
|
const [isFavorite, setIsFavorite] = useState(props.isFavorite);
|
||||||
|
const userStore = useUserStore();
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
async function _addToFavorite() {
|
||||||
|
if (userStore.user) {
|
||||||
|
setIsFavorite(!isFavorite);
|
||||||
|
if (isFavorite) {
|
||||||
|
fetch(
|
||||||
|
`${ENDPOINTS.collection.favoriteCollections}/delete/${props.id}?token=${userStore.token}`
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
fetch(
|
||||||
|
`${ENDPOINTS.collection.favoriteCollections}/add/${props.id}?token=${userStore.token}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function _deleteCollection() {
|
||||||
|
if (userStore.user) {
|
||||||
|
fetch(
|
||||||
|
`${ENDPOINTS.collection.delete}/${props.id}?token=${userStore.token}`
|
||||||
|
);
|
||||||
|
router.push("/collections");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card className="w-full h-fit ">
|
||||||
|
<Button color={"blue"} onClick={() => _addToFavorite()}>
|
||||||
|
<span
|
||||||
|
className={`iconify w-6 h-6 mr-2 ${
|
||||||
|
isFavorite ? "mdi--heart" : "mdi--heart-outline"
|
||||||
|
}`}
|
||||||
|
></span>
|
||||||
|
{!isFavorite ? "Добавить в избранное" : "Убрать из избранного"}
|
||||||
|
</Button>
|
||||||
|
{props.isPrivate && (
|
||||||
|
<p>Это приватная коллекция, доступ к ней имеете только вы</p>
|
||||||
|
)}
|
||||||
|
{userStore.user && userStore.user.id == props.authorId && (
|
||||||
|
<div className="flex flex-wrap gap-2">
|
||||||
|
<Button color={"blue"} className="w-full sm:max-w-64">
|
||||||
|
<span className={`iconify w-6 h-6 mr-2 mdi--pencil`}></span>{" "}
|
||||||
|
Редактировать
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
color={"red"}
|
||||||
|
className="w-full sm:max-w-64"
|
||||||
|
onClick={() => _deleteCollection()}
|
||||||
|
>
|
||||||
|
<span className={`iconify w-6 h-6 mr-2 mdi--trash`}></span> Удалить
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
};
|
|
@ -9,7 +9,7 @@ export const CollectionInfoLists = (props: {
|
||||||
total: number;
|
total: number;
|
||||||
}) => {
|
}) => {
|
||||||
return (
|
return (
|
||||||
<Card className="w-full max-w-full lg:max-w-[48%] h-fit ">
|
<Card className="w-full h-fit ">
|
||||||
<div
|
<div
|
||||||
className="flex w-full h-8 overflow-hidden rounded-md"
|
className="flex w-full h-8 overflow-hidden rounded-md"
|
||||||
style={
|
style={
|
||||||
|
@ -20,7 +20,7 @@ export const CollectionInfoLists = (props: {
|
||||||
"--watched-percent": `calc(var(--width-of-one) * (${props.completed} / ${props.total} * 100%))`,
|
"--watched-percent": `calc(var(--width-of-one) * (${props.completed} / ${props.total} * 100%))`,
|
||||||
"--delayed-percent": `calc(var(--width-of-one) * (${props.delayed} / ${props.total} * 100%))`,
|
"--delayed-percent": `calc(var(--width-of-one) * (${props.delayed} / ${props.total} * 100%))`,
|
||||||
"--abandoned-percent": `calc(var(--width-of-one) * (${props.abandoned} / ${props.total} * 100%))`,
|
"--abandoned-percent": `calc(var(--width-of-one) * (${props.abandoned} / ${props.total} * 100%))`,
|
||||||
"--no-list-percent": `calc(var(--width-of-one) * (${props.total - (props.watching - props.planned - props.completed - props.delayed - props.abandoned)} / ${props.total} * 100%))`,
|
"--no-list-percent": `calc(var(--width-of-one) * (${props.total - (props.watching + props.planned + props.completed + props.delayed + props.abandoned)} / ${props.total} * 100%))`,
|
||||||
} as React.CSSProperties
|
} as React.CSSProperties
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
|
|
|
@ -1,6 +1,9 @@
|
||||||
import { ReleaseLink } from "../ReleaseLink/ReleaseLink";
|
import { ReleaseLink } from "../ReleaseLink/ReleaseLink";
|
||||||
|
|
||||||
export const ReleaseSection = (props: {sectionTitle?: string, content: any}) => {
|
export const ReleaseSection = (props: {
|
||||||
|
sectionTitle?: string;
|
||||||
|
content: any;
|
||||||
|
}) => {
|
||||||
return (
|
return (
|
||||||
<section>
|
<section>
|
||||||
{props.sectionTitle && (
|
{props.sectionTitle && (
|
||||||
|
@ -19,6 +22,7 @@ export const ReleaseSection = (props: {sectionTitle?: string, content: any}) =>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
|
{props.content.length == 1 && <div></div>}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
|
@ -12,6 +12,7 @@ import { ReleaseSection } from "#/components/ReleaseSection/ReleaseSection";
|
||||||
|
|
||||||
import { CollectionInfoBasics } from "#/components/CollectionInfo/CollectionInfo.Basics";
|
import { CollectionInfoBasics } from "#/components/CollectionInfo/CollectionInfo.Basics";
|
||||||
import { CollectionInfoLists } from "#/components/CollectionInfo/CollectionInfoLists";
|
import { CollectionInfoLists } from "#/components/CollectionInfo/CollectionInfoLists";
|
||||||
|
import { CollectionInfoControls } from "#/components/CollectionInfo/CollectionInfoControls";
|
||||||
|
|
||||||
const fetcher = async (url: string) => {
|
const fetcher = async (url: string) => {
|
||||||
const res = await fetch(url);
|
const res = await fetch(url);
|
||||||
|
@ -81,37 +82,54 @@ export const ViewCollectionPage = (props: { id: number }) => {
|
||||||
return (
|
return (
|
||||||
<main className="container pt-2 pb-16 mx-auto sm:pt-4 sm:pb-0">
|
<main className="container pt-2 pb-16 mx-auto sm:pt-4 sm:pb-0">
|
||||||
{collectionInfoIsLoading ? (
|
{collectionInfoIsLoading ? (
|
||||||
<Spinner />
|
<div className="flex items-center justify-center w-full h-screen">
|
||||||
|
<Spinner />
|
||||||
|
</div>
|
||||||
) : (
|
) : (
|
||||||
<>
|
collectionInfo && (
|
||||||
<div className="flex flex-col flex-wrap gap-4 px-2 sm:flex-row">
|
<>
|
||||||
<CollectionInfoBasics
|
<div className="flex flex-col flex-wrap gap-4 px-2 pb-2 sm:flex-row">
|
||||||
image={collectionInfo.collection.image}
|
<CollectionInfoBasics
|
||||||
title={collectionInfo.collection.title}
|
image={collectionInfo.collection.image}
|
||||||
description={collectionInfo.collection.description}
|
title={collectionInfo.collection.title}
|
||||||
authorAvatar={collectionInfo.collection.creator.avatar}
|
description={collectionInfo.collection.description}
|
||||||
authorLogin={collectionInfo.collection.creator.login}
|
authorAvatar={collectionInfo.collection.creator.avatar}
|
||||||
authorId={collectionInfo.collection.creator.id}
|
authorLogin={collectionInfo.collection.creator.login}
|
||||||
creationDate={collectionInfo.collection.creation_date}
|
authorId={collectionInfo.collection.creator.id}
|
||||||
updateDate={collectionInfo.collection.last_update_date}
|
creationDate={collectionInfo.collection.creation_date}
|
||||||
/>
|
updateDate={collectionInfo.collection.last_update_date}
|
||||||
{userStore.token && !isLoading && (
|
/>
|
||||||
<CollectionInfoLists
|
{userStore.token && !isLoading && (
|
||||||
completed={collectionInfo.completed_count}
|
<div className="flex flex-col gap-4 w-full max-w-full lg:max-w-[48%]">
|
||||||
planned={collectionInfo.plan_count}
|
<CollectionInfoLists
|
||||||
abandoned={collectionInfo.dropped_count}
|
completed={collectionInfo.completed_count}
|
||||||
delayed={collectionInfo.hold_on_count}
|
planned={collectionInfo.plan_count}
|
||||||
watching={collectionInfo.watching_count}
|
abandoned={collectionInfo.dropped_count}
|
||||||
total={data[0].total_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}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</div>
|
</>
|
||||||
{isLoading || !content || !isLoadingEnd ? (
|
)
|
||||||
<Spinner />
|
|
||||||
) : (
|
|
||||||
<ReleaseSection content={content} />
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
)}
|
)}
|
||||||
</main>
|
</main>
|
||||||
);
|
);
|
||||||
|
|
Loading…
Add table
Reference in a new issue