From f4be4722d127e2394bc63e5443592f97ec5d978c Mon Sep 17 00:00:00 2001 From: Pinapelz Date: Sun, 7 Dec 2025 18:03:59 -0800 Subject: add score export on personal score view pages --- frontend/src/pages/Score.tsx | 53 +++++++++++++++++++++++++++++++++++++------- 1 file changed, 45 insertions(+), 8 deletions(-) (limited to 'frontend') diff --git a/frontend/src/pages/Score.tsx b/frontend/src/pages/Score.tsx index 8787137..f1d4049 100644 --- a/frontend/src/pages/Score.tsx +++ b/frontend/src/pages/Score.tsx @@ -1,9 +1,9 @@ import { useEffect, useState, useCallback } from "react"; -import LoadingDisplay from "../components/LoadingDisplay"; +import LoadingDisplay from "../components/LoadingDisplay"; import { useAuth } from "../contexts/AuthContext"; import { useNavigate } from "react-router"; import { NavBar } from "../components/NavBar"; -import type { SupportedGame} from "../types/game"; +import type { SupportedGame } from "../types/game"; import SessionExpiredPopup from "../components/SessionExpiredPopup"; import ScoreDisplay from "../components/displays/GenericScoreDisplay"; import DancerushScoreDisplay from "../components/displays/DancerushScoreDisplay"; @@ -48,6 +48,31 @@ const Score = () => { } }; + const exportScores = async () => { + try { + const response = await fetch( + import.meta.env.VITE_API_URL + + `/exportScores?internalGameName=${gameName}&page=${currentPage}`, + { + headers: { + "Content-Type": "application/json", + }, + credentials: "include", + }, + ); + const blob = await response.blob(); + const url = window.URL.createObjectURL(blob); + const a = document.createElement("a"); + a.href = url; + a.download = `scores_${gameName}_${currentPage}.json`; + a.click(); + window.URL.revokeObjectURL(url); + } catch (error) { + console.error("Export failed:", error); + alert("Network error during export. Please try again."); + } + }; + useEffect(() => { try{ fetch(import.meta.env.VITE_API_URL + `/supportedGames`) @@ -107,8 +132,7 @@ const Score = () => { } if(targetUserId && targetUserId !== user.id.toString()){ setViewingOwnScores(false); - } - else{ + } else { setViewingOwnScores(true); } url.searchParams.append("internalGameName", gameName); @@ -116,7 +140,9 @@ const Score = () => { url.searchParams.append("sortKey", requestOrder); url.searchParams.append("direction", "asc"); - const response = await fetch(url.toString(), {credentials: 'include'}); + const response = await fetch(url.toString(), { + credentials: "include", + }); if (!response.ok) throw new Error("Failed to fetch scores"); const data = await response.json(); setUsername(data.user); @@ -137,7 +163,11 @@ const Score = () => { const handleDeleteScore = async (scoreId: number) => { if (!user) return; - if (!confirm("Are you sure you want to delete this score? This action cannot be undone.")) { + if ( + !confirm( + "Are you sure you want to delete this score? This action cannot be undone.", + ) + ) { return; } @@ -180,7 +210,7 @@ const Score = () => { if (isLoading || loading) { return ( - + ); } @@ -191,7 +221,8 @@ const Score = () => {

- {viewingOwnScores ? "Your Scores" : `${username}'s Scores`} for {formattedGameName} + {viewingOwnScores ? "Your Scores" : `${username}'s Scores`} for{" "} + {formattedGameName}