Files
MOPC-Portal/src/components/charts/diversity-metrics.tsx
Matt 161cd1684a
All checks were successful
Build and Push Docker Image / build (push) Successful in 9m32s
Fix observer reports: charts, filtering, project preview, dashboard stats
- Rewrite diversity metrics: horizontal bar charts for ocean issues and
  geographic distribution (replaces unreadable vertical/donut charts)
- Rewrite juror score heatmap: expandable table with score distribution
- Rewrite juror consistency: horizontal bar visual with juror names
- Merge filtering tabs into single screening view with per-project
  AI reasoning and expandable rows
- Add project preview dialog for juror performance table
- Fix status breakdown for evaluation rounds (Fully/Partially/Not Reviewed)
- Show active round name instead of count on observer dashboard
- Move Global tab to last position, default to first round-specific tab
- Add 4-card stats layout for evaluation with reviews/project ratio
- Fix oceanIssue field (singular) and remove non-existent aiSummary

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-21 10:12:21 +01:00

214 lines
7.5 KiB
TypeScript

'use client'
import { BarChart } from '@tremor/react'
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card'
import { Badge } from '@/components/ui/badge'
interface DiversityData {
total: number
byCountry: { country: string; count: number; percentage: number }[]
byCategory: { category: string; count: number; percentage: number }[]
byOceanIssue: { issue: string; count: number; percentage: number }[]
byTag: { tag: string; count: number; percentage: number }[]
}
interface DiversityMetricsProps {
data: DiversityData
}
/** Convert ISO 3166-1 alpha-2 code to full country name using Intl API */
function getCountryName(code: string): string {
if (code === 'Others') return 'Others'
try {
const displayNames = new Intl.DisplayNames(['en'], { type: 'region' })
return displayNames.of(code.toUpperCase()) || code
} catch {
return code
}
}
/** Convert SCREAMING_SNAKE_CASE to Title Case */
function formatLabel(value: string): string {
if (!value) return value
return value
.replace(/_/g, ' ')
.toLowerCase()
.replace(/\b\w/g, (c) => c.toUpperCase())
}
export function DiversityMetricsChart({ data }: DiversityMetricsProps) {
if (!data || data.total === 0) {
return (
<Card>
<CardContent className="flex items-center justify-center py-12">
<p className="text-muted-foreground">No project data available</p>
</CardContent>
</Card>
)
}
// Top countries — horizontal bar chart for readability
const countryBarData = (data.byCountry || []).slice(0, 15).map((c) => ({
country: getCountryName(c.country),
Projects: c.count,
}))
const categoryData = (data.byCategory || []).slice(0, 10).map((c) => ({
category: formatLabel(c.category),
Projects: c.count,
}))
const oceanIssueData = (data.byOceanIssue || []).slice(0, 15).map((o) => ({
issue: formatLabel(o.issue),
Projects: o.count,
}))
return (
<div className="space-y-6">
{/* Summary stats row */}
<div className="grid grid-cols-2 gap-3 sm:grid-cols-4">
<Card>
<CardContent className="p-4">
<p className="text-2xl font-bold tabular-nums">{data.total}</p>
<p className="text-xs text-muted-foreground">Total Projects</p>
</CardContent>
</Card>
<Card>
<CardContent className="p-4">
<p className="text-2xl font-bold tabular-nums">{(data.byCountry || []).length}</p>
<p className="text-xs text-muted-foreground">Countries</p>
</CardContent>
</Card>
<Card>
<CardContent className="p-4">
<p className="text-2xl font-bold tabular-nums">{(data.byCategory || []).length}</p>
<p className="text-xs text-muted-foreground">Categories</p>
</CardContent>
</Card>
<Card>
<CardContent className="p-4">
<p className="text-2xl font-bold tabular-nums">{(data.byOceanIssue || []).length}</p>
<p className="text-xs text-muted-foreground">Ocean Issues</p>
</CardContent>
</Card>
</div>
<div className="grid gap-6 lg:grid-cols-2">
{/* Country Distribution — horizontal bars */}
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-base">Geographic Distribution</CardTitle>
</CardHeader>
<CardContent>
{countryBarData.length > 0 ? (
<BarChart
data={countryBarData}
index="country"
categories={['Projects']}
colors={['cyan']}
showLegend={false}
layout="horizontal"
yAxisWidth={120}
className="h-[360px]"
/>
) : (
<p className="text-muted-foreground text-center py-8">No geographic data</p>
)}
</CardContent>
</Card>
{/* Competition Categories — horizontal bars */}
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-base">Competition Categories</CardTitle>
</CardHeader>
<CardContent>
{categoryData.length > 0 ? (
categoryData.length <= 4 ? (
/* Clean stacked bars for few categories */
<div className="space-y-4 pt-2">
{categoryData.map((c) => {
const maxCount = Math.max(...categoryData.map((d) => d.Projects))
const pct = maxCount > 0 ? (c.Projects / maxCount) * 100 : 0
return (
<div key={c.category} className="space-y-1.5">
<div className="flex items-center justify-between text-sm">
<span className="font-medium">{c.category}</span>
<span className="tabular-nums text-muted-foreground">{c.Projects}</span>
</div>
<div className="h-3 w-full rounded-full bg-muted overflow-hidden">
<div
className="h-full rounded-full bg-[#053d57] transition-all duration-500"
style={{ width: `${pct}%` }}
/>
</div>
</div>
)
})}
</div>
) : (
<BarChart
data={categoryData}
index="category"
categories={['Projects']}
colors={['indigo']}
layout="horizontal"
yAxisWidth={140}
showLegend={false}
className="h-[280px]"
/>
)
) : (
<p className="text-muted-foreground text-center py-8">No category data</p>
)}
</CardContent>
</Card>
</div>
{/* Ocean Issues — horizontal bars for readability */}
{oceanIssueData.length > 0 && (
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-base">Ocean Issues Addressed</CardTitle>
</CardHeader>
<CardContent>
<BarChart
data={oceanIssueData}
index="issue"
categories={['Projects']}
colors={['blue']}
showLegend={false}
layout="horizontal"
yAxisWidth={200}
className="h-[400px]"
/>
</CardContent>
</Card>
)}
{/* Tags — clean pill cloud */}
{(data.byTag || []).length > 0 && (
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-base">Project Tags</CardTitle>
</CardHeader>
<CardContent>
<div className="flex flex-wrap gap-2">
{(data.byTag || []).slice(0, 30).map((tag) => (
<Badge
key={tag.tag}
variant="outline"
className="px-3 py-1 text-sm font-normal"
>
{tag.tag}
<span className="ml-1.5 text-muted-foreground tabular-nums">({tag.count})</span>
</Badge>
))}
</div>
</CardContent>
</Card>
)}
</div>
)
}