From 74bc8014e9383de6cc845da6fb0cf5c7dd5aa021 Mon Sep 17 00:00:00 2001 From: boragenel Date: Thu, 23 Jan 2025 11:10:27 +0300 Subject: [PATCH] Auto stash before merge of "main" and "origin/main" --- src/app/create-game/page.tsx | 441 +++++++++++++++++++++++++++++++- src/app/create-post/page.tsx | 11 +- src/components/editor/index.tsx | 9 +- 3 files changed, 454 insertions(+), 7 deletions(-) diff --git a/src/app/create-game/page.tsx b/src/app/create-game/page.tsx index 30344eb..dee117b 100644 --- a/src/app/create-game/page.tsx +++ b/src/app/create-game/page.tsx @@ -1,9 +1,446 @@ "use client"; +import Editor from "@/components/editor"; +import { getCookie, hasCookie } from "@/helpers/cookie"; +import { Avatar, Button, Form, Input, Spacer } from "@nextui-org/react"; +import { LoaderCircle } from "lucide-react"; +import { redirect } from "next/navigation"; +import { ReactNode, useEffect, useState } from "react"; +import { toast } from "react-toastify"; +import sanitizeHtml from "sanitize-html"; +import Select, { MultiValue, StylesConfig } from "react-select"; +import { useTheme } from "next-themes"; +import Timers from "@/components/timers"; +import Streams from "@/components/streams"; + export default function CreateGamePage() { + const [title, setTitle] = useState(""); + const [content, setContent] = useState(""); + const [errors, setErrors] = useState({}); + const [waitingPost, setWaitingPost] = useState(false); + const [selectedTags, setSelectedTags] = useState | null>(null); + const [mounted, setMounted] = useState(false); + const [options, setOptions] = useState< + { + value: string; + label: ReactNode; + id: number; + isFixed: boolean; + }[] + >(); + const { theme } = useTheme(); + // Add these state variables after existing useState declarations +const [windowsLink, setWindowsLink] = useState(""); +const [linuxLink, setLinuxLink] = useState(""); +const [macLink, setMacLink] = useState(""); +const [webGLLink, setWebGLLink] = useState(""); +const [gameSlug, setGameSlug] = useState(""); +const [thumbnailUrl, setThumbnailUrl] = useState(""); +const [authorSearch, setAuthorSearch] = useState(""); +const [selectedAuthors, setSelectedAuthors] = useState>([]); +const [searchResults, setSearchResults] = useState([]); + +// Add this function to handle author search +const handleAuthorSearch = async (query: string) => { + if (query.length < 2) return; + + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? `https://d2jam.com/api/v1/users/search?q=${query}` + : `http://localhost:3005/api/v1/users/search?q=${query}`, + { + headers: { authorization: `Bearer ${getCookie("token")}` }, + credentials: "include", + } + ); + + if (response.ok) { + const data = await response.json(); + setSearchResults(data); + } +}; + + useEffect(() => { + setMounted(true); + + const load = async () => { + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? `https://d2jam.com/api/v1/self?username=${getCookie("user")}` + : `http://localhost:3005/api/v1/self?username=${getCookie("user")}`, + { + headers: { authorization: `Bearer ${getCookie("token")}` }, + credentials: "include", + } + ); + + const user = await response.json(); + + const tagResponse = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? `https://d2jam.com/api/v1/tags` + : `http://localhost:3005/api/v1/tags` + ); + + if (tagResponse.ok) { + const newoptions: { + value: string; + label: ReactNode; + id: number; + isFixed: boolean; + }[] = []; + + for (const tag of await tagResponse.json()) { + if (tag.modOnly && !user.mod) { + continue; + } + newoptions.push({ + value: tag.name, + id: tag.id, + label: ( +
+ {tag.icon && ( + + )} +

+ {tag.name} + {tag.modOnly ? " (Mod Only)" : ""} +

+
+ ), + isFixed: tag.alwaysAdded, + }); + } + + setOptions(newoptions); + setSelectedTags(newoptions.filter((tag) => tag.isFixed)); + } + }; + load(); + }, []); + + const styles: StylesConfig< + { + value: string; + label: ReactNode; + isFixed: boolean; + }, + true + > = { + multiValue: (base, state) => { + return { + ...base, + backgroundColor: state.data.isFixed + ? theme == "dark" + ? "#222" + : "#ddd" + : theme == "dark" + ? "#444" + : "#eee", + }; + }, + multiValueLabel: (base, state) => { + return { + ...base, + color: state.data.isFixed + ? theme == "dark" + ? "#ddd" + : "#222" + : theme == "dark" + ? "#fff" + : "#444", + fontWeight: state.data.isFixed ? "normal" : "bold", + paddingRight: state.data.isFixed ? "8px" : "2px", + }; + }, + multiValueRemove: (base, state) => { + return { + ...base, + display: state.data.isFixed ? "none" : "flex", + color: theme == "dark" ? "#ddd" : "#222", + }; + }, + control: (styles) => ({ + ...styles, + backgroundColor: theme == "dark" ? "#181818" : "#fff", + minWidth: "300px", + }), + menu: (styles) => ({ + ...styles, + backgroundColor: theme == "dark" ? "#181818" : "#fff", + color: theme == "dark" ? "#fff" : "#444", + }), + option: (styles, { isFocused }) => ({ + ...styles, + backgroundColor: isFocused + ? theme == "dark" + ? "#333" + : "#ddd" + : undefined, + }), + }; + return ( -
-

Game creation coming soon

+
+
{ + e.preventDefault(); + + if (!title && !content) { + setErrors({ + title: "Please enter a valid title", + content: "Please enter valid content", + }); + toast.error("Please enter valid content"); + return; + } + + if (!title) { + setErrors({ title: "Please enter a valid title" }); + return; + } + + if (!content) { + setErrors({ content: "Please enter valid content" }); + toast.error("Please enter valid content"); + return; + } + + if (!hasCookie("token")) { + setErrors({ content: "You are not logged in" }); + return; + } + + const sanitizedHtml = sanitizeHtml(content); + setWaitingPost(true); + + const tags = []; + + if (selectedTags) { + for (const tag of selectedTags) { + tags.push( + options?.filter((option) => option.value == tag.value)[0].id + ); + } + } + + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/post" + : "http://localhost:3005/api/v1/post", + { + body: JSON.stringify({ + title: title, + content: sanitizedHtml, + username: getCookie("user"), + tags, + gameSlug, + thumbnailUrl, + windowsLink, + linuxLink, + macLink, + webGLLink, + authors: selectedAuthors.map(author => author.id) + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie("token")}`, + }, + credentials: "include", + } + ); + + if (response.status == 401) { + setErrors({ content: "Invalid user" }); + setWaitingPost(false); + return; + } + + if (response.ok) { + toast.success("Successfully created post"); + setWaitingPost(false); + redirect("/"); + } else { + toast.error("An error occured"); + setWaitingPost(false); + } + }} + > + + + + +
+ + { + setAuthorSearch(value); + handleAuthorSearch(value); + }} + /> + {searchResults.length > 0 && ( +
+ {searchResults.map((user) => ( +
{ + setSelectedAuthors([...selectedAuthors, user]); + setSearchResults([]); + setAuthorSearch(""); + }} + > + {user.name} +
+ ))} +
+ )} +
+ {selectedAuthors.map((author) => ( +
+ {author.name} + +
+ ))}
+
+ + + + + + {mounted && ( + + +
+ + + + +
+ + + +
+

Game Metrics

+
+
+

Views

+

0

+
+
+

Downloads

+

0

+
+
+

Rating

+

N/A

+
+
+

Comments

+

0

+
+
+
+
+ +
+
+ +
+ + +
+
); } diff --git a/src/app/create-post/page.tsx b/src/app/create-post/page.tsx index c810b96..4ff98f0 100644 --- a/src/app/create-post/page.tsx +++ b/src/app/create-post/page.tsx @@ -17,7 +17,8 @@ import { toast } from "react-toastify"; import sanitizeHtml from "sanitize-html"; import Select, { MultiValue, StylesConfig } from "react-select"; import { useTheme } from "next-themes"; -import { UserType } from "@/types/UserType"; +import Timers from "@/components/timers"; +import Streams from "@/components/streams"; export default function CreatePostPage() { const [title, setTitle] = useState(""); @@ -168,7 +169,7 @@ export default function CreatePostPage() { }; return ( -
+
-
+
+ + +
+ ); } diff --git a/src/components/editor/index.tsx b/src/components/editor/index.tsx index e075c7b..aa484d6 100644 --- a/src/components/editor/index.tsx +++ b/src/components/editor/index.tsx @@ -40,11 +40,12 @@ import { useTheme } from "next-themes"; type EditorProps = { content: string; setContent: (content: string) => void; + gameEditor: boolean; }; const limit = 32767; -export default function Editor({ content, setContent }: EditorProps) { +export default function Editor({ content, setContent,gameEditor }: EditorProps) { const { theme } = useTheme(); const editor = useEditor({ @@ -97,7 +98,11 @@ export default function Editor({ content, setContent }: EditorProps) { editorProps: { attributes: { class: - "prose dark:prose-invert min-h-[150px] max-h-[400px] overflow-y-auto cursor-text rounded-md border p-5 focus-within:outline-none focus-within:border-blue-500 !duration-250 !ease-linear !transition-all", + "prose dark:prose-invert " + + (gameEditor + ? "min-h-[600px] max-h-[600px]" + : "min-h-[150px] max-h-[400px]") + + " overflow-y-auto cursor-text rounded-md border p-5 focus-within:outline-none focus-within:border-blue-500 !duration-250 !ease-linear !transition-all", }, }, });