"use client"; import { ReactNode, useEffect, useState } from "react"; import PostCard from "./PostCard"; import { PostType } from "@/types/PostType"; import { Avatar, Button, Chip, Dropdown, DropdownItem, DropdownMenu, DropdownTrigger, Popover, PopoverContent, PopoverTrigger, } from "@nextui-org/react"; import { PostSort } from "@/types/PostSort"; import { PostStyle } from "@/types/PostStyle"; import { getCookie } from "@/helpers/cookie"; import { UserType } from "@/types/UserType"; import { Calendar, Calendar1, CalendarArrowDown, CalendarCog, CalendarDays, CalendarFold, CalendarRange, Check, Clock1, Clock2, Clock3, Clock4, ClockArrowDown, ClockArrowUp, LoaderCircle, Sparkles, Trophy, X, } from "lucide-react"; import { PostTime } from "@/types/PostTimes"; import { TagType } from "@/types/TagType"; import { useTheme } from "next-themes"; import StickyPostCard from "./StickyPostCard"; import { getTags } from "@/requests/tag"; import { getSelf } from "@/requests/user"; import { getPosts } from "@/requests/post"; 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"); const [user, setUser] = useState(); const [loading, setLoading] = useState(true); const [tags, setTags] = useState<{ [category: string]: { tags: TagType[]; priority: number }; }>(); const [tagRules, setTagRules] = useState<{ [key: number]: number }>(); const [reduceMotion, setReduceMotion] = useState(false); const { theme } = useTheme(); useEffect(() => { const mediaQuery = window.matchMedia("(prefers-reduced-motion: reduce)"); setReduceMotion(mediaQuery.matches); const handleChange = (event: MediaQueryListEvent) => { setReduceMotion(event.matches); }; mediaQuery.addEventListener("change", handleChange); return () => { mediaQuery.removeEventListener("change", handleChange); }; }, []); useEffect(() => { const loadUserAndPosts = async () => { setLoading(true); const tagResponse = await getTags(); if (tagResponse.ok) { const tagObject: { [category: string]: { tags: TagType[]; priority: number }; } = {}; for (const tag of await tagResponse.json()) { if (tag.name == "D2Jam") { continue; } if (tag.category) { if (tag.category.name in tagObject) { tagObject[tag.category.name].tags.push(tag); } else { tagObject[tag.category.name] = { tags: [tag], priority: tag.category.priority, }; } } } setTags(tagObject); } // Fetch the user const userResponse = await getSelf(); const userData = userResponse.ok ? await userResponse.json() : undefined; setUser(userData); // Fetch posts (with userSlug if user is available) const postsResponse = await getPosts(sort, time, false, tagRules, userData?.slug); setPosts(await postsResponse.json()); // Sticky posts // Fetch posts (with userSlug if user is available) const stickyPostsResponse = await getPosts(sort, time, true, tagRules, userData?.slug); setStickyPosts(await stickyPostsResponse.json()); setLoading(false); }; loadUserAndPosts(); }, [sort, time, tagRules]); const sorts: Record< PostSort, { name: string; icon: ReactNode; description: string } > = { top: { name: "Top", icon: , description: "Shows the most liked posts first", }, newest: { name: "Newest", icon: , description: "Shows the newest posts first", }, oldest: { name: "Oldest", icon: , description: "Shows the oldest posts first", }, }; const times: Record< PostTime, { name: string; icon: ReactNode; description: string } > = { hour: { name: "Hour", icon: , description: "Shows posts from the last hour", }, three_hours: { name: "Three Hours", icon: , description: "Shows posts from the last three hours", }, six_hours: { name: "Six Hours", icon: , description: "Shows posts from the last six hours", }, twelve_hours: { name: "Twelve Hours", icon: , description: "Shows posts from the last twelve hours", }, day: { name: "Day", icon: , description: "Shows posts from the last day", }, week: { name: "Week", icon: , description: "Shows posts from the last week", }, month: { name: "Month", icon: , description: "Shows posts from the last month", }, three_months: { name: "Three Months", icon: , description: "Shows posts from the last three months", }, six_months: { name: "Six Months", icon: , description: "Shows posts from the last six months", }, nine_months: { name: "Nine Months", icon: , description: "Shows posts from the last nine months", }, year: { name: "Year", icon: , description: "Shows posts from the last year", }, all: { name: "All Time", icon: , description: "Shows all posts", }, }; return (
{loading ? (
) : ( stickyPosts && stickyPosts.length > 0 && (
{stickyPosts.map((post) => ( ))}
) )}
{ setSort(key as PostSort); }} className="text-[#333] dark:text-white" > {Object.entries(sorts).map(([key, sort]) => ( {sort.name} ))} { setTime(key as PostTime); }} className="text-[#333] dark:text-white" > {Object.entries(times).map(([key, sort]) => ( {sort.name} ))}

Tag Filtering

{tags && Object.keys(tags).length > 0 ? ( Object.keys(tags) .sort( (tag1, tag2) => tags[tag2].priority - tags[tag1].priority ) .map((category: string) => (

{category}

{tags[category].tags.map((tag) => ( ) } key={tag.id} onClick={() => { if (!tagRules) { setTagRules({ [tag.id]: 1 }); } else { if (tag.id in tagRules) { if (tagRules[tag.id] === 1) { setTagRules({ ...tagRules, [tag.id]: -1, }); } else { const updatedRules = { ...tagRules }; delete updatedRules[tag.id]; setTagRules(updatedRules); } } else { setTagRules({ ...tagRules, [tag.id]: 1 }); } } }} className={`transition-all transform duration-500 ease-in-out cursor-pointer ${ !reduceMotion ? "hover:scale-110" : "" }`} style={{ color: tagRules && tag.id in tagRules ? tagRules[tag.id] === 1 ? theme == "dark" ? "#5ed4f7" : "#05b7eb" : theme == "dark" ? "#f78e5e" : "#eb2b05" : "", borderColor: tagRules && tag.id in tagRules ? tagRules[tag.id] === 1 ? theme == "dark" ? "#5ed4f7" : "#05b7eb" : theme == "dark" ? "#f78e5e" : "#eb2b05" : "", }} endContent={ tagRules && tag.id in tagRules && (tagRules[tag.id] === 1 ? ( ) : ( )) } > {tag.name} ))}
)) ) : (

No tags could be found

)}
{ setStyle(key as PostStyle); }} className="text-[#333] dark:text-white" > Cozy Compact Ultra Compact Adaptive
{loading ? (
) : (
{posts && posts.length > 0 ? ( posts.map((post) => ( )) ) : (

No posts match your filters

)}
)}
); return
; }