mirror of
https://github.com/Radiquum/AniX.git
synced 2025-04-06 00:04:39 +00:00
feat: add state to inputs for create new collection page
This commit is contained in:
parent
b46dc367cb
commit
2e64548f7a
4 changed files with 231 additions and 4 deletions
18
app/collections/create/page.tsx
Normal file
18
app/collections/create/page.tsx
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
import { CreateCollectionPage } from "#/pages/CreateCollection";
|
||||||
|
import dynamic from "next/dynamic";
|
||||||
|
|
||||||
|
export const metadata = {
|
||||||
|
title: "Создание коллекции",
|
||||||
|
description: "Создание новой коллекции",
|
||||||
|
};
|
||||||
|
|
||||||
|
const CreateCollectionDynamic = dynamic(
|
||||||
|
() => Promise.resolve(CreateCollectionPage),
|
||||||
|
{
|
||||||
|
ssr: false,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
export default function Collections() {
|
||||||
|
return <CreateCollectionDynamic />;
|
||||||
|
}
|
|
@ -2,7 +2,7 @@ import Link from "next/link";
|
||||||
|
|
||||||
export const AddCollectionLink = (props: any) => {
|
export const AddCollectionLink = (props: any) => {
|
||||||
return (
|
return (
|
||||||
<Link href={`/collection/create`}>
|
<Link href={`/collections/create`}>
|
||||||
<div className="flex flex-col items-center justify-center w-full gap-2 text-black transition-colors bg-gray-100 border hover:bg-gray-200 border-gray-50 hover:border-gray-100 dark:border-gray-700 dark:hover:border-gray-600 dark:hover:bg-gray-500 aspect-video group dark:bg-gray-600 dark:text-white">
|
<div className="flex flex-col items-center justify-center w-full gap-2 text-black transition-colors bg-gray-100 border hover:bg-gray-200 border-gray-50 hover:border-gray-100 dark:border-gray-700 dark:hover:border-gray-600 dark:hover:bg-gray-500 aspect-video group dark:bg-gray-600 dark:text-white">
|
||||||
<span className="w-8 h-8 iconify mdi--plus-circle dark:fill-white"></span>
|
<span className="w-8 h-8 iconify mdi--plus-circle dark:fill-white"></span>
|
||||||
<p>Новая коллекция</p>
|
<p>Новая коллекция</p>
|
||||||
|
|
|
@ -54,8 +54,14 @@ export const CollectionInfoControls = (props: {
|
||||||
)}
|
)}
|
||||||
{userStore.user && userStore.user.id == props.authorId && (
|
{userStore.user && userStore.user.id == props.authorId && (
|
||||||
<div className="flex flex-wrap gap-2">
|
<div className="flex flex-wrap gap-2">
|
||||||
<Button color={"blue"} className="w-full sm:max-w-64">
|
<Button
|
||||||
<span className={`iconify w-6 h-6 mr-2 mdi--pencil`}></span>{" "}
|
color={"blue"}
|
||||||
|
className="w-full sm:max-w-64"
|
||||||
|
onClick={() =>
|
||||||
|
router.push("/collections/create?mode=edit&id=" + props.id)
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<span className="w-6 h-6 mr-2 iconify mdi--pencil"></span>{" "}
|
||||||
Редактировать
|
Редактировать
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
|
@ -63,7 +69,7 @@ export const CollectionInfoControls = (props: {
|
||||||
className="w-full sm:max-w-64"
|
className="w-full sm:max-w-64"
|
||||||
onClick={() => _deleteCollection()}
|
onClick={() => _deleteCollection()}
|
||||||
>
|
>
|
||||||
<span className={`iconify w-6 h-6 mr-2 mdi--trash`}></span> Удалить
|
<span className="w-6 h-6 mr-2 iconify mdi--trash"></span> Удалить
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
203
app/pages/CreateCollection.tsx
Normal file
203
app/pages/CreateCollection.tsx
Normal file
|
@ -0,0 +1,203 @@
|
||||||
|
"use client";
|
||||||
|
import { useUserStore } from "#/store/auth";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { useSearchParams, useRouter } from "next/navigation";
|
||||||
|
import { ENDPOINTS } from "#/api/config";
|
||||||
|
import {
|
||||||
|
Card,
|
||||||
|
Button,
|
||||||
|
Checkbox,
|
||||||
|
TextInput,
|
||||||
|
Textarea,
|
||||||
|
FileInput,
|
||||||
|
Label,
|
||||||
|
} from "flowbite-react";
|
||||||
|
|
||||||
|
export const CreateCollectionPage = () => {
|
||||||
|
const userStore = useUserStore();
|
||||||
|
const searchParams = useSearchParams();
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
const [edit, setEdit] = useState(false);
|
||||||
|
|
||||||
|
const [imageData, setImageData] = useState<string>(null);
|
||||||
|
const [imageUrl, setImageUrl] = useState<string>(null);
|
||||||
|
const [isPrivate, setIsPrivate] = useState(false);
|
||||||
|
const [collectionInfo, setCollectionInfo] = useState({
|
||||||
|
title: "",
|
||||||
|
description: "",
|
||||||
|
});
|
||||||
|
|
||||||
|
const collection_id = searchParams.get("id") || null;
|
||||||
|
const mode = searchParams.get("mode") || null;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
async function _checkMode() {
|
||||||
|
if (mode === "edit" && collection_id) {
|
||||||
|
const res = await fetch(
|
||||||
|
`${ENDPOINTS.collection.base}/${collection_id}?token=${userStore.token}`
|
||||||
|
);
|
||||||
|
const data = await res.json();
|
||||||
|
|
||||||
|
if (
|
||||||
|
mode === "edit" &&
|
||||||
|
userStore.user.id == data.collection.creator.id
|
||||||
|
) {
|
||||||
|
setEdit(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (userStore.user) {
|
||||||
|
_checkMode();
|
||||||
|
}
|
||||||
|
}, [userStore.user]);
|
||||||
|
|
||||||
|
const handleFileRead = (e, fileReader, type) => {
|
||||||
|
const content = fileReader.result;
|
||||||
|
if (type === "URL") {
|
||||||
|
setImageUrl(content);
|
||||||
|
} else {
|
||||||
|
setImageData(content);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleFilePreview = (file) => {
|
||||||
|
const fileReader = new FileReader();
|
||||||
|
fileReader.onloadend = (e) => {
|
||||||
|
handleFileRead(e, fileReader, "URL");
|
||||||
|
};
|
||||||
|
fileReader.readAsDataURL(file);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleFileLoad = (file) => {
|
||||||
|
const fileReader = new FileReader();
|
||||||
|
fileReader.onloadend = (e) => {
|
||||||
|
handleFileRead(e, fileReader, "TEXT");
|
||||||
|
};
|
||||||
|
fileReader.readAsText(file);
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleInput(e) {
|
||||||
|
setCollectionInfo({
|
||||||
|
...collectionInfo,
|
||||||
|
[e.target.name]: e.target.value,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function submit(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
console.log({
|
||||||
|
...collectionInfo,
|
||||||
|
private: isPrivate,
|
||||||
|
image: imageData,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<main className="container pt-2 pb-16 mx-auto sm:pt-4 sm:pb-0">
|
||||||
|
<Card>
|
||||||
|
<p className="text-xl font-bold">
|
||||||
|
{edit ? "Редактирование коллекции" : "Создание коллекции"}
|
||||||
|
</p>
|
||||||
|
<form
|
||||||
|
className="flex flex-wrap w-full gap-2"
|
||||||
|
onSubmit={(e) => submit(e)}
|
||||||
|
>
|
||||||
|
<Label
|
||||||
|
htmlFor="dropzone-file"
|
||||||
|
className="flex flex-col items-center w-[600px] h-[337px] border-2 border-gray-300 border-dashed rounded-lg cursor-pointer bg-gray-50 hover:bg-gray-100 dark:border-gray-600 dark:bg-gray-700 dark:hover:border-gray-500 dark:hover:bg-gray-600"
|
||||||
|
>
|
||||||
|
<div className="flex flex-col items-center justify-center w-[595px] h-[inherit] rounded-[inherit] pt-5 pb-6 overflow-hidden">
|
||||||
|
{!imageUrl ? (
|
||||||
|
<>
|
||||||
|
<svg
|
||||||
|
className="w-8 h-8 mb-4 text-gray-500 dark:text-gray-400"
|
||||||
|
aria-hidden="true"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
fill="none"
|
||||||
|
viewBox="0 0 20 16"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
strokeWidth="2"
|
||||||
|
d="M13 13h3a3 3 0 0 0 0-6h-.025A5.56 5.56 0 0 0 16 6.5 5.5 5.5 0 0 0 5.207 5.021C5.137 5.017 5.071 5 5 5a4 4 0 0 0 0 8h2.167M10 15V6m0 0L8 8m2-2 2 2"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
<p className="mb-2 text-sm text-gray-500 dark:text-gray-400">
|
||||||
|
<span className="font-semibold">Нажмите для загрузки</span>{" "}
|
||||||
|
или перетащите файл
|
||||||
|
</p>
|
||||||
|
<p className="text-xs text-gray-500 dark:text-gray-400">
|
||||||
|
PNG или JPG (Макс. 600x337 пикселей)
|
||||||
|
</p>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<img
|
||||||
|
src={imageUrl}
|
||||||
|
className="object-cover w-[inherit] h-[inherit]"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<FileInput
|
||||||
|
id="dropzone-file"
|
||||||
|
className="hidden"
|
||||||
|
accept="image/jpg, image/jpeg, image/png"
|
||||||
|
onChange={(e) => {
|
||||||
|
handleFilePreview(e.target.files[0]);
|
||||||
|
handleFileLoad(e.target.files[0]);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Label>
|
||||||
|
<div className="flex-1">
|
||||||
|
<div className="block mb-2">
|
||||||
|
<Label
|
||||||
|
htmlFor="title"
|
||||||
|
value="Название (минимум 10, максимум 60 символов)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<TextInput
|
||||||
|
id="title"
|
||||||
|
name="title"
|
||||||
|
type="text"
|
||||||
|
sizing="md"
|
||||||
|
className="w-full"
|
||||||
|
required={true}
|
||||||
|
onChange={(e) => handleInput(e)}
|
||||||
|
value={collectionInfo.title}
|
||||||
|
/>
|
||||||
|
<div className="block mt-4 mb-2">
|
||||||
|
<Label
|
||||||
|
htmlFor="description"
|
||||||
|
value="Описание (максимум 1000 символов)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<Textarea
|
||||||
|
rows={4}
|
||||||
|
id="description"
|
||||||
|
className="w-full"
|
||||||
|
name="description"
|
||||||
|
onChange={(e) => handleInput(e)}
|
||||||
|
value={collectionInfo.description}
|
||||||
|
/>
|
||||||
|
<div className="mt-4">
|
||||||
|
<div className="flex items-center gap-1">
|
||||||
|
<Checkbox
|
||||||
|
id="private"
|
||||||
|
name="private"
|
||||||
|
checked={isPrivate}
|
||||||
|
onChange={(e) => setIsPrivate(e.target.checked)}
|
||||||
|
/>
|
||||||
|
<Label htmlFor="private" value="Приватная коллекция" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<Button color={"blue"} className="mt-4" type="submit">
|
||||||
|
{edit ? "Обновить" : "Создать"}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</Card>
|
||||||
|
</main>
|
||||||
|
);
|
||||||
|
};
|
Loading…
Add table
Reference in a new issue