mirror of
https://github.com/Ategon/Jamjar.git
synced 2025-02-12 06:16:21 +00:00
Add stickied posts
This commit is contained in:
parent
94a3d31e8d
commit
42c730e073
6 changed files with 307 additions and 5 deletions
|
@ -2,7 +2,14 @@
|
||||||
|
|
||||||
import Editor from "@/components/editor";
|
import Editor from "@/components/editor";
|
||||||
import { getCookie, hasCookie } from "@/helpers/cookie";
|
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 { LoaderCircle } from "lucide-react";
|
||||||
import { redirect } from "next/navigation";
|
import { redirect } from "next/navigation";
|
||||||
import { ReactNode, useEffect, useState } from "react";
|
import { ReactNode, useEffect, useState } from "react";
|
||||||
|
@ -10,6 +17,7 @@ import { toast } from "react-toastify";
|
||||||
import sanitizeHtml from "sanitize-html";
|
import sanitizeHtml from "sanitize-html";
|
||||||
import Select, { MultiValue, StylesConfig } from "react-select";
|
import Select, { MultiValue, StylesConfig } from "react-select";
|
||||||
import { useTheme } from "next-themes";
|
import { useTheme } from "next-themes";
|
||||||
|
import { UserType } from "@/types/UserType";
|
||||||
|
|
||||||
export default function CreatePostPage() {
|
export default function CreatePostPage() {
|
||||||
const [title, setTitle] = useState("");
|
const [title, setTitle] = useState("");
|
||||||
|
@ -31,6 +39,8 @@ export default function CreatePostPage() {
|
||||||
}[]
|
}[]
|
||||||
>();
|
>();
|
||||||
const { theme } = useTheme();
|
const { theme } = useTheme();
|
||||||
|
const [user, setUser] = useState<UserType>();
|
||||||
|
const [sticky, setSticky] = useState(false);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setMounted(true);
|
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(
|
const tagResponse = await fetch(
|
||||||
process.env.NEXT_PUBLIC_MODE === "PROD"
|
process.env.NEXT_PUBLIC_MODE === "PROD"
|
||||||
|
@ -63,7 +74,7 @@ export default function CreatePostPage() {
|
||||||
}[] = [];
|
}[] = [];
|
||||||
|
|
||||||
for (const tag of await tagResponse.json()) {
|
for (const tag of await tagResponse.json()) {
|
||||||
if (tag.modOnly && !user.mod) {
|
if (tag.modOnly && !localuser.mod) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
newoptions.push({
|
newoptions.push({
|
||||||
|
@ -90,7 +101,6 @@ export default function CreatePostPage() {
|
||||||
}
|
}
|
||||||
|
|
||||||
setOptions(newoptions);
|
setOptions(newoptions);
|
||||||
setSelectedTags(newoptions.filter((tag) => tag.isFixed));
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
load();
|
load();
|
||||||
|
@ -211,8 +221,14 @@ export default function CreatePostPage() {
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
title: title,
|
title: title,
|
||||||
content: sanitizedHtml,
|
content: sanitizedHtml,
|
||||||
|
sticky,
|
||||||
username: getCookie("user"),
|
username: getCookie("user"),
|
||||||
tags,
|
tags: [
|
||||||
|
...tags,
|
||||||
|
...(options
|
||||||
|
? options.filter((tag) => tag.isFixed).map((tag) => tag.id)
|
||||||
|
: []),
|
||||||
|
],
|
||||||
}),
|
}),
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
|
@ -268,6 +284,15 @@ export default function CreatePostPage() {
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{user && user.mod && (
|
||||||
|
<div>
|
||||||
|
<Spacer />
|
||||||
|
<Checkbox isSelected={sticky} onValueChange={setSticky}>
|
||||||
|
Sticky
|
||||||
|
</Checkbox>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
<Spacer />
|
<Spacer />
|
||||||
|
|
||||||
<div className="flex gap-2">
|
<div className="flex gap-2">
|
||||||
|
|
|
@ -28,6 +28,8 @@ import {
|
||||||
Shield,
|
Shield,
|
||||||
ShieldAlert,
|
ShieldAlert,
|
||||||
ShieldX,
|
ShieldX,
|
||||||
|
Star,
|
||||||
|
StarOff,
|
||||||
Trash,
|
Trash,
|
||||||
X,
|
X,
|
||||||
} from "lucide-react";
|
} from "lucide-react";
|
||||||
|
@ -295,6 +297,81 @@ export default function PostPage() {
|
||||||
>
|
>
|
||||||
Remove
|
Remove
|
||||||
</DropdownItem>
|
</DropdownItem>
|
||||||
|
{post.sticky ? (
|
||||||
|
<DropdownItem
|
||||||
|
key="unsticky"
|
||||||
|
startContent={<StarOff />}
|
||||||
|
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
|
||||||
|
</DropdownItem>
|
||||||
|
) : (
|
||||||
|
<DropdownItem
|
||||||
|
key="sticky"
|
||||||
|
startContent={<Star />}
|
||||||
|
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
|
||||||
|
</DropdownItem>
|
||||||
|
)}
|
||||||
{user?.admin && !post.author.mod ? (
|
{user?.admin && !post.author.mod ? (
|
||||||
<DropdownItem
|
<DropdownItem
|
||||||
key="promote-mod"
|
key="promote-mod"
|
||||||
|
|
|
@ -25,6 +25,8 @@ import {
|
||||||
Shield,
|
Shield,
|
||||||
ShieldAlert,
|
ShieldAlert,
|
||||||
ShieldX,
|
ShieldX,
|
||||||
|
Star,
|
||||||
|
StarOff,
|
||||||
Trash,
|
Trash,
|
||||||
X,
|
X,
|
||||||
} from "lucide-react";
|
} from "lucide-react";
|
||||||
|
@ -296,6 +298,81 @@ export default function PostCard({
|
||||||
>
|
>
|
||||||
Remove
|
Remove
|
||||||
</DropdownItem>
|
</DropdownItem>
|
||||||
|
{post.sticky ? (
|
||||||
|
<DropdownItem
|
||||||
|
key="unsticky"
|
||||||
|
startContent={<StarOff />}
|
||||||
|
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
|
||||||
|
</DropdownItem>
|
||||||
|
) : (
|
||||||
|
<DropdownItem
|
||||||
|
key="sticky"
|
||||||
|
startContent={<Star />}
|
||||||
|
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
|
||||||
|
</DropdownItem>
|
||||||
|
)}
|
||||||
{user?.admin && !post.author.mod ? (
|
{user?.admin && !post.author.mod ? (
|
||||||
<DropdownItem
|
<DropdownItem
|
||||||
key="promote-mod"
|
key="promote-mod"
|
||||||
|
|
54
src/components/posts/StickyPostCard.tsx
Normal file
54
src/components/posts/StickyPostCard.tsx
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
"use client";
|
||||||
|
|
||||||
|
import { Avatar, Card, CardBody } from "@nextui-org/react";
|
||||||
|
import { formatDistance } from "date-fns";
|
||||||
|
import Link from "next/link";
|
||||||
|
import { PostType } from "@/types/PostType";
|
||||||
|
import { Megaphone, NotebookText } from "lucide-react";
|
||||||
|
|
||||||
|
export default function StickyPostCard({ post }: { post: PostType }) {
|
||||||
|
return (
|
||||||
|
<Card className="bg-opacity-60 !duration-250 !ease-linear !transition-all flex">
|
||||||
|
<CardBody className="p-5">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<div className="flex items-center gap-4">
|
||||||
|
<div className="flex gap-4 items-center text-blue-600 dark:text-blue-400 transition-all duration-250 ease-linear">
|
||||||
|
{post.tags.filter((tag) => tag.name === "Changelog").length >
|
||||||
|
0 ? (
|
||||||
|
<NotebookText />
|
||||||
|
) : (
|
||||||
|
<Megaphone />
|
||||||
|
)}
|
||||||
|
<Link href={`/p/${post.slug}`}>
|
||||||
|
<p>{post.title}</p>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex items-center gap-3 text-xs text-default-500 pt-1">
|
||||||
|
<p>By</p>
|
||||||
|
<Link
|
||||||
|
href={`/u/${post.author.slug}`}
|
||||||
|
className="flex items-center gap-2"
|
||||||
|
>
|
||||||
|
<Avatar
|
||||||
|
size="sm"
|
||||||
|
className="w-6 h-6"
|
||||||
|
src={post.author.profilePicture}
|
||||||
|
classNames={{
|
||||||
|
base: "bg-transparent",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<p>{post.author.name}</p>
|
||||||
|
</Link>
|
||||||
|
<p>
|
||||||
|
{formatDistance(new Date(post.createdAt), new Date(), {
|
||||||
|
addSuffix: true,
|
||||||
|
})}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</CardBody>
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
|
@ -42,9 +42,11 @@ import {
|
||||||
import { PostTime } from "@/types/PostTimes";
|
import { PostTime } from "@/types/PostTimes";
|
||||||
import { TagType } from "@/types/TagType";
|
import { TagType } from "@/types/TagType";
|
||||||
import { useTheme } from "next-themes";
|
import { useTheme } from "next-themes";
|
||||||
|
import StickyPostCard from "./StickyPostCard";
|
||||||
|
|
||||||
export default function Posts() {
|
export default function Posts() {
|
||||||
const [posts, setPosts] = useState<PostType[]>();
|
const [posts, setPosts] = useState<PostType[]>();
|
||||||
|
const [stickyPosts, setStickyPosts] = useState<PostType[]>();
|
||||||
const [sort, setSort] = useState<PostSort>("newest");
|
const [sort, setSort] = useState<PostSort>("newest");
|
||||||
const [time, setTime] = useState<PostTime>("all");
|
const [time, setTime] = useState<PostTime>("all");
|
||||||
const [style, setStyle] = useState<PostStyle>("cozy");
|
const [style, setStyle] = useState<PostStyle>("cozy");
|
||||||
|
@ -144,6 +146,31 @@ export default function Posts() {
|
||||||
}`
|
}`
|
||||||
);
|
);
|
||||||
setPosts(await postsResponse.json());
|
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);
|
setLoading(false);
|
||||||
} else {
|
} else {
|
||||||
setUser(undefined);
|
setUser(undefined);
|
||||||
|
@ -167,6 +194,26 @@ export default function Posts() {
|
||||||
}`
|
}`
|
||||||
);
|
);
|
||||||
setPosts(await postsResponse.json());
|
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);
|
setLoading(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -263,6 +310,27 @@ export default function Posts() {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
|
{loading ? (
|
||||||
|
<div className="flex justify-center p-6">
|
||||||
|
<LoaderCircle
|
||||||
|
className="animate-spin text-[#333] dark:text-[#999]"
|
||||||
|
size={24}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="flex flex-col gap-3 p-4">
|
||||||
|
{stickyPosts && stickyPosts.length > 0 ? (
|
||||||
|
stickyPosts.map((post) => (
|
||||||
|
<StickyPostCard key={post.id} post={post} />
|
||||||
|
))
|
||||||
|
) : (
|
||||||
|
<p className="text-center text-[#333] dark:text-white transition-color duration-250 ease-linear">
|
||||||
|
No posts match your filters
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
<div className="flex justify-between p-4 pb-0">
|
<div className="flex justify-between p-4 pb-0">
|
||||||
<div className="flex gap-2">
|
<div className="flex gap-2">
|
||||||
<Dropdown backdrop="opaque">
|
<Dropdown backdrop="opaque">
|
||||||
|
|
|
@ -5,6 +5,7 @@ export interface PostType {
|
||||||
id: number;
|
id: number;
|
||||||
slug: string;
|
slug: string;
|
||||||
title: string;
|
title: string;
|
||||||
|
sticky: boolean;
|
||||||
content: string;
|
content: string;
|
||||||
author: UserType;
|
author: UserType;
|
||||||
createdAt: Date;
|
createdAt: Date;
|
||||||
|
|
Loading…
Reference in a new issue