Fix build errors

This commit is contained in:
Ategon 2025-01-22 20:27:17 -05:00
parent a2307ec34e
commit 5462af1a5a
4 changed files with 737 additions and 734 deletions

View file

@ -19,7 +19,7 @@ interface NavbarUserProps {
isInJam?: boolean; isInJam?: boolean;
} }
export default function MobileNavbarUser({ export default async function MobileNavbarUser({
user, user,
jam, jam,
setIsInJam, setIsInJam,
@ -38,7 +38,7 @@ export default function MobileNavbarUser({
/> />
</DropdownTrigger> </DropdownTrigger>
<DropdownMenu> <DropdownMenu>
{jam && isInJam ? ( {jam && (await getCurrentJam())?.jam && isInJam ? (
<DropdownItem <DropdownItem
key="create-game" key="create-game"
href="/create-game" href="/create-game"
@ -47,7 +47,7 @@ export default function MobileNavbarUser({
Create Game Create Game
</DropdownItem> </DropdownItem>
) : null} ) : null}
{jam && !isInJam ? ( {jam && (await getCurrentJam())?.jam && !isInJam ? (
<DropdownItem <DropdownItem
key="join-event" key="join-event"
className="text-black" className="text-black"
@ -55,12 +55,12 @@ export default function MobileNavbarUser({
try { try {
const currentJam = await getCurrentJam(); const currentJam = await getCurrentJam();
if (!currentJam) { if (!currentJam || !currentJam.jam) {
toast.error("There is no jam to join"); toast.error("There is no jam to join");
return; return;
} }
if (await joinJam(currentJam.id)) { if (await joinJam(currentJam.jam.id)) {
setIsInJam(true); setIsInJam(true);
} }
} catch (error) { } catch (error) {

View file

@ -1,285 +1,286 @@
"use client"; "use client";
import React, { useState, useEffect, useCallback } from "react"; // import React, { useState, useEffect, useCallback } from "react";
import { getCookie } from "@/helpers/cookie"; // import { getCookie } from "@/helpers/cookie";
import { // import {
getCurrentJam, // getCurrentJam,
hasJoinedCurrentJam, // hasJoinedCurrentJam,
ActiveJamResponse, // ActiveJamResponse,
} from "@/helpers/jam"; // } from "@/helpers/jam";
export default function ThemeSlaughter() { export default function ThemeSlaughter() {
const [randomTheme, setRandomTheme] = useState(null); // const [randomTheme, setRandomTheme] = useState(null);
const [votedThemes, setVotedThemes] = useState([]); // const [votedThemes, setVotedThemes] = useState([]);
const [loading, setLoading] = useState(false); // const [loading, setLoading] = useState(false);
const [token, setToken] = useState(null); // const [token, setToken] = useState<string | null>(null);
const [hasJoined, setHasJoined] = useState<boolean>(false); // const [hasJoined, setHasJoined] = useState<boolean>(false);
const [activeJamResponse, setActiveJam] = useState<ActiveJamResponse | null>( // const [activeJamResponse, setActiveJam] = useState<ActiveJamResponse | null>(
null // null
); // );
const [phaseLoading, setPhaseLoading] = useState(true); // const [phaseLoading, setPhaseLoading] = useState(true);
// Fetch token on the client side // // Fetch token on the client side
useEffect(() => { // useEffect(() => {
const fetchedToken = getCookie("token"); // const fetchedToken = getCookie("token");
setToken(fetchedToken); // setToken(fetchedToken);
}, []); // }, []);
// Fetch the current jam phase using helpers/jam // // Fetch the current jam phase using helpers/jam
useEffect(() => { // useEffect(() => {
const fetchCurrentJamPhase = async () => { // const fetchCurrentJamPhase = async () => {
try { // try {
const activeJam = await getCurrentJam(); // const activeJam = await getCurrentJam();
setActiveJam(activeJam); // Set active jam details // setActiveJam(activeJam); // Set active jam details
} catch (error) { // } catch (error) {
console.error("Error fetching current jam:", error); // console.error("Error fetching current jam:", error);
} finally { // } finally {
setPhaseLoading(false); // Stop loading when phase is fetched // setPhaseLoading(false); // Stop loading when phase is fetched
} // }
}; // };
fetchCurrentJamPhase(); // fetchCurrentJamPhase();
}, []); // }, []);
// Fetch a random theme // // Fetch a random theme
const fetchRandomTheme = useCallback(async () => { // const fetchRandomTheme = useCallback(async () => {
if (!token) return; // Wait until token is available // if (!token) return; // Wait until token is available
if (!activeJamResponse) return; // if (!activeJamResponse) return;
if ( // if (
activeJamResponse && // activeJamResponse &&
activeJamResponse.jam && // activeJamResponse.jam &&
activeJamResponse.phase != "Survival" // activeJamResponse.phase != "Survival"
) { // ) {
return ( // return (
<div> // <div>
<h1>It&apos;s not Theme Survival phase.</h1> // <h1>It&apos;s not Theme Survival phase.</h1>
</div> // </div>
); // );
} // }
try { // try {
const response = await fetch( // const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD" // process.env.NEXT_PUBLIC_MODE === "PROD"
? "https://d2jam.com/api/v1/themes/random" // ? "https://d2jam.com/api/v1/themes/random"
: "http://localhost:3005/api/v1/themes/random", // : "http://localhost:3005/api/v1/themes/random",
{ // {
headers: { Authorization: `Bearer ${token}` }, // headers: { Authorization: `Bearer ${token}` },
credentials: "include", // credentials: "include",
} // }
); // );
if (response.ok) { // if (response.ok) {
const data = await response.json(); // const data = await response.json();
setRandomTheme(data); // setRandomTheme(data);
} else { // } else {
console.error("Failed to fetch random theme."); // console.error("Failed to fetch random theme.");
} // }
} catch (error) { // } catch (error) {
console.error("Error fetching random theme:", error); // console.error("Error fetching random theme:", error);
} // }
}, [activeJamResponse, token]); // }, [activeJamResponse, token]);
// Fetch voted themes // // Fetch voted themes
const fetchVotedThemes = useCallback(async () => { // const fetchVotedThemes = useCallback(async () => {
if (!token) return; // Wait until token is available // if (!token) return; // Wait until token is available
try { // try {
const response = await fetch( // const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD" // process.env.NEXT_PUBLIC_MODE === "PROD"
? "https://d2jam.com/api/v1/themes/voteSlaughter" // ? "https://d2jam.com/api/v1/themes/voteSlaughter"
: "http://localhost:3005/api/v1/themes/voteSlaughter", // : "http://localhost:3005/api/v1/themes/voteSlaughter",
{ // {
headers: { Authorization: `Bearer ${token}` }, // headers: { Authorization: `Bearer ${token}` },
credentials: "include", // credentials: "include",
} // }
); // );
if (response.ok) { // if (response.ok) {
const data = await response.json(); // const data = await response.json();
setVotedThemes(data); // setVotedThemes(data);
} else { // } else {
console.error("Failed to fetch voted themes."); // console.error("Failed to fetch voted themes.");
} // }
} catch (error) { // } catch (error) {
console.error("Error fetching voted themes:", error); // console.error("Error fetching voted themes:", error);
} // }
}, [token]); // }, [token]);
// Handle voting // // Handle voting
const handleVote = async (voteType) => { // const handleVote = async (voteType: string) => {
if (!randomTheme) return; // if (!randomTheme) return;
setLoading(true); // setLoading(true);
try { // try {
const response = await fetch( // const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD" // process.env.NEXT_PUBLIC_MODE === "PROD"
? "https://d2jam.com/api/v1/themes/voteSlaughter" // ? "https://d2jam.com/api/v1/themes/voteSlaughter"
: "http://localhost:3005/api/v1/themes/voteSlaughter", // : "http://localhost:3005/api/v1/themes/voteSlaughter",
{ // {
method: "POST", // method: "POST",
headers: { // headers: {
"Content-Type": "application/json", // "Content-Type": "application/json",
Authorization: `Bearer ${token}`, // Authorization: `Bearer ${token}`,
}, // },
credentials: "include", // credentials: "include",
body: JSON.stringify({ // body: JSON.stringify({
suggestionId: randomTheme.id, // suggestionId: randomTheme.id,
voteType, // voteType,
}), // }),
} // }
); // );
if (response.ok) { // if (response.ok) {
// Refresh data after voting // // Refresh data after voting
fetchRandomTheme(); // fetchRandomTheme();
fetchVotedThemes(); // fetchVotedThemes();
} else { // } else {
console.error("Failed to submit vote."); // console.error("Failed to submit vote.");
} // }
} catch (error) { // } catch (error) {
console.error("Error submitting vote:", error); // console.error("Error submitting vote:", error);
} finally { // } finally {
setLoading(false); // setLoading(false);
} // }
}; // };
// Handle resetting a vote from the grid // // Handle resetting a vote from the grid
const handleResetVote = async (themeId) => { // const handleResetVote = async (themeId: number) => {
try { // try {
setRandomTheme(votedThemes.find((theme) => theme.id === themeId)); // setRandomTheme(votedThemes.find((theme) => theme.id === themeId));
setVotedThemes((prev) => // setVotedThemes((prev) =>
prev.map((theme) => // prev.map((theme) =>
theme.id === themeId ? { ...theme, slaughterScore: 0 } : theme // theme.id === themeId ? { ...theme, slaughterScore: 0 } : theme
) // )
); // );
} catch (error) { // } catch (error) {
console.error("Error resetting vote:", error); // console.error("Error resetting vote:", error);
} // }
}; // };
useEffect(() => { // useEffect(() => {
if (token && activeJamResponse?.phase === "Survival") { // if (token && activeJamResponse?.phase === "Survival") {
fetchRandomTheme(); // fetchRandomTheme();
fetchVotedThemes(); // fetchVotedThemes();
} // }
}, [token, activeJamResponse, fetchRandomTheme, fetchVotedThemes]); // }, [token, activeJamResponse, fetchRandomTheme, fetchVotedThemes]);
useEffect(() => { // useEffect(() => {
const init = async () => { // const init = async () => {
const joined = await hasJoinedCurrentJam(); // const joined = await hasJoinedCurrentJam();
setHasJoined(joined); // setHasJoined(joined);
setLoading(false); // setLoading(false);
}; // };
init(); // init();
}, []); // }, []);
if (phaseLoading || loading) { // if (phaseLoading || loading) {
return <div>Loading...</div>; // return <div>Loading...</div>;
} // }
if (!hasJoined) { // if (!hasJoined) {
return ( // return (
<div className="p-6 bg-gray-100 dark:bg-gray-800 min-h-screen"> // <div className="p-6 bg-gray-100 dark:bg-gray-800 min-h-screen">
<h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-6"> // <h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-6">
Join the Jam First // Join the Jam First
</h1> // </h1>
<p className="text-gray-600 dark:text-gray-400"> // <p className="text-gray-600 dark:text-gray-400">
You need to join the current jam before you can join Theme Survival. // You need to join the current jam before you can join Theme Survival.
</p> // </p>
<button // <button
onClick={() => joinJam(activeJamResponse?.jam?.id)} // onClick={() => joinJam(activeJamResponse?.jam?.id)}
className="mt-4 px-6 py-2 bg-blue-500 text-white rounded-lg hover:bg-blue-600" // className="mt-4 px-6 py-2 bg-blue-500 text-white rounded-lg hover:bg-blue-600"
> // >
Join Jam // Join Jam
</button> // </button>
</div> // </div>
); // );
} // }
// Render message if not in Theme Slaughter phase // // Render message if not in Theme Slaughter phase
if (activeJamResponse?.phase !== "Survival") { // if (activeJamResponse?.phase !== "Survival") {
return ( // return (
<div className="p-6 bg-gray-100 dark:bg-gray-800 min-h-screen"> // <div className="p-6 bg-gray-100 dark:bg-gray-800 min-h-screen">
<h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-6"> // <h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-6">
Not in Theme Slaughter Phase // Not in Theme Slaughter Phase
</h1> // </h1>
<p className="text-gray-600 dark:text-gray-400"> // <p className="text-gray-600 dark:text-gray-400">
The current phase is{" "} // The current phase is{" "}
<strong>{activeJamResponse?.phase || "Unknown"}</strong>. Please come // <strong>{activeJamResponse?.phase || "Unknown"}</strong>. Please come
back during the Theme Slaughter phase. // back during the Theme Slaughter phase.
</p> // </p>
</div> // </div>
); // );
} // }
const loggedIn = getCookie("token"); // const loggedIn = getCookie("token");
if (!loggedIn) { // if (!loggedIn) {
return <div>Sign in to be able to join the Theme Survival</div>; // return <div>Sign in to be able to join the Theme Survival</div>;
} // }
return ( // return (
<div className="flex h-screen"> // <div className="flex h-screen">
{/* Left Side */} // {/* Left Side */}
<div className="w-1/2 p-6 bg-gray-100 dark:bg-gray-800 flex flex-col justify-start items-center"> // <div className="w-1/2 p-6 bg-gray-100 dark:bg-gray-800 flex flex-col justify-start items-center">
{randomTheme ? ( // {randomTheme ? (
<> // <>
<h2 className="text-2xl font-bold text-gray-800 dark:text-white mb-4"> // <h2 className="text-2xl font-bold text-gray-800 dark:text-white mb-4">
{randomTheme.suggestion} // {randomTheme.suggestion}
</h2> // </h2>
<div className="flex gap-4"> // <div className="flex gap-4">
<button // <button
onClick={() => handleVote("YES")} // onClick={() => handleVote("YES")}
className="px-6 py-3 bg-green-500 text-white font-bold rounded-lg hover:bg-green-600" // className="px-6 py-3 bg-green-500 text-white font-bold rounded-lg hover:bg-green-600"
disabled={loading} // disabled={loading}
> // >
YES // YES
</button> // </button>
<button // <button
onClick={() => handleVote("NO")} // onClick={() => handleVote("NO")}
className="px-6 py-3 bg-red-500 text-white font-bold rounded-lg hover:bg-red-600" // className="px-6 py-3 bg-red-500 text-white font-bold rounded-lg hover:bg-red-600"
disabled={loading} // disabled={loading}
> // >
NO // NO
</button> // </button>
<button // <button
onClick={() => handleVote("SKIP")} // onClick={() => handleVote("SKIP")}
className="px-6 py-3 bg-gray-500 text-white font-bold rounded-lg hover:bg-gray-600" // className="px-6 py-3 bg-gray-500 text-white font-bold rounded-lg hover:bg-gray-600"
disabled={loading} // disabled={loading}
> // >
SKIP // SKIP
</button> // </button>
</div> // </div>
</> // </>
) : ( // ) : (
<p className="text-gray-600 dark:text-gray-400"> // <p className="text-gray-600 dark:text-gray-400">
No themes available. // No themes available.
</p> // </p>
)} // )}
</div> // </div>
{/* Right Side */} // {/* Right Side */}
<div className="w-1/2 p-6 bg-white dark:bg-gray-900 overflow-y-auto"> // <div className="w-1/2 p-6 bg-white dark:bg-gray-900 overflow-y-auto">
<h3 className="text-xl font-bold text-gray-800 dark:text-white mb-4"> // <h3 className="text-xl font-bold text-gray-800 dark:text-white mb-4">
Your Votes // Your Votes
</h3> // </h3>
<div className="grid grid-cols-4 gap-4"> // <div className="grid grid-cols-4 gap-4">
{votedThemes.map((theme) => ( // {votedThemes.map((theme) => (
<div // <div
key={theme.id} // key={theme.id}
onClick={() => handleResetVote(theme.id)} // onClick={() => handleResetVote(theme.id)}
className={`p-4 rounded-lg cursor-pointer ${ // className={`p-4 rounded-lg cursor-pointer ${
theme.slaughterScore > 0 // theme.slaughterScore > 0
? "bg-green-500 text-white" // ? "bg-green-500 text-white"
: theme.slaughterScore < 0 // : theme.slaughterScore < 0
? "bg-red-500 text-white" // ? "bg-red-500 text-white"
: "bg-gray-300 text-black" // : "bg-gray-300 text-black"
}`} // }`}
> // >
{theme.suggestion} // {theme.suggestion}
</div> // </div>
))} // ))}
</div> // </div>
</div> // </div>
</div> // </div>
); // );
return <></>;
} }

View file

@ -1,281 +1,282 @@
"use client"; "use client";
import React, { useState, useEffect } from "react"; // import React, { useState, useEffect } from "react";
import { getCookie } from "@/helpers/cookie"; // import { getCookie } from "@/helpers/cookie";
import { // import {
getCurrentJam, // getCurrentJam,
hasJoinedCurrentJam, // hasJoinedCurrentJam,
ActiveJamResponse, // ActiveJamResponse,
} from "@/helpers/jam"; // } from "@/helpers/jam";
export default function ThemeSuggestions() { export default function ThemeSuggestions() {
const [suggestion, setSuggestion] = useState(""); // const [suggestion, setSuggestion] = useState("");
const [loading, setLoading] = useState(false); // const [loading, setLoading] = useState(false);
const [successMessage, setSuccessMessage] = useState(""); // const [successMessage, setSuccessMessage] = useState("");
const [errorMessage, setErrorMessage] = useState(""); // const [errorMessage, setErrorMessage] = useState("");
const [userSuggestions, setUserSuggestions] = useState([]); // const [userSuggestions, setUserSuggestions] = useState([]);
const [themeLimit, setThemeLimit] = useState(0); // const [themeLimit, setThemeLimit] = useState(0);
const [hasJoined, setHasJoined] = useState<boolean>(false); // const [hasJoined, setHasJoined] = useState<boolean>(false);
const [activeJamResponse, setActiveJamResponse] = // const [activeJamResponse, setActiveJamResponse] =
useState<ActiveJamResponse | null>(null); // useState<ActiveJamResponse | null>(null);
const [phaseLoading, setPhaseLoading] = useState(true); // Loading state for fetching phase // const [phaseLoading, setPhaseLoading] = useState(true); // Loading state for fetching phase
// Fetch the current jam phase using helpers/jam // // Fetch the current jam phase using helpers/jam
useEffect(() => { // useEffect(() => {
const fetchCurrentJamPhase = async () => { // const fetchCurrentJamPhase = async () => {
try { // try {
const activeJam = await getCurrentJam(); // const activeJam = await getCurrentJam();
setActiveJamResponse(activeJam); // Set active jam details // setActiveJamResponse(activeJam); // Set active jam details
if (activeJam?.jam) { // if (activeJam?.jam) {
setThemeLimit(activeJam.jam.themePerUser || Infinity); // Set theme limit // setThemeLimit(activeJam.jam.themePerUser || Infinity); // Set theme limit
} // }
} catch (error) { // } catch (error) {
console.error("Error fetching current jam:", error); // console.error("Error fetching current jam:", error);
} finally { // } finally {
setPhaseLoading(false); // Stop loading when phase is fetched // setPhaseLoading(false); // Stop loading when phase is fetched
} // }
}; // };
fetchCurrentJamPhase(); // fetchCurrentJamPhase();
}, []); // }, []);
// Fetch all suggestions for the logged-in user // // Fetch all suggestions for the logged-in user
const fetchSuggestions = async () => { // const fetchSuggestions = async () => {
try { // try {
const response = await fetch( // const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD" // process.env.NEXT_PUBLIC_MODE === "PROD"
? "https://d2jam.com/api/v1/themes/suggestion" // ? "https://d2jam.com/api/v1/themes/suggestion"
: "http://localhost:3005/api/v1/themes/suggestion", // : "http://localhost:3005/api/v1/themes/suggestion",
{ // {
headers: { Authorization: `Bearer ${getCookie("token")}` }, // headers: { Authorization: `Bearer ${getCookie("token")}` },
credentials: "include", // credentials: "include",
} // }
); // );
if (response.ok) { // if (response.ok) {
const data = await response.json(); // const data = await response.json();
setUserSuggestions(data); // setUserSuggestions(data);
} // }
} catch (error) { // } catch (error) {
console.error("Error fetching suggestions:", error); // console.error("Error fetching suggestions:", error);
} // }
}; // };
// Fetch suggestions only when phase is "Suggestion" // // Fetch suggestions only when phase is "Suggestion"
useEffect(() => { // useEffect(() => {
if (activeJamResponse?.phase === "Suggestion") { // if (activeJamResponse?.phase === "Suggestion") {
fetchSuggestions(); // fetchSuggestions();
} // }
}, [activeJamResponse]); // }, [activeJamResponse]);
// Handle form submission to add a new suggestion // // Handle form submission to add a new suggestion
const handleSubmit = async (e: React.FormEvent) => { // const handleSubmit = async (e: React.FormEvent) => {
e.preventDefault(); // e.preventDefault();
setLoading(true); // setLoading(true);
setSuccessMessage(""); // setSuccessMessage("");
setErrorMessage(""); // setErrorMessage("");
if (!suggestion.trim()) { // if (!suggestion.trim()) {
setErrorMessage("Suggestion cannot be empty."); // setErrorMessage("Suggestion cannot be empty.");
setLoading(false); // setLoading(false);
return; // return;
} // }
try { // try {
const token = getCookie("token"); // const token = getCookie("token");
if (!token) { // if (!token) {
throw new Error("User is not authenticated. Please log in."); // throw new Error("User is not authenticated. Please log in.");
} // }
const response = await fetch( // const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD" // process.env.NEXT_PUBLIC_MODE === "PROD"
? "https://d2jam.com/api/v1/themes/suggestion" // ? "https://d2jam.com/api/v1/themes/suggestion"
: "http://localhost:3005/api/v1/themes/suggestion", // : "http://localhost:3005/api/v1/themes/suggestion",
{ // {
method: "POST", // method: "POST",
headers: { // headers: {
"Content-Type": "application/json", // "Content-Type": "application/json",
Authorization: `Bearer ${token}`, // Authorization: `Bearer ${token}`,
}, // },
credentials: "include", // credentials: "include",
body: JSON.stringify({ suggestionText: suggestion }), // body: JSON.stringify({ suggestionText: suggestion }),
} // }
); // );
if (!response.ok) { // if (!response.ok) {
const errorData = await response.json(); // const errorData = await response.json();
throw new Error(errorData.error || "Failed to submit suggestion."); // throw new Error(errorData.error || "Failed to submit suggestion.");
} // }
setSuccessMessage("Suggestion added successfully!"); // setSuccessMessage("Suggestion added successfully!");
setSuggestion(""); // Clear input field // setSuggestion(""); // Clear input field
fetchSuggestions(); // Refresh suggestions list // fetchSuggestions(); // Refresh suggestions list
} catch (error) { // } catch (error) {
if (error instanceof Error) { // if (error instanceof Error) {
console.error("Error submitting suggestion:", error.message); // console.error("Error submitting suggestion:", error.message);
setErrorMessage(error.message || "An unexpected error occurred."); // setErrorMessage(error.message || "An unexpected error occurred.");
} else { // } else {
console.error("Unknown error:", error); // console.error("Unknown error:", error);
setErrorMessage("An unexpected error occurred."); // setErrorMessage("An unexpected error occurred.");
} // }
} finally { // } finally {
setLoading(false); // setLoading(false);
} // }
}; // };
// Handle deleting a suggestion // // Handle deleting a suggestion
const handleDelete = async (id: number) => { // const handleDelete = async (id: number) => {
try { // try {
const token = getCookie("token"); // const token = getCookie("token");
const response = await fetch( // const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD" // process.env.NEXT_PUBLIC_MODE === "PROD"
? `https://d2jam.com/api/v1/themes/suggestion/${id}` // ? `https://d2jam.com/api/v1/themes/suggestion/${id}`
: `http://localhost:3005/api/v1/themes/suggestion/${id}`, // : `http://localhost:3005/api/v1/themes/suggestion/${id}`,
{ // {
method: "DELETE", // method: "DELETE",
headers: { Authorization: `Bearer ${token}` }, // headers: { Authorization: `Bearer ${token}` },
credentials: "include", // credentials: "include",
} // }
); // );
if (!response.ok) { // if (!response.ok) {
throw new Error("Failed to delete suggestion."); // throw new Error("Failed to delete suggestion.");
} // }
fetchSuggestions(); // Refresh suggestions list // fetchSuggestions(); // Refresh suggestions list
} catch (error) { // } catch (error) {
console.error("Error deleting suggestion:", error); // console.error("Error deleting suggestion:", error);
} // }
}; // };
useEffect(() => { // useEffect(() => {
const init = async () => { // const init = async () => {
const joined = await hasJoinedCurrentJam(); // const joined = await hasJoinedCurrentJam();
setHasJoined(joined); // setHasJoined(joined);
setLoading(false); // setLoading(false);
}; // };
init(); // init();
}, []); // }, []);
// Render loading state while fetching phase // // Render loading state while fetching phase
if (phaseLoading || loading) { // if (phaseLoading || loading) {
return <div>Loading...</div>; // return <div>Loading...</div>;
} // }
if (!hasJoined) { // if (!hasJoined) {
return ( // return (
<div className="p-6 bg-gray-100 dark:bg-gray-800 min-h-screen"> // <div className="p-6 bg-gray-100 dark:bg-gray-800 min-h-screen">
<h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-6"> // <h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-6">
Join the Jam First // Join the Jam First
</h1> // </h1>
<p className="text-gray-600 dark:text-gray-400"> // <p className="text-gray-600 dark:text-gray-400">
You need to join the current jam before you can suggest themes. // You need to join the current jam before you can suggest themes.
</p> // </p>
<button // <button
onClick={() => joinJam(activeJamResponse?.jam?.id)} // onClick={() => joinJam(activeJamResponse?.jam?.id)}
className="mt-4 px-6 py-2 bg-blue-500 text-white rounded-lg hover:bg-blue-600" // className="mt-4 px-6 py-2 bg-blue-500 text-white rounded-lg hover:bg-blue-600"
> // >
Join Jam // Join Jam
</button> // </button>
</div> // </div>
); // );
} // }
const token = getCookie("token"); // const token = getCookie("token");
if (!token) { // if (!token) {
return <div>Sign in to be able to suggest themes</div>; // return <div>Sign in to be able to suggest themes</div>;
} // }
// Render message if not in Suggestion phase // // Render message if not in Suggestion phase
if (activeJamResponse?.phase !== "Suggestion") { // if (activeJamResponse?.phase !== "Suggestion") {
return ( // return (
<div className="p-6 bg-gray-100 dark:bg-gray-800 min-h-screen"> // <div className="p-6 bg-gray-100 dark:bg-gray-800 min-h-screen">
<h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-6"> // <h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-6">
Not in Suggestion Phase // Not in Suggestion Phase
</h1> // </h1>
<p className="text-gray-600 dark:text-gray-400"> // <p className="text-gray-600 dark:text-gray-400">
The current phase is{" "} // The current phase is{" "}
<strong>{activeJamResponse?.phase || "Unknown"}</strong>. Please come // <strong>{activeJamResponse?.phase || "Unknown"}</strong>. Please come
back during the Suggestion phase. // back during the Suggestion phase.
</p> // </p>
</div> // </div>
); // );
} // }
return ( // return (
<div className="max-w-md mx-auto mt-8 p-6 bg-white dark:bg-gray-800 rounded-lg shadow-md"> // <div className="max-w-md mx-auto mt-8 p-6 bg-white dark:bg-gray-800 rounded-lg shadow-md">
<h2 className="text-xl font-bold text-gray-800 dark:text-white mb-4"> // <h2 className="text-xl font-bold text-gray-800 dark:text-white mb-4">
Submit Your Theme Suggestion // Submit Your Theme Suggestion
</h2> // </h2>
{/* Hide form if user has reached their limit */} // {/* Hide form if user has reached their limit */}
{userSuggestions.length < themeLimit ? ( // {userSuggestions.length < themeLimit ? (
<form onSubmit={handleSubmit} className="flex flex-col gap-4"> // <form onSubmit={handleSubmit} className="flex flex-col gap-4">
<textarea // <textarea
className="w-full p-3 border rounded-lg focus:outline-none text-gray-600 focus:ring focus:ring-blue-300 dark:bg-gray-700 dark:text-white" // className="w-full p-3 border rounded-lg focus:outline-none text-gray-600 focus:ring focus:ring-blue-300 dark:bg-gray-700 dark:text-white"
placeholder="Enter your theme suggestion..." // placeholder="Enter your theme suggestion..."
value={suggestion} // value={suggestion}
onChange={(e) => { // onChange={(e) => {
if (e.target.value.length <= 32) { // if (e.target.value.length <= 32) {
setSuggestion(e.target.value); // setSuggestion(e.target.value);
} // }
}} // }}
rows={1} // rows={1}
maxLength={32} // maxLength={32}
></textarea> // ></textarea>
{errorMessage && ( // {errorMessage && (
<p className="text-red-500 text-sm">{errorMessage}</p> // <p className="text-red-500 text-sm">{errorMessage}</p>
)} // )}
{successMessage && ( // {successMessage && (
<p className="text-green-500 text-sm">{successMessage}</p> // <p className="text-green-500 text-sm">{successMessage}</p>
)} // )}
<button // <button
type="submit" // type="submit"
className={`w-full py-2 px-4 bg-blue-500 text-white font-semibold rounded-lg shadow-md hover:bg-blue-600 focus:outline-none focus:ring focus:ring-blue-300 ${ // className={`w-full py-2 px-4 bg-blue-500 text-white font-semibold rounded-lg shadow-md hover:bg-blue-600 focus:outline-none focus:ring focus:ring-blue-300 ${
loading ? "opacity-50 cursor-not-allowed" : "" // loading ? "opacity-50 cursor-not-allowed" : ""
}`} // }`}
disabled={loading} // disabled={loading}
> // >
{loading ? "Submitting..." : "Submit Suggestion"} // {loading ? "Submitting..." : "Submit Suggestion"}
</button> // </button>
</form> // </form>
) : ( // ) : (
<p className="text-yellow-500 text-sm"> // <p className="text-yellow-500 text-sm">
You&apos;ve reached your theme suggestion limit for this jam! // You&apos;ve reached your theme suggestion limit for this jam!
</p> // </p>
)} // )}
{/* List of user's suggestions */} // {/* List of user's suggestions */}
<div className="mt-6"> // <div className="mt-6">
<h3 className="text-lg font-semibold text-gray-800 dark:text-white mb-4"> // <h3 className="text-lg font-semibold text-gray-800 dark:text-white mb-4">
Your Suggestions // Your Suggestions
</h3> // </h3>
{userSuggestions.length > 0 ? ( // {userSuggestions.length > 0 ? (
<ul className="space-y-4"> // <ul className="space-y-4">
{userSuggestions.map((suggestion) => ( // {userSuggestions.map((suggestion) => (
<li // <li
key={suggestion.id} // key={suggestion.id}
className="flex justify-between items-center text-gray-400 bg-gray-100 dark:bg-gray-700 p-3 rounded-lg shadow-sm" // className="flex justify-between items-center text-gray-400 bg-gray-100 dark:bg-gray-700 p-3 rounded-lg shadow-sm"
> // >
<span>{suggestion.suggestion}</span> // <span>{suggestion.suggestion}</span>
<button // <button
onClick={() => handleDelete(suggestion.id)} // onClick={() => handleDelete(suggestion.id)}
className="text-red-500 hover:text-red-700 font-semibold" // className="text-red-500 hover:text-red-700 font-semibold"
> // >
Delete // Delete
</button> // </button>
</li> // </li>
))} // ))}
</ul> // </ul>
) : ( // ) : (
<p className="text-gray-600 dark:text-gray-400"> // <p className="text-gray-600 dark:text-gray-400">
You haven&apos;t submitted any suggestions yet. // You haven&apos;t submitted any suggestions yet.
</p> // </p>
)} // )}
</div> // </div>
</div> // </div>
); // );
return <></>;
} }

View file

@ -1,244 +1,245 @@
"use client"; "use client";
import React, { useState, useEffect } from "react"; // import React, { useState, useEffect } from "react";
import { getCookie } from "@/helpers/cookie"; // import { getCookie } from "@/helpers/cookie";
import { // import {
getCurrentJam, // getCurrentJam,
hasJoinedCurrentJam, // hasJoinedCurrentJam,
ActiveJamResponse, // ActiveJamResponse,
} from "@/helpers/jam"; // } from "@/helpers/jam";
export default function VotingPage() { export default function VotingPage() {
const [themes, setThemes] = useState([]); return <></>;
const [loading, setLoading] = useState(false); // const [themes, setThemes] = useState([]);
const [activeJamResponse, setActiveJamResponse] = // const [loading, setLoading] = useState(false);
useState<ActiveJamResponse | null>(null); // const [activeJamResponse, setActiveJamResponse] =
const [hasJoined, setHasJoined] = useState<boolean>(false); // useState<ActiveJamResponse | null>(null);
const [phaseLoading, setPhaseLoading] = useState(true); // Loading state for fetching phase // const [hasJoined, setHasJoined] = useState<boolean>(false);
const token = getCookie("token"); // const [phaseLoading, setPhaseLoading] = useState(true); // Loading state for fetching phase
// const token = getCookie("token");
// Fetch the current jam phase using helpers/jam // // Fetch the current jam phase using helpers/jam
useEffect(() => { // useEffect(() => {
const fetchCurrentJamPhase = async () => { // const fetchCurrentJamPhase = async () => {
try { // try {
const activeJam = await getCurrentJam(); // const activeJam = await getCurrentJam();
setActiveJamResponse(activeJam); // Set active jam details // setActiveJamResponse(activeJam); // Set active jam details
} catch (error) { // } catch (error) {
console.error("Error fetching current jam:", error); // console.error("Error fetching current jam:", error);
} finally { // } finally {
setPhaseLoading(false); // Stop loading when phase is fetched // setPhaseLoading(false); // Stop loading when phase is fetched
} // }
}; // };
fetchCurrentJamPhase(); // fetchCurrentJamPhase();
}, []); // }, []);
// Fetch themes only when phase is "Voting" // // Fetch themes only when phase is "Voting"
useEffect(() => { // useEffect(() => {
// Fetch top N themes with voting scores // // Fetch top N themes with voting scores
async function fetchThemes() { // async function fetchThemes() {
if (!token || !activeJamResponse) return; // if (!token || !activeJamResponse) return;
try { // try {
const response = await fetch( // const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD" // process.env.NEXT_PUBLIC_MODE === "PROD"
? "https://d2jam.com/api/v1/themes/top-themes" // ? "https://d2jam.com/api/v1/themes/top-themes"
: "http://localhost:3005/api/v1/themes/top-themes", // : "http://localhost:3005/api/v1/themes/top-themes",
{ // {
headers: { Authorization: `Bearer ${token}` }, // headers: { Authorization: `Bearer ${token}` },
credentials: "include", // credentials: "include",
} // }
); // );
if (response.ok) { // if (response.ok) {
const themes = await response.json(); // const themes = await response.json();
console.log("Fetched themes:", themes); // Debug log // console.log("Fetched themes:", themes); // Debug log
// Fetch user's votes for these themes // // Fetch user's votes for these themes
const votesResponse = await fetch( // const votesResponse = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD" // process.env.NEXT_PUBLIC_MODE === "PROD"
? "https://d2jam.com/api/v1/themes/votes" // ? "https://d2jam.com/api/v1/themes/votes"
: "http://localhost:3005/api/v1/themes/votes", // : "http://localhost:3005/api/v1/themes/votes",
{ // {
headers: { Authorization: `Bearer ${token}` }, // headers: { Authorization: `Bearer ${token}` },
credentials: "include", // credentials: "include",
} // }
); // );
if (votesResponse.ok) { // if (votesResponse.ok) {
const votes = await votesResponse.json(); // const votes = await votesResponse.json();
console.log("Fetched votes:", votes); // Debug log // console.log("Fetched votes:", votes); // Debug log
// Merge themes with user's votes // // Merge themes with user's votes
const themesWithVotes = themes.map((theme) => { // const themesWithVotes = themes.map((theme) => {
const vote = votes.find((v) => v.themeSuggestionId === theme.id); // const vote = votes.find((v) => v.themeSuggestionId === theme.id);
console.log(`Theme ${theme.id} vote:`, vote); // Debug log // console.log(`Theme ${theme.id} vote:`, vote); // Debug log
return { // return {
...theme, // ...theme,
votingScore: vote ? vote.votingScore : null, // votingScore: vote ? vote.votingScore : null,
}; // };
}); // });
console.log("Themes with votes:", themesWithVotes); // Debug log // console.log("Themes with votes:", themesWithVotes); // Debug log
setThemes(themesWithVotes); // setThemes(themesWithVotes);
} // }
} else { // } else {
console.error("Failed to fetch themes."); // console.error("Failed to fetch themes.");
} // }
} catch (error) { // } catch (error) {
console.error("Error fetching themes:", error); // console.error("Error fetching themes:", error);
} // }
} // }
if (activeJamResponse?.phase === "Voting") { // if (activeJamResponse?.phase === "Voting") {
fetchThemes(); // fetchThemes();
} // }
}, [activeJamResponse, token]); // }, [activeJamResponse, token]);
// Handle voting // // Handle voting
const handleVote = async (themeId, votingScore) => { // const handleVote = async (themeId, votingScore) => {
setLoading(true); // setLoading(true);
try { // try {
const response = await fetch( // const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD" // process.env.NEXT_PUBLIC_MODE === "PROD"
? "https://d2jam.com/api/v1/themes/vote" // ? "https://d2jam.com/api/v1/themes/vote"
: "http://localhost:3005/api/v1/themes/vote", // : "http://localhost:3005/api/v1/themes/vote",
{ // {
method: "POST", // method: "POST",
headers: { // headers: {
"Content-Type": "application/json", // "Content-Type": "application/json",
Authorization: `Bearer ${token}`, // Authorization: `Bearer ${token}`,
}, // },
credentials: "include", // credentials: "include",
body: JSON.stringify({ suggestionId: themeId, votingScore }), // body: JSON.stringify({ suggestionId: themeId, votingScore }),
} // }
); // );
if (response.ok) { // if (response.ok) {
// Just update the local state instead of re-fetching all themes // // Just update the local state instead of re-fetching all themes
setThemes((prevThemes) => // setThemes((prevThemes) =>
prevThemes.map((theme) => // prevThemes.map((theme) =>
theme.id === themeId ? { ...theme, votingScore } : theme // theme.id === themeId ? { ...theme, votingScore } : theme
) // )
); // );
} else { // } else {
console.error("Failed to submit vote."); // console.error("Failed to submit vote.");
} // }
} catch (error) { // } catch (error) {
console.error("Error submitting vote:", error); // console.error("Error submitting vote:", error);
} finally { // } finally {
setLoading(false); // setLoading(false);
} // }
}; // };
useEffect(() => { // useEffect(() => {
const init = async () => { // const init = async () => {
const joined = await hasJoinedCurrentJam(); // const joined = await hasJoinedCurrentJam();
setHasJoined(joined); // setHasJoined(joined);
setLoading(false); // setLoading(false);
}; // };
init(); // init();
}, []); // }, []);
if (phaseLoading || loading) { // if (phaseLoading || loading) {
return <div>Loading...</div>; // return <div>Loading...</div>;
} // }
if (!hasJoined) { // if (!hasJoined) {
return ( // return (
<div className="p-6 bg-gray-100 dark:bg-gray-800 min-h-screen"> // <div className="p-6 bg-gray-100 dark:bg-gray-800 min-h-screen">
<h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-6"> // <h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-6">
Join the Jam First // Join the Jam First
</h1> // </h1>
<p className="text-gray-600 dark:text-gray-400"> // <p className="text-gray-600 dark:text-gray-400">
You need to join the current jam before you can vote themes. // You need to join the current jam before you can vote themes.
</p> // </p>
<button // <button
onClick={() => joinJam(activeJamResponse?.jam?.id)} // onClick={() => joinJam(activeJamResponse?.jam?.id)}
className="mt-4 px-6 py-2 bg-blue-500 text-white rounded-lg hover:bg-blue-600" // className="mt-4 px-6 py-2 bg-blue-500 text-white rounded-lg hover:bg-blue-600"
> // >
Join Jam // Join Jam
</button> // </button>
</div> // </div>
); // );
} // }
if (activeJamResponse?.phase !== "Voting") { // if (activeJamResponse?.phase !== "Voting") {
return ( // return (
<div className="p-4 bg-gray-100 dark:bg-gray-800 min-h-screen"> // <div className="p-4 bg-gray-100 dark:bg-gray-800 min-h-screen">
<h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-4"> // <h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-4">
Not in Voting Phase // Not in Voting Phase
</h1> // </h1>
<p className="text-gray-600 dark:text-gray-400"> // <p className="text-gray-600 dark:text-gray-400">
The current phase is{" "} // The current phase is{" "}
<strong>{activeJamResponse?.phase || "Unknown"}</strong>. Please come // <strong>{activeJamResponse?.phase || "Unknown"}</strong>. Please come
back during the Voting phase. // back during the Voting phase.
</p> // </p>
</div> // </div>
); // );
} // }
const loggedIn = getCookie("token"); // const loggedIn = getCookie("token");
if (!loggedIn) { // if (!loggedIn) {
return <div>Sign in to be able to vote</div>; // return <div>Sign in to be able to vote</div>;
} // }
return ( // return (
<div className="p-3 bg-gray-100 dark:bg-gray-800 min-h-screen"> // <div className="p-3 bg-gray-100 dark:bg-gray-800 min-h-screen">
<h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-4"> // <h1 className="text-2xl font-bold text-gray-800 dark:text-white mb-4">
Voting Phase // Voting Phase
</h1> // </h1>
<div className="space-y-2"> // <div className="space-y-2">
{themes.map((theme) => ( // {themes.map((theme) => (
<div // <div
key={theme.id} // key={theme.id}
className="p-3 bg-white dark:bg-gray-900 rounded-lg shadow-md flex items-center" // className="p-3 bg-white dark:bg-gray-900 rounded-lg shadow-md flex items-center"
> // >
{/* Voting Buttons */} // {/* Voting Buttons */}
<div className="flex gap-1 mr-4"> // <div className="flex gap-1 mr-4">
<button // <button
onClick={() => handleVote(theme.id, -1)} // onClick={() => handleVote(theme.id, -1)}
className={`px-3 py-2 rounded-lg ${ // className={`px-3 py-2 rounded-lg ${
theme.votingScore === -1 // theme.votingScore === -1
? "bg-red-500 text-white" // ? "bg-red-500 text-white"
: "bg-gray-300 text-black hover:bg-red-500 hover:text-white" // : "bg-gray-300 text-black hover:bg-red-500 hover:text-white"
}`} // }`}
disabled={loading} // disabled={loading}
> // >
-1 // -1
</button> // </button>
<button // <button
onClick={() => handleVote(theme.id, 0)} // onClick={() => handleVote(theme.id, 0)}
className={`px-3 py-2 rounded-lg ${ // className={`px-3 py-2 rounded-lg ${
theme.votingScore === 0 // theme.votingScore === 0
? "bg-gray-500 text-white" // ? "bg-gray-500 text-white"
: "bg-gray-300 text-black hover:bg-gray-500 hover:text-white" // : "bg-gray-300 text-black hover:bg-gray-500 hover:text-white"
}`} // }`}
disabled={loading} // disabled={loading}
> // >
0 // 0
</button> // </button>
<button // <button
onClick={() => handleVote(theme.id, +1)} // onClick={() => handleVote(theme.id, +1)}
className={`px-3 py-2 rounded-lg ${ // className={`px-3 py-2 rounded-lg ${
theme.votingScore === +1 // theme.votingScore === +1
? "bg-green-500 text-white" // ? "bg-green-500 text-white"
: "bg-gray-300 text-black hover:bg-green-500 hover:text-white" // : "bg-gray-300 text-black hover:bg-green-500 hover:text-white"
}`} // }`}
disabled={loading} // disabled={loading}
> // >
+1 // +1
</button> // </button>
</div> // </div>
{/* Theme Suggestion */} // {/* Theme Suggestion */}
<div className="text-gray-800 dark:text-white"> // <div className="text-gray-800 dark:text-white">
{theme.suggestion} // {theme.suggestion}
</div> // </div>
</div> // </div>
))} // ))}
</div> // </div>
</div> // </div>
); // );
} }