Add post times

This commit is contained in:
Ategon 2025-01-20 00:37:30 -05:00
parent 3df39a4405
commit feb80916cc
5 changed files with 188 additions and 16 deletions

View file

@ -1,7 +1,17 @@
import type { NextConfig } from "next";
const nextConfig: NextConfig = {
/* config options here */
images: {
remotePatterns: [
{
protocol: "https",
hostname: "static-cdn.jtvnw.net",
port: "",
pathname: "/**",
search: "",
},
],
},
};
export default nextConfig;

View file

@ -1,6 +1,6 @@
"use client";
import { useEffect, useState } from "react";
import { ReactNode, useEffect, useState } from "react";
import PostCard from "./PostCard";
import { PostType } from "@/types/PostType";
import {
@ -14,12 +14,31 @@ import { PostSort } from "@/types/PostSort";
import { PostStyle } from "@/types/PostStyle";
import { getCookie } from "@/helpers/cookie";
import { UserType } from "@/types/UserType";
import { LoaderCircle } from "lucide-react";
import {
Calendar,
Calendar1,
CalendarArrowDown,
CalendarCog,
CalendarDays,
CalendarFold,
CalendarRange,
Clock1,
Clock2,
Clock3,
Clock4,
ClockArrowDown,
ClockArrowUp,
LoaderCircle,
Sparkles,
Trophy,
} from "lucide-react";
import { toast } from "react-toastify";
import { PostTime } from "@/types/PostTimes";
export default function Posts() {
const [posts, setPosts] = useState<PostType[]>();
const [sort, setSort] = useState<PostSort>("newest");
const [time, setTime] = useState<PostTime>("all");
const [style, setStyle] = useState<PostStyle>("cozy");
const [user, setUser] = useState<UserType>();
const [loading, setLoading] = useState<boolean>(true);
@ -45,8 +64,8 @@ export default function Posts() {
// Fetch posts with userSlug if user is available
const postsResponse = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD"
? `https://d2jam.com/api/v1/posts?sort=${sort}&user=${userData.slug}`
: `http://localhost:3005/api/v1/posts?sort=${sort}&user=${userData.slug}`
? `https://d2jam.com/api/v1/posts?sort=${sort}&user=${userData.slug}&time=${time}`
: `http://localhost:3005/api/v1/posts?sort=${sort}&user=${userData.slug}&time=${time}`
);
setPosts(await postsResponse.json());
setLoading(false);
@ -56,8 +75,8 @@ export default function Posts() {
// Fetch posts without userSlug if user is not available
const postsResponse = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD"
? `https://d2jam.com/api/v1/posts?sort=${sort}`
: `http://localhost:3005/api/v1/posts?sort=${sort}`
? `https://d2jam.com/api/v1/posts?sort=${sort}&time=${time}`
: `http://localhost:3005/api/v1/posts?sort=${sort}&time=${time}`
);
setPosts(await postsResponse.json());
setLoading(false);
@ -65,7 +84,94 @@ export default function Posts() {
};
loadUserAndPosts();
}, [sort]);
}, [sort, time]);
const sorts: Record<
PostSort,
{ name: string; icon: ReactNode; description: string }
> = {
top: {
name: "Top",
icon: <Trophy />,
description: "Shows the most liked posts first",
},
newest: {
name: "Newest",
icon: <ClockArrowUp />,
description: "Shows the newest posts first",
},
oldest: {
name: "Oldest",
icon: <ClockArrowDown />,
description: "Shows the oldest posts first",
},
};
const times: Record<
PostTime,
{ name: string; icon: ReactNode; description: string }
> = {
hour: {
name: "Hour",
icon: <Clock1 />,
description: "Shows posts from the last hour",
},
three_hours: {
name: "Three Hours",
icon: <Clock2 />,
description: "Shows posts from the last three hours",
},
six_hours: {
name: "Six Hours",
icon: <Clock3 />,
description: "Shows posts from the last six hours",
},
twelve_hours: {
name: "Twelve Hours",
icon: <Clock4 />,
description: "Shows posts from the last twelve hours",
},
day: {
name: "Day",
icon: <Calendar />,
description: "Shows posts from the last day",
},
week: {
name: "Week",
icon: <CalendarDays />,
description: "Shows posts from the last week",
},
month: {
name: "Month",
icon: <CalendarRange />,
description: "Shows posts from the last month",
},
three_months: {
name: "Three Months",
icon: <CalendarFold />,
description: "Shows posts from the last three months",
},
six_months: {
name: "Six Months",
icon: <CalendarCog />,
description: "Shows posts from the last six months",
},
nine_months: {
name: "Nine Months",
icon: <CalendarArrowDown />,
description: "Shows posts from the last nine months",
},
year: {
name: "Year",
icon: <Calendar1 />,
description: "Shows posts from the last year",
},
all: {
name: "All Times",
icon: <Sparkles />,
description: "Shows all posts",
},
};
return (
<div>
@ -78,7 +184,7 @@ export default function Posts() {
className="text-xs bg-white dark:bg-[#252525] !duration-250 !ease-linear !transition-all text-[#333] dark:text-white"
variant="faded"
>
{sort.charAt(0).toUpperCase() + sort.slice(1)}
{sorts[sort]?.name}
</Button>
</DropdownTrigger>
<DropdownMenu
@ -87,9 +193,42 @@ export default function Posts() {
}}
className="text-[#333] dark:text-white"
>
<DropdownItem key="newest">Newest</DropdownItem>
<DropdownItem key="top">Top</DropdownItem>
<DropdownItem key="oldest">Oldest</DropdownItem>
{Object.entries(sorts).map(([key, sort]) => (
<DropdownItem
key={key}
startContent={sort.icon}
description={sort.description}
>
{sort.name}
</DropdownItem>
))}
</DropdownMenu>
</Dropdown>
<Dropdown>
<DropdownTrigger>
<Button
size="sm"
className="text-xs bg-white dark:bg-[#252525] !duration-250 !ease-linear !transition-all text-[#333] dark:text-white"
variant="faded"
>
{times[time]?.name}
</Button>
</DropdownTrigger>
<DropdownMenu
onAction={(key) => {
setTime(key as PostTime);
}}
className="text-[#333] dark:text-white"
>
{Object.entries(times).map(([key, sort]) => (
<DropdownItem
key={key}
startContent={sort.icon}
description={sort.description}
>
{sort.name}
</DropdownItem>
))}
</DropdownMenu>
</Dropdown>
<Button
@ -138,10 +277,15 @@ export default function Posts() {
</div>
) : (
<div className="flex flex-col gap-3 p-4">
{posts &&
{posts && posts.length > 0 ? (
posts.map((post) => (
<PostCard key={post.id} post={post} style={style} user={user} />
))}
))
) : (
<p className="text-center text-[#333] dark:text-white transition-color duration-250 ease-linear">
No posts match your filters
</p>
)}
</div>
)}
</div>

View file

@ -2,6 +2,8 @@
import { useEffect, useState } from "react";
import { FeaturedStreamerType } from "@/types/FeaturedStreamerType";
import { Image } from "@nextui-org/react";
import NextImage from "next/image";
export default function Streams() {
const [streamers, setStreamers] = useState<FeaturedStreamerType[]>([]);
@ -64,10 +66,13 @@ export default function Streams() {
margin: "0 auto",
}}
>
<img
<Image
as={NextImage}
src={currentStreamer.thumbnailUrl}
alt={`${currentStreamer.userName}'s thumbnail`}
style={{ width: "100%", borderRadius: "4px", marginBottom: "10px" }}
width={320}
height={180}
/>
<a
href={`https://twitch.tv/${currentStreamer.userName}`}

View file

@ -1 +1 @@
export type PostSort = "newest" | "oldest";
export type PostSort = "newest" | "oldest" | "top";

13
src/types/PostTimes.ts Normal file
View file

@ -0,0 +1,13 @@
export type PostTime =
| "hour"
| "three_hours"
| "six_hours"
| "twelve_hours"
| "day"
| "week"
| "month"
| "three_months"
| "six_months"
| "nine_months"
| "year"
| "all";