project-standalo-todo-super/app/components/TransactionHistory.tsx

147 lines
5.8 KiB
TypeScript

"use client";
import { useEffect, useState } from "react";
interface Transaction {
id: string;
type: "earned" | "spent" | "bonus" | "penalty";
description: string;
amount: number;
timestamp: string;
}
const transactionIcons = {
earned: (
<svg className="w-5 h-5 text-green-400" fill="currentColor" viewBox="0 0 20 20">
<path fillRule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z" clipRule="evenodd" />
</svg>
),
spent: (
<svg className="w-5 h-5 text-red-400" fill="currentColor" viewBox="0 0 20 20">
<path fillRule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z" clipRule="evenodd" />
</svg>
),
bonus: (
<svg className="w-5 h-5 text-yellow-400" fill="currentColor" viewBox="0 0 20 20">
<path d="M9.049 2.927c.3-.921 1.603-.921 1.902 0l1.07 3.292a1 1 0 00.95.69h3.462c.969 0 1.371 1.24.588 1.81l-2.8 2.034a1 1 0 00-.364 1.118l1.07 3.292c.3.921-.755 1.688-1.54 1.118l-2.8-2.034a1 1 0 00-1.175 0l-2.8 2.034c-.784.57-1.838-.197-1.539-1.118l1.07-3.292a1 1 0 00-.364-1.118L2.98 8.72c-.783-.57-.38-1.81.588-1.81h3.461a1 1 0 00.951-.69l1.07-3.292z" />
</svg>
),
penalty: (
<svg className="w-5 h-5 text-orange-400" fill="currentColor" viewBox="0 0 20 20">
<path fillRule="evenodd" d="M8.257 3.099c.765-1.36 2.722-1.36 3.486 0l5.58 9.92c.75 1.334-.213 2.98-1.742 2.98H4.42c-1.53 0-2.493-1.646-1.743-2.98l5.58-9.92zM11 13a1 1 0 11-2 0 1 1 0 012 0zm-1-8a1 1 0 00-1 1v3a1 1 0 002 0V6a1 1 0 00-1-1z" clipRule="evenodd" />
</svg>
),
};
export default function TransactionHistory() {
const [transactions, setTransactions] = useState<Transaction[]>([]);
const [loading, setLoading] = useState(true);
const [filter, setFilter] = useState<"all" | Transaction["type"]>("all");
useEffect(() => {
fetchTransactions();
}, []);
const fetchTransactions = async () => {
try {
const token = localStorage.getItem("token");
const response = await fetch("/api/users/me/points", {
headers: { Authorization: `Bearer ${token}` },
});
const data = await response.json();
if (data.success && data.data) {
setTransactions(data.data.transactions || []);
}
} catch (error) {
console.error("Failed to fetch transactions:", error);
} finally {
setLoading(false);
}
};
const filteredTransactions = filter === "all"
? transactions
: transactions.filter(t => t.type === filter);
const formatDate = (timestamp: string) => {
const date = new Date(timestamp);
const now = new Date();
const diffMs = now.getTime() - date.getTime();
const diffMins = Math.floor(diffMs / 60000);
const diffHours = Math.floor(diffMs / 3600000);
const diffDays = Math.floor(diffMs / 86400000);
if (diffMins < 1) return "Just now";
if (diffMins < 60) return `${diffMins}m ago`;
if (diffHours < 24) return `${diffHours}h ago`;
if (diffDays < 7) return `${diffDays}d ago`;
return date.toLocaleDateString();
};
if (loading) {
return (
<div className="flex items-center justify-center py-12">
<div className="animate-spin rounded-full h-12 w-12 border-b-2 border-yellow-500"></div>
</div>
);
}
return (
<div className="space-y-6">
<div className="flex gap-2 flex-wrap">
{(["all", "earned", "spent", "bonus", "penalty"] as const).map(type => (
<button
key={type}
onClick={() => setFilter(type)}
className={`px-4 py-2 rounded-lg font-medium transition capitalize ${
filter === type
? "bg-gradient-to-r from-yellow-500 to-orange-500 text-white"
: "bg-gray-800 text-gray-300 hover:bg-gray-700"
}`}
>
{type}
</button>
))}
</div>
<div className="bg-gray-800 border border-gray-700 rounded-xl divide-y divide-gray-700">
{filteredTransactions.length === 0 ? (
<div className="text-center py-12">
<div className="text-gray-400 mb-2">
<svg className="w-16 h-16 mx-auto opacity-50" fill="currentColor" viewBox="0 0 20 20">
<path fillRule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zM7 9a1 1 0 000 2h6a1 1 0 100-2H7z" clipRule="evenodd" />
</svg>
</div>
<p className="text-gray-400">No transactions found</p>
</div>
) : (
filteredTransactions.map(transaction => (
<div key={transaction.id} className="p-4 hover:bg-gray-750 transition">
<div className="flex items-center justify-between">
<div className="flex items-center gap-3">
<div className={`p-2 rounded-lg ${
transaction.type === "earned" || transaction.type === "bonus"
? "bg-green-500/20"
: "bg-red-500/20"
}`}>
{transactionIcons[transaction.type]}
</div>
<div>
<p className="text-white font-medium">{transaction.description}</p>
<p className="text-gray-400 text-sm">{formatDate(transaction.timestamp)}</p>
</div>
</div>
<div className={`text-xl font-bold ${
transaction.amount > 0 ? "text-green-400" : "text-red-400"
}`}>
{transaction.amount > 0 ? "+" : ""}{transaction.amount.toLocaleString()}
</div>
</div>
</div>
))
)}
</div>
</div>
);
}