aboutsummaryrefslogtreecommitdiffstats
path: root/frontend/src/pages/Import.tsx
diff options
context:
space:
mode:
authorPinapelz <yukais@pinapelz.com>2025-07-05 21:42:22 -0700
committerPinapelz <yukais@pinapelz.com>2025-07-05 21:42:22 -0700
commit400d772cc391d979747510776fa8acfb5a1d00cb (patch)
tree55e1d4bd5bdbd65418e4d6f5822bd3c1c1fc1e32 /frontend/src/pages/Import.tsx
parent943014fd38a3d784542f78cd4625d1ef2e220980 (diff)
implement generic score viewer and import deduplication
Diffstat (limited to 'frontend/src/pages/Import.tsx')
-rw-r--r--frontend/src/pages/Import.tsx200
1 files changed, 98 insertions, 102 deletions
diff --git a/frontend/src/pages/Import.tsx b/frontend/src/pages/Import.tsx
index efd1d03..fe2501f 100644
--- a/frontend/src/pages/Import.tsx
+++ b/frontend/src/pages/Import.tsx
@@ -1,40 +1,42 @@
-import { useState, useEffect } from 'react';
-import { Link, useNavigate } from 'react-router';
-import { useAuth } from '../contexts/AuthContext';
-import JsonUploadModal from '../components/modals/JsonUploadModal';
-import EamusementModal from '../components/modals/EamusementModal';
-import type { SupportedGame } from '../types/game';
-import { uploadScore } from '../utils/scoreUpload';
-
-
+import { useState, useEffect } from "react";
+import { useNavigate } from "react-router";
+import { useAuth } from "../contexts/AuthContext";
+import JsonUploadModal from "../components/modals/JsonUploadModal";
+import EamusementModal from "../components/modals/EamusementModal";
+import SessionExpiredPopup from "../components/SessionExpiredPopup";
+import type { SupportedGame } from "../types/game";
+import { uploadScore } from "../utils/scoreUpload";
+import { NavBar } from "../components/NavBar";
const Import = () => {
const { user, isLoading, logout } = useAuth();
const navigate = useNavigate();
- const [selectedGame, setSelectedGame] = useState('');
+ const [selectedGame, setSelectedGame] = useState("");
const [isJsonModalOpen, setIsJsonModalOpen] = useState(false);
const [isEamusementModalOpen, setIsEamusementModalOpen] = useState(false);
const [supportedGames, setSupportedGames] = useState<SupportedGame[]>([]);
const [gamesLoading, setGamesLoading] = useState(true);
const [uploadStatus, setUploadStatus] = useState<{
- type: 'success' | 'error' | null;
+ type: "success" | "error" | null;
message: string;
- }>({ type: null, message: '' });
+ }>({ type: null, message: "" });
useEffect(() => {
const fetchSupportedGames = async () => {
try {
- const response = await fetch(import.meta.env.VITE_API_URL+'/supportedGames');
+ const response = await fetch(
+ import.meta.env.VITE_API_URL + "/supportedGames",
+ );
if (!response.ok) {
- throw new Error('Failed to fetch supported games');
+ throw new Error("Failed to fetch supported games");
}
const data = await response.json();
setSupportedGames(data);
} catch (error) {
- console.error('Failed to fetch supported games:', error);
+ console.error("Failed to fetch supported games:", error);
setUploadStatus({
- type: 'error',
- message: 'Failed to load supported games. Please refresh the page.'
+ type: "error",
+ message: "Failed to load supported games. Please refresh the page.",
});
} finally {
setGamesLoading(false);
@@ -47,10 +49,10 @@ const Import = () => {
const handleLogout = async () => {
try {
await logout();
- navigate('/');
+ navigate("/");
} catch (error) {
- console.error('Logout failed:', error);
- alert('Network error during logout. Please try again.');
+ console.error("Logout failed:", error);
+ alert("Network error during logout. Please try again.");
}
};
@@ -58,30 +60,33 @@ const Import = () => {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const handleJsonUpload = async (data: any) => {
try {
- console.log('Uploading data for game:', selectedGame, data);
+ console.log("Uploading data for game:", selectedGame, data);
const result = await uploadScore({
meta: {
game: data.meta.game,
service: data.meta.service,
- playtype: data.meta.playtype
+ playtype: data.meta.playtype,
},
- scores: data.scores
+ scores: data.scores,
});
setUploadStatus({
- type: 'success',
- message: `Successfully imported ${result.scoreCount} score(s) for ${supportedGames.find(g => g.internalName === data.meta.game)?.formattedName || data.meta.game}`
+ type: "success",
+ message: `Successfully imported ${result.scoreCount} score(s) for ${supportedGames.find((g) => g.internalName === data.meta.game)?.formattedName || data.meta.game}`,
});
setTimeout(() => {
- setUploadStatus({ type: null, message: '' });
+ setUploadStatus({ type: null, message: "" });
}, 5000);
} catch (error) {
- console.error('Upload failed:', error);
+ console.error("Upload failed:", error);
setUploadStatus({
- type: 'error',
- message: error instanceof Error ? error.message : 'Failed to import data. Please try again.'
+ type: "error",
+ message:
+ error instanceof Error
+ ? error.message
+ : "Failed to import data. Please try again.",
});
}
};
@@ -89,8 +94,18 @@ const Import = () => {
const JsonUploadCard = () => (
<div className="bg-slate-800 rounded-lg border border-slate-700 p-6 hover:border-violet-500 transition-colors">
<div className="w-12 h-12 bg-violet-600/20 rounded-lg flex items-center justify-center mb-4">
- <svg className="w-6 h-6 text-violet-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
- <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M7 16a4 4 0 01-.88-7.903A5 5 0 1115.9 6L16 6a5 5 0 011 9.9M15 13l-3-3m0 0l-3 3m3-3v12" />
+ <svg
+ className="w-6 h-6 text-violet-400"
+ fill="none"
+ stroke="currentColor"
+ viewBox="0 0 24 24"
+ >
+ <path
+ strokeLinecap="round"
+ strokeLinejoin="round"
+ strokeWidth={2}
+ d="M7 16a4 4 0 01-.88-7.903A5 5 0 1115.9 6L16 6a5 5 0 011 9.9M15 13l-3-3m0 0l-3 3m3-3v12"
+ />
</svg>
</div>
<h4 className="text-white font-semibold mb-2">Batch-Manual Upload</h4>
@@ -111,11 +126,23 @@ const Import = () => {
{/* e-amusement Card */}
<div className="bg-slate-800 rounded-lg border border-slate-700 p-6 hover:border-violet-500 transition-colors">
<div className="w-12 h-12 bg-blue-600/20 rounded-lg flex items-center justify-center mb-4">
- <svg className="w-6 h-6 text-blue-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
- <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M10 6H6a2 2 0 00-2 2v10a2 2 0 002 2h10a2 2 0 002-2v-4M14 4h6m0 0v6m0-6L10 14" />
+ <svg
+ className="w-6 h-6 text-blue-400"
+ fill="none"
+ stroke="currentColor"
+ viewBox="0 0 24 24"
+ >
+ <path
+ strokeLinecap="round"
+ strokeLinejoin="round"
+ strokeWidth={2}
+ d="M10 6H6a2 2 0 00-2 2v10a2 2 0 002 2h10a2 2 0 002-2v-4M14 4h6m0 0v6m0-6L10 14"
+ />
</svg>
</div>
- <h4 className="text-white font-semibold mb-2">e-amusement Play History</h4>
+ <h4 className="text-white font-semibold mb-2">
+ e-amusement Play History
+ </h4>
<p className="text-slate-400 text-sm mb-4">
Import via scraping your playdata from KONAMI e-amusement
</p>
@@ -131,12 +158,12 @@ const Import = () => {
const renderImportOptions = () => {
switch (selectedGame) {
- case 'dancerush':
+ case "dancerush":
return (
<>
{/* JSON Upload Card */}
<JsonUploadCard />
- <EamusementScrapeUploadCard/>
+ <EamusementScrapeUploadCard />
</>
);
@@ -157,83 +184,40 @@ const Import = () => {
}
if (!user) {
- return (
- <div className="min-h-screen bg-slate-950 flex items-center justify-center">
- <div className="text-center max-w-md mx-auto px-6">
- <div className="bg-slate-900 rounded-lg p-8 border border-slate-700">
- <h2 className="text-2xl font-bold text-white mb-4">Session Expired</h2>
- <p className="text-slate-300 mb-6">Please sign in to import your data.</p>
- <div className="space-y-3">
- <Link
- to="/login"
- className="block w-full bg-violet-600 hover:bg-violet-700 text-white py-3 rounded-md font-medium transition-colors"
- >
- Sign In
- </Link>
- <Link
- to="/"
- className="block w-full border border-slate-600 hover:border-violet-500 text-slate-300 hover:text-white py-3 rounded-md font-medium transition-colors"
- >
- Back to Home
- </Link>
- </div>
- </div>
- </div>
- </div>
- );
+ return <SessionExpiredPopup />;
}
return (
<div className="min-h-screen bg-slate-950">
{/* Navigation */}
- <nav className="border-b border-slate-800 bg-slate-950/95 backdrop-blur-sm sticky top-0 z-50">
- <div className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
- <div className="flex items-center justify-between h-16">
- <div className="flex items-center space-x-3">
- <Link to="/home" className="flex items-center space-x-3 group">
- <div className="w-8 h-8 bg-violet-600 rounded-md flex items-center justify-center group-hover:bg-violet-700 transition-colors">
- <span className="text-white font-bold text-sm">M</span>
- </div>
- <span className="text-white font-semibold text-lg">Mirage</span>
- </Link>
- </div>
- <div className="flex items-center space-x-4">
- <Link
- to="/home"
- className="text-slate-300 hover:text-white px-3 py-2 rounded-md text-sm font-medium transition-colors"
- >
- Dashboard
- </Link>
- <span className="text-slate-300 text-sm">Welcome back, {user.username}</span>
- <button
- onClick={handleLogout}
- className="text-slate-300 hover:text-white px-3 py-2 rounded-md text-sm font-medium transition-colors"
- >
- Sign Out
- </button>
- </div>
- </div>
- </div>
- </nav>
+ <NavBar user={user} handleLogout={handleLogout} currentPage="import"/>
{/* Main Content */}
<div className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
{/* Header */}
<div className="mb-8">
<h1 className="text-3xl font-bold text-white mb-2">Import Data</h1>
- <p className="text-slate-400">Import your game scores and progress from various sources</p>
+ <p className="text-slate-400">
+ Import your game scores and progress from various sources
+ </p>
</div>
{/* Status Message */}
{uploadStatus.type && (
- <div className={`mb-6 rounded-md p-4 ${
- uploadStatus.type === 'success'
- ? 'bg-green-500/10 border border-green-500/20'
- : 'bg-red-500/10 border border-red-500/20'
- }`}>
- <p className={`text-sm ${
- uploadStatus.type === 'success' ? 'text-green-400' : 'text-red-400'
- }`}>
+ <div
+ className={`mb-6 rounded-md p-4 ${
+ uploadStatus.type === "success"
+ ? "bg-green-500/10 border border-green-500/20"
+ : "bg-red-500/10 border border-red-500/20"
+ }`}
+ >
+ <p
+ className={`text-sm ${
+ uploadStatus.type === "success"
+ ? "text-green-400"
+ : "text-red-400"
+ }`}
+ >
{uploadStatus.message}
</p>
</div>
@@ -262,7 +246,11 @@ const Import = () => {
>
<option value="">Select a game</option>
{supportedGames.map((game) => (
- <option key={game.internalName} value={game.internalName} title={game.description}>
+ <option
+ key={game.internalName}
+ value={game.internalName}
+ title={game.description}
+ >
{game.formattedName}
</option>
))}
@@ -273,7 +261,9 @@ const Import = () => {
{/* Import Options */}
{selectedGame && (
<div className="space-y-6 mt-8">
- <h3 className="text-lg font-semibold text-white">Import Options</h3>
+ <h3 className="text-lg font-semibold text-white">
+ Import Options
+ </h3>
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
{renderImportOptions()}
@@ -288,14 +278,20 @@ const Import = () => {
isOpen={isJsonModalOpen}
onClose={() => setIsJsonModalOpen(false)}
onUpload={handleJsonUpload}
- game={supportedGames.find(g => g.internalName === selectedGame)?.formattedName || ''}
+ game={
+ supportedGames.find((g) => g.internalName === selectedGame)
+ ?.formattedName || ""
+ }
/>
{/* Eamusement Modal */}
<EamusementModal
isOpen={isEamusementModalOpen}
onClose={() => setIsEamusementModalOpen(false)}
- game={supportedGames.find(g => g.internalName === selectedGame) || undefined}
+ game={
+ supportedGames.find((g) => g.internalName === selectedGame) ||
+ undefined
+ }
/>
</div>
);
send patches to the email below
yukais@pinapelz.com
include the subject [PATCH repo_name]
pinapelz.com
homepage