diff --git a/src/app/p/[slug]/page.tsx b/src/app/p/[slug]/page.tsx index cb8f110..fff860e 100644 --- a/src/app/p/[slug]/page.tsx +++ b/src/app/p/[slug]/page.tsx @@ -1,7 +1,7 @@ "use client"; import LikeButton from "@/components/posts/LikeButton"; -import { getCookie } from "@/helpers/cookie"; +import { getCookie, hasCookie } from "@/helpers/cookie"; import { PostType } from "@/types/PostType"; import { TagType } from "@/types/TagType"; import { UserType } from "@/types/UserType"; @@ -36,6 +36,9 @@ import { import { redirect, useParams } from "next/navigation"; import { useEffect, useState } from "react"; import { toast } from "react-toastify"; +import Editor from "@/components/editor"; +import sanitizeHtml from "sanitize-html"; +import CommentCard from "@/components/posts/CommentCard"; export default function PostPage() { const [post, setPost] = useState<PostType>(); @@ -43,6 +46,8 @@ export default function PostPage() { const [reduceMotion, setReduceMotion] = useState<boolean>(false); const [user, setUser] = useState<UserType>(); const [loading, setLoading] = useState<boolean>(true); + const [content, setContent] = useState(""); + const [waitingPost, setWaitingPost] = useState(false); useEffect(() => { const mediaQuery = window.matchMedia("(prefers-reduced-motion: reduce)"); @@ -110,210 +115,136 @@ export default function PostPage() { /> </div> ) : ( - <Card className="bg-opacity-60 !duration-250 !ease-linear !transition-all"> - <CardBody className="p-5"> - <div> - {post && ( - <div> - <Link href={`/p/${post.slug}`}> - <p className="text-2xl">{post.title}</p> - </Link> - - <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> + <> + <Card className="bg-opacity-60 !duration-250 !ease-linear !transition-all"> + <CardBody className="p-5"> + <div> + {post && ( + <div> + <Link href={`/p/${post.slug}`}> + <p className="text-2xl">{post.title}</p> </Link> - <p> - {formatDistance(new Date(post.createdAt), new Date(), { - addSuffix: true, - })} - </p> - </div> - <Spacer y={4} /> - - <div - className="prose dark:prose-invert !duration-250 !ease-linear !transition-all" - dangerouslySetInnerHTML={{ __html: post.content }} - /> - - <Spacer y={4} /> - - {post.tags.filter((tag) => tag.name != "D2Jam").length > 0 ? ( - <div className="flex gap-1"> - {post.tags - .filter((tag) => tag.name != "D2Jam") - .map((tag: TagType) => ( - <Link - href="/" - key={tag.id} - className={`transition-all transform duration-500 ease-in-out ${ - !reduceMotion ? "hover:scale-110" : "" - }`} - > - <Chip - radius="sm" - size="sm" - className="!duration-250 !ease-linear !transition-all" - variant="faded" - avatar={ - tag.icon && ( - <Avatar - src={tag.icon} - classNames={{ base: "bg-transparent" }} - /> - ) - } - > - {tag.name} - </Chip> - </Link> - ))} + <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> - ) : ( - <></> - )} - {post.tags.length > 0 && <Spacer y={4} />} + <Spacer y={4} /> - <div className="flex gap-3"> - <LikeButton post={post} /> - <Button - size="sm" - variant="bordered" - onPress={() => { - toast.warning("Comment functionality coming soon"); - }} - > - <MessageCircle size={16} /> {0} - </Button> - <Dropdown backdrop="opaque"> - <DropdownTrigger> - <Button size="sm" variant="bordered" isIconOnly> - <MoreVertical size={16} /> + <div + className="prose dark:prose-invert !duration-250 !ease-linear !transition-all" + dangerouslySetInnerHTML={{ __html: post.content }} + /> + + <Spacer y={4} /> + + {post.tags.filter((tag) => tag.name != "D2Jam").length > + 0 ? ( + <div className="flex gap-1"> + {post.tags + .filter((tag) => tag.name != "D2Jam") + .map((tag: TagType) => ( + <Link + href="/" + key={tag.id} + className={`transition-all transform duration-500 ease-in-out ${ + !reduceMotion ? "hover:scale-110" : "" + }`} + > + <Chip + radius="sm" + size="sm" + className="!duration-250 !ease-linear !transition-all" + variant="faded" + avatar={ + tag.icon && ( + <Avatar + src={tag.icon} + classNames={{ base: "bg-transparent" }} + /> + ) + } + > + {tag.name} + </Chip> + </Link> + ))} + </div> + ) : ( + <></> + )} + + {post.tags.length > 0 && <Spacer y={4} />} + + <div className="flex gap-3"> + <LikeButton + likes={post.likes.length} + liked={post.hasLiked} + parentId={post.id} + /> + <Link href="#create-comment"> + <Button size="sm" variant="bordered"> + <MessageCircle size={16} /> {post.comments.length} </Button> - </DropdownTrigger> - <DropdownMenu className="text-[#333] dark:text-white"> - <DropdownSection - showDivider={user?.mod} - title="Actions" - > - <DropdownItem - key="report" - startContent={<Flag />} - description="Report this post to moderators to handle" - onPress={() => { - toast.warning("Report functionality coming soon"); - }} + </Link> + <Dropdown backdrop="opaque"> + <DropdownTrigger> + <Button size="sm" variant="bordered" isIconOnly> + <MoreVertical size={16} /> + </Button> + </DropdownTrigger> + <DropdownMenu className="text-[#333] dark:text-white"> + <DropdownSection + showDivider={user?.mod} + title="Actions" > - Create Report - </DropdownItem> - {user?.slug == post.author.slug ? ( <DropdownItem - key="delete" - startContent={<Trash />} - description="Delete your post" - onPress={async () => { - 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({ - postId: post.id, - username: getCookie("user"), - }), - method: "DELETE", - headers: { - "Content-Type": "application/json", - authorization: `Bearer ${getCookie( - "token" - )}`, - }, - credentials: "include", - } + key="report" + startContent={<Flag />} + description="Report this post to moderators to handle" + onPress={() => { + toast.warning( + "Report functionality coming soon" ); - - if (response.ok) { - toast.success("Deleted post"); - redirect("/"); - } else { - toast.error("Error while deleting post"); - } }} > - Delete + Create Report </DropdownItem> - ) : ( - <></> - )} - </DropdownSection> - {user?.mod ? ( - <DropdownSection title="Mod Zone"> - <DropdownItem - key="remove" - startContent={<X />} - description="Remove this post" - onPress={async () => { - 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({ - postId: post.id, - username: getCookie("user"), - }), - method: "DELETE", - headers: { - "Content-Type": "application/json", - authorization: `Bearer ${getCookie( - "token" - )}`, - }, - credentials: "include", - } - ); - - if (response.ok) { - toast.success("Removed post"); - redirect("/"); - } else { - toast.error("Error while removing post"); - } - }} - > - Remove - </DropdownItem> - {post.sticky ? ( + {user?.slug == post.author.slug ? ( <DropdownItem - key="unsticky" - startContent={<StarOff />} - description="Unsticky post" + key="delete" + startContent={<Trash />} + description="Delete your 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", + ? "https://d2jam.com/api/v1/post" + : "http://localhost:3005/api/v1/post", { body: JSON.stringify({ postId: post.id, - sticky: false, username: getCookie("user"), }), - method: "POST", + method: "DELETE", headers: { "Content-Type": "application/json", authorization: `Bearer ${getCookie( @@ -325,234 +256,383 @@ export default function PostPage() { ); if (response.ok) { - toast.success("Unsticked post"); + toast.success("Deleted post"); redirect("/"); } else { - toast.error("Error while removing post"); + toast.error("Error while deleting 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 ? ( - <DropdownItem - key="promote-mod" - startContent={<Shield />} - description="Promote user to Mod" - onPress={async () => { - const response = await fetch( - process.env.NEXT_PUBLIC_MODE === "PROD" - ? "https://d2jam.com/api/v1/mod" - : "http://localhost:3005/api/v1/mod", - { - body: JSON.stringify({ - targetname: post.author.slug, - mod: true, - username: getCookie("user"), - }), - method: "POST", - headers: { - "Content-Type": "application/json", - authorization: `Bearer ${getCookie( - "token" - )}`, - }, - credentials: "include", - } - ); - - if (response.ok) { - toast.success("Promoted User to Mod"); - window.location.reload(); - } else { - toast.error( - "Error while promoting user to Mod" - ); - } - }} - > - Appoint as mod - </DropdownItem> - ) : ( - <></> - )} - {user?.admin && - post.author.mod && - !post.author.admin ? ( - <DropdownItem - key="demote-mod" - startContent={<ShieldX />} - description="Demote user from Mod" - onPress={async () => { - const response = await fetch( - process.env.NEXT_PUBLIC_MODE === "PROD" - ? "https://d2jam.com/api/v1/mod" - : "http://localhost:3005/api/v1/mod", - { - body: JSON.stringify({ - targetname: post.author.slug, - username: getCookie("user"), - }), - method: "POST", - headers: { - "Content-Type": "application/json", - authorization: `Bearer ${getCookie( - "token" - )}`, - }, - credentials: "include", - } - ); - - if (response.ok) { - toast.success("Demoted User"); - window.location.reload(); - } else { - toast.error("Error while demoting user"); - } - }} - > - Remove as mod - </DropdownItem> - ) : ( - <></> - )} - {user?.admin && !post.author.admin ? ( - <DropdownItem - key="promote-admin" - startContent={<ShieldAlert />} - description="Promote user to Admin" - onPress={async () => { - const response = await fetch( - process.env.NEXT_PUBLIC_MODE === "PROD" - ? "https://d2jam.com/api/v1/mod" - : "http://localhost:3005/api/v1/mod", - { - body: JSON.stringify({ - targetname: post.author.slug, - admin: true, - username: getCookie("user"), - }), - method: "POST", - headers: { - "Content-Type": "application/json", - authorization: `Bearer ${getCookie( - "token" - )}`, - }, - credentials: "include", - } - ); - - if (response.ok) { - toast.success("Promoted User to Admin"); - window.location.reload(); - } else { - toast.error( - "Error while promoting user to Admin" - ); - } - }} - > - Appoint as admin - </DropdownItem> - ) : ( - <></> - )} - {user?.admin && - post.author.admin && - post.author.id !== user.id ? ( - <DropdownItem - key="demote-admin" - startContent={<ShieldX />} - description="Demote user to mod" - onPress={async () => { - const response = await fetch( - process.env.NEXT_PUBLIC_MODE === "PROD" - ? "https://d2jam.com/api/v1/mod" - : "http://localhost:3005/api/v1/mod", - { - body: JSON.stringify({ - targetname: post.author.slug, - mod: true, - username: getCookie("user"), - }), - method: "POST", - headers: { - "Content-Type": "application/json", - authorization: `Bearer ${getCookie( - "token" - )}`, - }, - credentials: "include", - } - ); - - if (response.ok) { - toast.success("Demoted User to Mod"); - window.location.reload(); - } else { - toast.error( - "Error while demoting user to mod" - ); - } - }} - > - Remove as admin + Delete </DropdownItem> ) : ( <></> )} </DropdownSection> - ) : ( - <></> - )} - </DropdownMenu> - </Dropdown> + {user?.mod ? ( + <DropdownSection title="Mod Zone"> + <DropdownItem + key="remove" + startContent={<X />} + description="Remove this post" + onPress={async () => { + 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({ + postId: post.id, + username: getCookie("user"), + }), + method: "DELETE", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie( + "token" + )}`, + }, + credentials: "include", + } + ); + + if (response.ok) { + toast.success("Removed post"); + redirect("/"); + } else { + toast.error("Error while removing post"); + } + }} + > + Remove + </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 ? ( + <DropdownItem + key="promote-mod" + startContent={<Shield />} + description="Promote user to Mod" + onPress={async () => { + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/mod" + : "http://localhost:3005/api/v1/mod", + { + body: JSON.stringify({ + targetname: post.author.slug, + mod: true, + username: getCookie("user"), + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie( + "token" + )}`, + }, + credentials: "include", + } + ); + + if (response.ok) { + toast.success("Promoted User to Mod"); + window.location.reload(); + } else { + toast.error( + "Error while promoting user to Mod" + ); + } + }} + > + Appoint as mod + </DropdownItem> + ) : ( + <></> + )} + {user?.admin && + post.author.mod && + !post.author.admin ? ( + <DropdownItem + key="demote-mod" + startContent={<ShieldX />} + description="Demote user from Mod" + onPress={async () => { + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/mod" + : "http://localhost:3005/api/v1/mod", + { + body: JSON.stringify({ + targetname: post.author.slug, + username: getCookie("user"), + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie( + "token" + )}`, + }, + credentials: "include", + } + ); + + if (response.ok) { + toast.success("Demoted User"); + window.location.reload(); + } else { + toast.error("Error while demoting user"); + } + }} + > + Remove as mod + </DropdownItem> + ) : ( + <></> + )} + {user?.admin && !post.author.admin ? ( + <DropdownItem + key="promote-admin" + startContent={<ShieldAlert />} + description="Promote user to Admin" + onPress={async () => { + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/mod" + : "http://localhost:3005/api/v1/mod", + { + body: JSON.stringify({ + targetname: post.author.slug, + admin: true, + username: getCookie("user"), + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie( + "token" + )}`, + }, + credentials: "include", + } + ); + + if (response.ok) { + toast.success("Promoted User to Admin"); + window.location.reload(); + } else { + toast.error( + "Error while promoting user to Admin" + ); + } + }} + > + Appoint as admin + </DropdownItem> + ) : ( + <></> + )} + {user?.admin && + post.author.admin && + post.author.id !== user.id ? ( + <DropdownItem + key="demote-admin" + startContent={<ShieldX />} + description="Demote user to mod" + onPress={async () => { + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/mod" + : "http://localhost:3005/api/v1/mod", + { + body: JSON.stringify({ + targetname: post.author.slug, + mod: true, + username: getCookie("user"), + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie( + "token" + )}`, + }, + credentials: "include", + } + ); + + if (response.ok) { + toast.success("Demoted User to Mod"); + window.location.reload(); + } else { + toast.error( + "Error while demoting user to mod" + ); + } + }} + > + Remove as admin + </DropdownItem> + ) : ( + <></> + )} + </DropdownSection> + ) : ( + <></> + )} + </DropdownMenu> + </Dropdown> + </div> </div> - </div> - )} - </div> - </CardBody> - </Card> + )} + </div> + </CardBody> + </Card> + <div id="create-comment" /> + <Spacer y={10} /> + <Editor content={content} setContent={setContent} /> + <Spacer /> + <Button + color="primary" + onPress={async () => { + if (!content) { + toast.error("Please enter valid content"); + return; + } + + if (!hasCookie("token")) { + toast.error("You are not logged in"); + return; + } + + const sanitizedHtml = sanitizeHtml(content); + setWaitingPost(true); + + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/comment" + : "http://localhost:3005/api/v1/comment", + { + body: JSON.stringify({ + content: sanitizedHtml, + username: getCookie("user"), + postId: post?.id, + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie("token")}`, + }, + credentials: "include", + } + ); + + if (response.status == 401) { + toast.error("Invalid User"); + setWaitingPost(false); + return; + } + + if (response.ok) { + toast.success("Successfully created comment"); + setWaitingPost(false); + window.location.reload(); + } else { + toast.error("An error occured"); + setWaitingPost(false); + } + }} + > + {waitingPost ? ( + <LoaderCircle className="animate-spin" size={16} /> + ) : ( + <p>Create Comment</p> + )} + </Button> + <Spacer y={10} /> + + <div className="flex flex-col gap-3"> + {post?.comments.map((comment) => ( + <div key={comment.id}> + <CommentCard comment={comment} /> + </div> + ))} + </div> + </> )} </> ); diff --git a/src/components/posts/CommentCard.tsx b/src/components/posts/CommentCard.tsx new file mode 100644 index 0000000..ff6b469 --- /dev/null +++ b/src/components/posts/CommentCard.tsx @@ -0,0 +1,162 @@ +import { CommentType } from "@/types/CommentType"; +import { Avatar, Button, Card, CardBody, Spacer } from "@nextui-org/react"; +import { formatDistance } from "date-fns"; +import { LoaderCircle, Reply } from "lucide-react"; +import Link from "next/link"; +import { useState } from "react"; +import Editor from "../editor"; +import { toast } from "react-toastify"; +import { getCookie, hasCookie } from "@/helpers/cookie"; +import sanitizeHtml from "sanitize-html"; +import LikeButton from "./LikeButton"; + +export default function CommentCard({ comment }: { comment: CommentType }) { + const [creatingReply, setCreatingReply] = useState<boolean>(false); + const [content, setContent] = useState(""); + const [waitingPost, setWaitingPost] = useState(false); + + return ( + <Card className="bg-opacity-60 !duration-250 !ease-linear !transition-all"> + <CardBody className="p-5"> + <div> + <div className="flex items-center gap-3 text-xs text-default-500 pt-1"> + <p>By</p> + <Link + href={`/u/${comment.author.slug}`} + className="flex items-center gap-2" + > + <Avatar + size="sm" + className="w-6 h-6" + src={comment.author.profilePicture} + classNames={{ + base: "bg-transparent", + }} + /> + <p>{comment.author.name}</p> + </Link> + <p> + {formatDistance(new Date(comment.createdAt), new Date(), { + addSuffix: true, + })} + </p> + </div> + + <Spacer y={4} /> + + <div + className="prose dark:prose-invert !duration-250 !ease-linear !transition-all" + dangerouslySetInnerHTML={{ __html: comment.content }} + /> + + <Spacer y={4} /> + + <div className="flex gap-3"> + <LikeButton + likes={comment.likes.length} + liked={comment.hasLiked} + parentId={comment.id} + isComment + /> + + <Button + size="sm" + variant="bordered" + onPress={() => { + setCreatingReply(!creatingReply); + }} + > + <Reply size={16} /> + </Button> + </div> + + <Spacer y={4} /> + + {creatingReply && ( + <> + <Editor content={content} setContent={setContent} /> + <div id="create-comment" /> + <Spacer /> + <Button + color="primary" + onPress={async () => { + if (!content) { + toast.error("Please enter valid content"); + return; + } + + if (!hasCookie("token")) { + toast.error("You are not logged in"); + return; + } + + const sanitizedHtml = sanitizeHtml(content); + setWaitingPost(true); + + const response = await fetch( + process.env.NEXT_PUBLIC_MODE === "PROD" + ? "https://d2jam.com/api/v1/comment" + : "http://localhost:3005/api/v1/comment", + { + body: JSON.stringify({ + content: sanitizedHtml, + username: getCookie("user"), + commentId: comment?.id, + }), + method: "POST", + headers: { + "Content-Type": "application/json", + authorization: `Bearer ${getCookie("token")}`, + }, + credentials: "include", + } + ); + + if (response.status == 401) { + toast.error("Invalid User"); + setWaitingPost(false); + return; + } + + if (response.ok) { + toast.success("Successfully created comment"); + setWaitingPost(false); + window.location.reload(); + } else { + toast.error("An error occured"); + setWaitingPost(false); + } + }} + > + {waitingPost ? ( + <LoaderCircle className="animate-spin" size={16} /> + ) : ( + <p>Create Reply</p> + )} + </Button> + <Spacer y={4} /> + </> + )} + + {comment.children.length > 0 && + (comment.children[0].author ? ( + <div className="flex flex-col gap-3"> + {comment.children.map((comment) => ( + <CommentCard key={comment.id} comment={comment} /> + ))} + </div> + ) : ( + <Button + variant="bordered" + onPress={() => { + toast.warning("Feature coming soon"); + }} + > + Load replies + </Button> + ))} + </div> + </CardBody> + </Card> + ); +} diff --git a/src/components/posts/LikeButton.tsx b/src/components/posts/LikeButton.tsx index be88e03..6106fa9 100644 --- a/src/components/posts/LikeButton.tsx +++ b/src/components/posts/LikeButton.tsx @@ -1,7 +1,6 @@ "use client"; import { Button } from "@nextui-org/react"; -import { PostType } from "@/types/PostType"; import { Heart } from "lucide-react"; import { toast } from "react-toastify"; import { getCookie } from "@/helpers/cookie"; @@ -9,11 +8,22 @@ import { redirect } from "next/navigation"; import { useState, useEffect } from "react"; import { useTheme } from "next-themes"; -export default function LikeButton({ post }: { post: PostType }) { - const [likes, setLikes] = useState<number>(post.likes.length); - const [liked, setLiked] = useState<boolean>(false); +export default function LikeButton({ + likes, + liked, + parentId, + isComment = false, +}: { + likes: number; + liked: boolean; + parentId: number; + isComment?: boolean; +}) { const { theme } = useTheme(); const [reduceMotion, setReduceMotion] = useState<boolean>(false); + const [likeEffect, setLikeEffect] = useState<boolean>(false); + const [updatedLikes, setUpdatedLikes] = useState<number>(likes); + const [updatedLiked, setUpdatedLiked] = useState<boolean>(liked); useEffect(() => { const mediaQuery = window.matchMedia("(prefers-reduced-motion: reduce)"); @@ -34,8 +44,8 @@ export default function LikeButton({ post }: { post: PostType }) { size="sm" variant="bordered" style={{ - color: post.hasLiked ? (theme == "dark" ? "#5ed4f7" : "#05b7eb") : "", - borderColor: post.hasLiked + color: updatedLiked ? (theme == "dark" ? "#5ed4f7" : "#05b7eb") : "", + borderColor: updatedLiked ? theme == "dark" ? "#5ed4f744" : "#05b7eb44" @@ -59,27 +69,28 @@ export default function LikeButton({ post }: { post: PostType }) { credentials: "include", body: JSON.stringify({ username: getCookie("user"), - postId: post.id, + postId: !isComment ? parentId : 0, + commentId: isComment ? parentId : 0, }), } ); - post.hasLiked = !post.hasLiked; - - if (post.hasLiked) { - setLiked(true); - setTimeout(() => setLiked(false), 1000); - setLikes(likes + 1); + if (!updatedLiked) { + setLikeEffect(true); + setTimeout(() => setLikeEffect(false), 1000); + setUpdatedLikes(updatedLikes + 1); } else { - setLiked(false); - setLikes(likes - 1); + setLikeEffect(false); + setUpdatedLikes(updatedLikes - 1); } + setUpdatedLiked(!updatedLiked); + if (!response.ok) { if (response.status == 401) { redirect("/login"); } else { - post.hasLiked = !post.hasLiked; + setUpdatedLiked(!updatedLiked); toast.error("An error occurred"); return; } @@ -91,7 +102,7 @@ export default function LikeButton({ post }: { post: PostType }) { <Heart size={16} className={ - liked && !reduceMotion + likeEffect && !reduceMotion ? "animate-ping absolute top-0 left-0" : "absolute top-0 left-0" } @@ -102,7 +113,7 @@ export default function LikeButton({ post }: { post: PostType }) { zIndex: "10", }} /> - <p>{likes}</p> + <p>{updatedLikes}</p> </div> </Button> ); diff --git a/src/components/posts/PostCard.tsx b/src/components/posts/PostCard.tsx index ef7ba1f..e439bdc 100644 --- a/src/components/posts/PostCard.tsx +++ b/src/components/posts/PostCard.tsx @@ -197,16 +197,16 @@ export default function PostCard({ {post.tags.length > 0 && <Spacer y={4} />} <div className="flex gap-3"> - <LikeButton post={post} /> - <Button - size="sm" - variant="bordered" - onPress={() => { - toast.warning("Comment functionality coming soon"); - }} - > - <MessageCircle size={16} /> {0} - </Button> + <LikeButton + likes={post.likes.length} + liked={post.hasLiked} + parentId={post.id} + /> + <Link href={`/p/${post.slug}#create-comment`}> + <Button size="sm" variant="bordered"> + <MessageCircle size={16} /> {post.comments.length} + </Button> + </Link> <Dropdown backdrop="opaque"> <DropdownTrigger> <Button size="sm" variant="bordered" isIconOnly> diff --git a/src/types/CommentType.ts b/src/types/CommentType.ts new file mode 100644 index 0000000..6eb8d89 --- /dev/null +++ b/src/types/CommentType.ts @@ -0,0 +1,11 @@ +import { UserType } from "./UserType"; + +export interface CommentType { + id: number; + content: string; + children: CommentType[]; + author: UserType; + createdAt: Date; + likes: []; + hasLiked: boolean; +} diff --git a/src/types/PostType.ts b/src/types/PostType.ts index 111a8bc..acff8b5 100644 --- a/src/types/PostType.ts +++ b/src/types/PostType.ts @@ -1,3 +1,4 @@ +import { CommentType } from "./CommentType"; import { TagType } from "./TagType"; import { UserType } from "./UserType"; @@ -9,6 +10,7 @@ export interface PostType { content: string; author: UserType; createdAt: Date; + comments: CommentType[]; tags: TagType[]; likes: []; hasLiked: boolean;