mirror of
https://github.com/pezkuwichain/pwap.git
synced 2026-04-22 17:07:57 +00:00
199 lines
7.5 KiB
TypeScript
199 lines
7.5 KiB
TypeScript
import React, { useState, useEffect } from 'react';
|
|
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@/components/ui/card';
|
|
import { Progress } from '@/components/ui/progress';
|
|
import { Badge } from '@/components/ui/badge';
|
|
import { Button } from '@/components/ui/button';
|
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs';
|
|
import { useTranslation } from 'react-i18next';
|
|
import {
|
|
DollarSign,
|
|
TrendingUp,
|
|
TrendingDown,
|
|
PieChart,
|
|
Activity,
|
|
AlertCircle,
|
|
CheckCircle,
|
|
Clock,
|
|
ArrowUpRight,
|
|
ArrowDownRight
|
|
} from 'lucide-react';
|
|
|
|
interface TreasuryMetrics {
|
|
totalBalance: number;
|
|
monthlyIncome: number;
|
|
monthlyExpenses: number;
|
|
pendingProposals: number;
|
|
approvedBudget: number;
|
|
healthScore: number;
|
|
}
|
|
|
|
interface BudgetCategory {
|
|
id: string;
|
|
name: string;
|
|
allocated: number;
|
|
spent: number;
|
|
remaining: number;
|
|
color: string;
|
|
}
|
|
|
|
export const TreasuryOverview: React.FC = () => {
|
|
const { t } = useTranslation();
|
|
const [metrics, setMetrics] = useState<TreasuryMetrics>({
|
|
totalBalance: 2500000,
|
|
monthlyIncome: 150000,
|
|
monthlyExpenses: 120000,
|
|
pendingProposals: 8,
|
|
approvedBudget: 1800000,
|
|
healthScore: 85
|
|
});
|
|
|
|
const [categories] = useState<BudgetCategory[]>([
|
|
{ id: '1', name: 'Development', allocated: 500000, spent: 320000, remaining: 180000, color: 'bg-blue-500' },
|
|
{ id: '2', name: 'Marketing', allocated: 200000, spent: 150000, remaining: 50000, color: 'bg-purple-500' },
|
|
{ id: '3', name: 'Operations', allocated: 300000, spent: 180000, remaining: 120000, color: 'bg-green-500' },
|
|
{ id: '4', name: 'Community', allocated: 150000, spent: 80000, remaining: 70000, color: 'bg-yellow-500' },
|
|
{ id: '5', name: 'Research', allocated: 250000, spent: 100000, remaining: 150000, color: 'bg-pink-500' },
|
|
{ id: '6', name: 'Infrastructure', allocated: 400000, spent: 350000, remaining: 50000, color: 'bg-indigo-500' }
|
|
]);
|
|
|
|
const getHealthStatus = (score: number) => {
|
|
if (score >= 80) return { label: 'Excellent', color: 'text-green-500', icon: CheckCircle };
|
|
if (score >= 60) return { label: 'Good', color: 'text-blue-500', icon: Activity };
|
|
if (score >= 40) return { label: 'Fair', color: 'text-yellow-500', icon: AlertCircle };
|
|
return { label: 'Critical', color: 'text-red-500', icon: AlertCircle };
|
|
};
|
|
|
|
const healthStatus = getHealthStatus(metrics.healthScore);
|
|
const HealthIcon = healthStatus.icon;
|
|
|
|
return (
|
|
<div className="space-y-6">
|
|
{/* Treasury Health Score */}
|
|
<Card>
|
|
<CardHeader>
|
|
<CardTitle className="flex items-center justify-between">
|
|
<span>Treasury Health</span>
|
|
<HealthIcon className={`h-6 w-6 ${healthStatus.color}`} />
|
|
</CardTitle>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<div className="space-y-4">
|
|
<div className="flex items-center justify-between">
|
|
<span className="text-2xl font-bold">{metrics.healthScore}%</span>
|
|
<Badge className={healthStatus.color}>{healthStatus.label}</Badge>
|
|
</div>
|
|
<Progress value={metrics.healthScore} className="h-3" />
|
|
<div className="grid grid-cols-2 gap-4 text-sm">
|
|
<div>
|
|
<p className="text-muted-foreground">Runway</p>
|
|
<p className="font-semibold">20.8 months</p>
|
|
</div>
|
|
<div>
|
|
<p className="text-muted-foreground">Burn Rate</p>
|
|
<p className="font-semibold">$120k/month</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
|
|
{/* Key Metrics */}
|
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
|
|
<Card>
|
|
<CardContent className="p-6">
|
|
<div className="flex items-center justify-between">
|
|
<div>
|
|
<p className="text-sm text-muted-foreground">Total Balance</p>
|
|
<p className="text-2xl font-bold">${(metrics.totalBalance / 1000000).toFixed(2)}M</p>
|
|
<p className="text-xs text-green-500 flex items-center mt-1">
|
|
<ArrowUpRight className="h-3 w-3 mr-1" />
|
|
+12.5% this month
|
|
</p>
|
|
</div>
|
|
<DollarSign className="h-8 w-8 text-green-500" />
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
|
|
<Card>
|
|
<CardContent className="p-6">
|
|
<div className="flex items-center justify-between">
|
|
<div>
|
|
<p className="text-sm text-muted-foreground">Monthly Income</p>
|
|
<p className="text-2xl font-bold">${(metrics.monthlyIncome / 1000).toFixed(0)}k</p>
|
|
<p className="text-xs text-green-500 flex items-center mt-1">
|
|
<TrendingUp className="h-3 w-3 mr-1" />
|
|
+8.3% vs last month
|
|
</p>
|
|
</div>
|
|
<TrendingUp className="h-8 w-8 text-blue-500" />
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
|
|
<Card>
|
|
<CardContent className="p-6">
|
|
<div className="flex items-center justify-between">
|
|
<div>
|
|
<p className="text-sm text-muted-foreground">Monthly Expenses</p>
|
|
<p className="text-2xl font-bold">${(metrics.monthlyExpenses / 1000).toFixed(0)}k</p>
|
|
<p className="text-xs text-red-500 flex items-center mt-1">
|
|
<ArrowDownRight className="h-3 w-3 mr-1" />
|
|
-5.2% vs last month
|
|
</p>
|
|
</div>
|
|
<TrendingDown className="h-8 w-8 text-red-500" />
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
|
|
<Card>
|
|
<CardContent className="p-6">
|
|
<div className="flex items-center justify-between">
|
|
<div>
|
|
<p className="text-sm text-muted-foreground">Pending Proposals</p>
|
|
<p className="text-2xl font-bold">{metrics.pendingProposals}</p>
|
|
<p className="text-xs text-yellow-500 flex items-center mt-1">
|
|
<Clock className="h-3 w-3 mr-1" />
|
|
$450k requested
|
|
</p>
|
|
</div>
|
|
<Clock className="h-8 w-8 text-yellow-500" />
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
</div>
|
|
|
|
{/* Budget Categories */}
|
|
<Card>
|
|
<CardHeader>
|
|
<CardTitle>Budget Allocation by Category</CardTitle>
|
|
<CardDescription>Current quarter budget utilization</CardDescription>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<div className="space-y-4">
|
|
{categories.map((category) => {
|
|
const utilization = (category.spent / category.allocated) * 100;
|
|
return (
|
|
<div key={category.id} className="space-y-2">
|
|
<div className="flex items-center justify-between text-sm">
|
|
<span className="font-medium">{category.name}</span>
|
|
<div className="flex items-center gap-4">
|
|
<span className="text-muted-foreground">
|
|
${(category.spent / 1000).toFixed(0)}k / ${(category.allocated / 1000).toFixed(0)}k
|
|
</span>
|
|
<Badge variant={utilization > 80 ? 'destructive' : 'secondary'}>
|
|
{utilization.toFixed(0)}%
|
|
</Badge>
|
|
</div>
|
|
</div>
|
|
<Progress value={utilization} className="h-2" />
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
</div>
|
|
);
|
|
}; |