This commit is contained in:
5t4l1n
2025-07-25 23:58:20 +05:30
parent b70e81d201
commit 4d8061616d
112 changed files with 14445 additions and 59 deletions
+131
View File
@@ -0,0 +1,131 @@
"use client"
import { useState, useEffect } from "react"
import { useAuth } from "@/context/auth-context"
import { useRouter } from "next/navigation"
import { toast } from "react-hot-toast"
import type { CodingProblem } from "@/lib/types"
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"
import { Badge } from "@/components/ui/badge"
import { Button } from "@/components/ui/button"
import Link from "next/link"
import { Loader2, CheckCircle2 } from "lucide-react"
import { api } from "@/lib/api" // Import api
export function CodingProblemList() {
const { user, firebaseUser, isLoadingAuth, authMethod, token } = useAuth() // Check token for access
const router = useRouter()
const [problems, setProblems] = useState<CodingProblem[]>([])
const [isLoadingProblems, setIsLoadingProblems] = useState(true)
const [error, setError] = useState<string | null>(null)
useEffect(() => {
if (!isLoadingAuth && !user && !firebaseUser) {
// Allow either MetaMask or Firebase user
toast.error("Please login to view coding problems.")
router.push("/")
return
}
const fetchProblems = async () => {
setIsLoadingProblems(true)
setError(null)
try {
// --- ORIGINAL API CALL (UNCOMMENT WHEN BACKEND IS READY) ---
const response = await api.get<CodingProblem[]>("/api/coding/problems")
setProblems(response.data)
// --- ORIGINAL API CALL (UNCOMMENT WHEN BACKEND IS READY) ---
} catch (err: any) {
console.error("Failed to fetch coding problems:", err)
setError(err.response?.data?.message || "Failed to load coding problems.")
toast.error(err.response?.data?.message || "Failed to load coding problems.")
} finally {
setIsLoadingProblems(false)
}
}
if (user || firebaseUser) {
// Only fetch if either user type is logged in
fetchProblems()
}
}, [user, firebaseUser, isLoadingAuth, router, token])
const getDifficultyColor = (difficulty: CodingProblem["difficulty"]) => {
switch (difficulty) {
case "Easy":
return "bg-success text-white"
case "Medium":
return "bg-warning text-white"
case "Hard":
return "bg-destructive text-white"
default:
return "bg-gray-500 text-white"
}
}
if (isLoadingAuth || isLoadingProblems) {
return (
<div className="flex justify-center items-center min-h-[calc(100vh-64px)]">
<Loader2 className="h-8 w-8 animate-spin text-primary-purple" />
<span className="ml-2 text-lg">Loading coding problems...</span>
</div>
)
}
if (error) {
return (
<div className="flex justify-center items-center min-h-[calc(100vh-64px)] text-red-500">
<p>{error}</p>
</div>
)
}
if (problems.length === 0) {
return (
<div className="flex flex-col items-center justify-center min-h-[calc(100vh-64px)] text-gray-600 dark:text-gray-300">
<p className="text-xl mb-4">No coding problems available yet.</p>
<p>Check back later for new challenges!</p>
</div>
)
}
return (
<div className="container mx-auto py-8 px-4">
{authMethod === "firebase" && !token && (
<div className="bg-yellow-100 border-l-4 border-yellow-500 text-yellow-700 p-4 mb-6 rounded-md dark:bg-yellow-900 dark:border-yellow-600 dark:text-yellow-200">
<p className="font-bold">Limited Access</p>
<p>
You are logged in with email. Full functionality, including progress tracking and data persistence, requires
connecting your MetaMask wallet.
</p>
</div>
)}
<h1 className="text-3xl font-bold text-primary-purple mb-8 text-center">Coding Practice</h1>
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
{problems.map((problem) => (
<Card
key={problem.id}
className="bg-white shadow-md rounded-lg overflow-hidden dark:bg-gray-800 dark:text-gray-100"
>
<CardHeader>
<div className="flex items-center justify-between mb-2">
<CardTitle className="text-xl font-semibold">{problem.title}</CardTitle>
{problem.solved && <CheckCircle2 className="h-6 w-6 text-success" />}
</div>
<CardDescription className="flex items-center gap-2">
<Badge className={getDifficultyColor(problem.difficulty)}>{problem.difficulty}</Badge>
<Badge variant="secondary">{problem.category}</Badge>
</CardDescription>
</CardHeader>
<CardContent className="space-y-4">
<p className="text-gray-700 dark:text-gray-200 line-clamp-3">{problem.description}</p>
<Link href={`/coding/${problem.id}`}>
<Button className="w-full bg-primary-blue hover:bg-primary-blue/90 text-white">Solve Problem</Button>
</Link>
</CardContent>
</Card>
))}
</div>
</div>
)
}