From 42c730e073f5e0eea1fa4730b3144ce8670eb149 Mon Sep 17 00:00:00 2001 From: Ategon Date: Wed, 22 Jan 2025 22:08:09 -0500 Subject: [PATCH] Add stickied posts --- src/app/create-post/page.tsx | 35 +++++++++-- src/app/p/[slug]/page.tsx | 77 +++++++++++++++++++++++++ src/components/posts/PostCard.tsx | 77 +++++++++++++++++++++++++ src/components/posts/StickyPostCard.tsx | 54 +++++++++++++++++ src/components/posts/index.tsx | 68 ++++++++++++++++++++++ src/types/PostType.ts | 1 + 6 files changed, 307 insertions(+), 5 deletions(-) create mode 100644 src/components/posts/StickyPostCard.tsx diff --git a/src/app/create-post/page.tsx b/src/app/create-post/page.tsx index e25923f..c810b96 100644 --- a/src/app/create-post/page.tsx +++ b/src/app/create-post/page.tsx @@ -2,7 +2,14 @@ import Editor from "@/components/editor"; import { getCookie, hasCookie } from "@/helpers/cookie"; -import { Avatar, Button, Form, Input, Spacer } from "@nextui-org/react"; +import { + Avatar, + Button, + Checkbox, + Form, + Input, + Spacer, +} from "@nextui-org/react"; import { LoaderCircle } from "lucide-react"; import { redirect } from "next/navigation"; import { ReactNode, useEffect, useState } from "react"; @@ -10,6 +17,7 @@ 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"; export default function CreatePostPage() { const [title, setTitle] = useState(""); @@ -31,6 +39,8 @@ export default function CreatePostPage() { }[] >(); const { theme } = useTheme(); + const [user, setUser] = useState(); + const [sticky, setSticky] = useState(false); useEffect(() => { setMounted(true); @@ -46,7 +56,8 @@ export default function CreatePostPage() { } ); - const user = await response.json(); + const localuser = await response.json(); + setUser(localuser); const tagResponse = await fetch( process.env.NEXT_PUBLIC_MODE === "PROD" @@ -63,7 +74,7 @@ export default function CreatePostPage() { }[] = []; for (const tag of await tagResponse.json()) { - if (tag.modOnly && !user.mod) { + if (tag.modOnly && !localuser.mod) { continue; } newoptions.push({ @@ -90,7 +101,6 @@ export default function CreatePostPage() { } setOptions(newoptions); - setSelectedTags(newoptions.filter((tag) => tag.isFixed)); } }; load(); @@ -211,8 +221,14 @@ export default function CreatePostPage() { body: JSON.stringify({ title: title, content: sanitizedHtml, + sticky, username: getCookie("user"), - tags, + tags: [ + ...tags, + ...(options + ? options.filter((tag) => tag.isFixed).map((tag) => tag.id) + : []), + ], }), method: "POST", headers: { @@ -268,6 +284,15 @@ export default function CreatePostPage() { /> )} + {user && user.mod && ( +
+ + + Sticky + +
+ )} +
diff --git a/src/app/p/[slug]/page.tsx b/src/app/p/[slug]/page.tsx index 24b54f3..9d76271 100644 --- a/src/app/p/[slug]/page.tsx +++ b/src/app/p/[slug]/page.tsx @@ -28,6 +28,8 @@ import { Shield, ShieldAlert, ShieldX, + Star, + StarOff, Trash, X, } from "lucide-react"; @@ -295,6 +297,81 @@ export default function PostPage() { > Remove + {post.sticky ? ( + } + description="Unsticky post" + onPress={async () => { + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/post/sticky" + : "http://localhost:3005/api/v1/post/sticky", + { + body: JSON.stringify({ + postId: post.id, + sticky: false, + username: getCookie("user"), + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie( + "token" + )}`, + }, + credentials: "include", + } + ); + + if (response.ok) { + toast.success("Unsticked post"); + redirect("/"); + } else { + toast.error("Error while removing post"); + } + }} + > + Unsticky + + ) : ( + } + description="Sticky post" + onPress={async () => { + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/post/sticky" + : "http://localhost:3005/api/v1/post/sticky", + { + body: JSON.stringify({ + postId: post.id, + sticky: true, + username: getCookie("user"), + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie( + "token" + )}`, + }, + credentials: "include", + } + ); + + if (response.ok) { + toast.success("Unsticked post"); + redirect("/"); + } else { + toast.error("Error while removing post"); + } + }} + > + Sticky + + )} {user?.admin && !post.author.mod ? ( Remove + {post.sticky ? ( + } + description="Unsticky post" + onPress={async () => { + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/post/sticky" + : "http://localhost:3005/api/v1/post/sticky", + { + body: JSON.stringify({ + postId: post.id, + sticky: false, + username: getCookie("user"), + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie( + "token" + )}`, + }, + credentials: "include", + } + ); + + if (response.ok) { + toast.success("Unsticked post"); + window.location.reload(); + } else { + toast.error("Error while removing post"); + } + }} + > + Unsticky + + ) : ( + } + description="Sticky post" + onPress={async () => { + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/post/sticky" + : "http://localhost:3005/api/v1/post/sticky", + { + body: JSON.stringify({ + postId: post.id, + sticky: true, + username: getCookie("user"), + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie( + "token" + )}`, + }, + credentials: "include", + } + ); + + if (response.ok) { + toast.success("Stickied post"); + window.location.reload(); + } else { + toast.error("Error while removing post"); + } + }} + > + Sticky + + )} {user?.admin && !post.author.mod ? ( + +
+
+
+ {post.tags.filter((tag) => tag.name === "Changelog").length > + 0 ? ( + + ) : ( + + )} + +

{post.title}

+ +
+ +
+

By

+ + +

{post.author.name}

+ +

+ {formatDistance(new Date(post.createdAt), new Date(), { + addSuffix: true, + })} +

+
+
+
+
+ + ); +} diff --git a/src/components/posts/index.tsx b/src/components/posts/index.tsx index c7ac54a..043b53c 100644 --- a/src/components/posts/index.tsx +++ b/src/components/posts/index.tsx @@ -42,9 +42,11 @@ import { import { PostTime } from "@/types/PostTimes"; import { TagType } from "@/types/TagType"; import { useTheme } from "next-themes"; +import StickyPostCard from "./StickyPostCard"; export default function Posts() { const [posts, setPosts] = useState(); + const [stickyPosts, setStickyPosts] = useState(); const [sort, setSort] = useState("newest"); const [time, setTime] = useState("all"); const [style, setStyle] = useState("cozy"); @@ -144,6 +146,31 @@ export default function Posts() { }` ); setPosts(await postsResponse.json()); + + // Sticky posts + // Fetch posts with userSlug if user is available + const stickyPostsResponse = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? `https://d2jam.com/api/v1/posts?sort=${sort}&user=${ + userData.slug + }&time=${time}&tags=${ + tagRules + ? Object.entries(tagRules) + .map((key) => `${key}`) + .join("_") + : "" + }&sticky=true` + : `http://localhost:3005/api/v1/posts?sort=${sort}&user=${ + userData.slug + }&time=${time}&tags=${ + tagRules + ? Object.entries(tagRules) + .map((key) => `${key}`) + .join("_") + : "" + }&sticky=true` + ); + setStickyPosts(await stickyPostsResponse.json()); setLoading(false); } else { setUser(undefined); @@ -167,6 +194,26 @@ export default function Posts() { }` ); setPosts(await postsResponse.json()); + + // Fetch posts without userSlug if user is not available + const stickyPostsResponse = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? `https://d2jam.com/api/v1/posts?sort=${sort}&time=${time}&tags=${ + tagRules + ? Object.entries(tagRules) + .map((key, value) => `${key}-${value}`) + .join("_") + : "" + }&sticky=true` + : `http://localhost:3005/api/v1/posts?sort=${sort}&time=${time}&tags=${ + tagRules + ? Object.entries(tagRules) + .map((key, value) => `${key}-${value}`) + .join("_") + : "" + }&sticky=true` + ); + setStickyPosts(await stickyPostsResponse.json()); setLoading(false); } }; @@ -263,6 +310,27 @@ export default function Posts() { return (
+ {loading ? ( +
+ +
+ ) : ( +
+ {stickyPosts && stickyPosts.length > 0 ? ( + stickyPosts.map((post) => ( + + )) + ) : ( +

+ No posts match your filters +

+ )} +
+ )} +
diff --git a/src/types/PostType.ts b/src/types/PostType.ts index 458855f..111a8bc 100644 --- a/src/types/PostType.ts +++ b/src/types/PostType.ts @@ -5,6 +5,7 @@ export interface PostType { id: number; slug: string; title: string; + sticky: boolean; content: string; author: UserType; createdAt: Date;