Files
breakpilot-compliance/admin-compliance/app/sdk/cookie-banner/_components/AnalyticsDashboard.tsx
T
Benjamin Admin c3fcfe88ee feat: Vendor-level consent + Consent analytics (F4 + F6)
F4: Granular Vendor-Level Consent
- Migration 113: vendor_consents JSONB on banner_consents + audit_log
- ConsentCreate schema + BannerConsentDB model extended
- banner_consent_service stores vendor_consents alongside categories
- Audit trail includes vendor-level decisions + user_agent

F6: Consent Rate Analytics
- Migration 114: user_agent on audit_log + time-series index
- BannerAnalyticsService: time series, category breakdown, device stats
- banner_analytics_routes: 4 endpoints (overview, time-series, categories, devices)
- AnalyticsDashboard.tsx: KPIs, bar chart, category bars, device breakdown
- New "Analytik" tab in cookie-banner page

[migration-approved]

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-05-03 20:58:06 +02:00

192 lines
7.6 KiB
TypeScript

'use client'
import { useState, useEffect } from 'react'
interface TimeSeriesPoint {
period: string
given: number
updated: number
withdrawn: number
total: number
opt_in_rate: number
}
interface CategoryStats {
[key: string]: { count: number; total: number; rate: number }
}
interface DeviceStats {
desktop: number
mobile: number
tablet: number
unknown: number
}
interface OverviewStats {
period_days: number
total_interactions: number
consents_given: number
consents_updated: number
consents_withdrawn: number
opt_in_rate: number
}
const PERIODS = [
{ value: 7, label: '7 Tage' },
{ value: 30, label: '30 Tage' },
{ value: 90, label: '90 Tage' },
]
const CAT_COLORS: Record<string, string> = {
necessary: '#22c55e',
statistics: '#eab308',
marketing: '#ef4444',
functional: '#3b82f6',
preferences: '#8b5cf6',
}
export function AnalyticsDashboard({ siteId }: { siteId?: string }) {
const [days, setDays] = useState(30)
const [overview, setOverview] = useState<OverviewStats | null>(null)
const [timeSeries, setTimeSeries] = useState<TimeSeriesPoint[]>([])
const [categories, setCategories] = useState<CategoryStats>({})
const [devices, setDevices] = useState<DeviceStats>({ desktop: 0, mobile: 0, tablet: 0, unknown: 0 })
const [loading, setLoading] = useState(true)
const sid = siteId || 'preview-test-site'
useEffect(() => {
setLoading(true)
const base = `/api/sdk/v1/compliance/banner/analytics/${sid}`
Promise.all([
fetch(`${base}/overview?days=${days}`).then(r => r.ok ? r.json() : null),
fetch(`${base}/time-series?days=${days}&period=daily`).then(r => r.ok ? r.json() : []),
fetch(`${base}/categories?days=${days}`).then(r => r.ok ? r.json() : {}),
fetch(`${base}/devices?days=${days}`).then(r => r.ok ? r.json() : {}),
]).then(([o, ts, cats, devs]) => {
setOverview(o)
setTimeSeries(ts || [])
setCategories(cats || {})
setDevices(devs || { desktop: 0, mobile: 0, tablet: 0, unknown: 0 })
}).catch(() => {}).finally(() => setLoading(false))
}, [sid, days])
const deviceTotal = devices.desktop + devices.mobile + devices.tablet + devices.unknown
if (loading) return <div className="text-center py-12 text-gray-400">Lade Analytik...</div>
return (
<div className="space-y-6">
{/* Period Selector */}
<div className="flex items-center gap-2">
{PERIODS.map(p => (
<button key={p.value} onClick={() => setDays(p.value)}
className={`px-3 py-1.5 text-xs rounded-full border transition-colors ${
days === p.value ? 'bg-purple-100 border-purple-300 text-purple-700' : 'bg-white border-gray-200 text-gray-600 hover:border-gray-300'
}`}>
{p.label}
</button>
))}
</div>
{/* Overview KPIs */}
{overview && (
<div className="grid grid-cols-2 md:grid-cols-4 gap-4">
<div className="bg-white rounded-xl border border-gray-200 p-4">
<div className="text-xs text-gray-500">Opt-In-Rate</div>
<div className="text-2xl font-bold text-green-600">{overview.opt_in_rate}%</div>
</div>
<div className="bg-white rounded-xl border border-gray-200 p-4">
<div className="text-xs text-gray-500">Einwilligungen</div>
<div className="text-2xl font-bold text-gray-900">{overview.consents_given}</div>
</div>
<div className="bg-white rounded-xl border border-gray-200 p-4">
<div className="text-xs text-gray-500">Aktualisiert</div>
<div className="text-2xl font-bold text-blue-600">{overview.consents_updated}</div>
</div>
<div className="bg-white rounded-xl border border-gray-200 p-4">
<div className="text-xs text-gray-500">Widerrufen</div>
<div className="text-2xl font-bold text-red-600">{overview.consents_withdrawn}</div>
</div>
</div>
)}
{/* Time Series (simple bar visualization) */}
{timeSeries.length > 0 && (
<div className="bg-white rounded-xl border border-gray-200 p-6">
<h3 className="font-semibold text-gray-900 mb-4">Opt-In-Rate im Zeitverlauf</h3>
<div className="flex items-end gap-1 h-32">
{timeSeries.map((pt, i) => {
const height = Math.max(pt.opt_in_rate, 2)
const date = new Date(pt.period)
return (
<div key={i} className="flex-1 flex flex-col items-center gap-1 group relative">
<div className="w-full bg-purple-500 rounded-t transition-all hover:bg-purple-600"
style={{ height: `${height}%` }}
title={`${date.toLocaleDateString('de-DE')}: ${pt.opt_in_rate}% (${pt.total} Interaktionen)`}
/>
{i % Math.max(1, Math.floor(timeSeries.length / 6)) === 0 && (
<span className="text-[8px] text-gray-400">{date.toLocaleDateString('de-DE', { day: '2-digit', month: '2-digit' })}</span>
)}
</div>
)
})}
</div>
</div>
)}
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
{/* Category Acceptance */}
<div className="bg-white rounded-xl border border-gray-200 p-6">
<h3 className="font-semibold text-gray-900 mb-4">Akzeptanz nach Kategorie</h3>
<div className="space-y-3">
{Object.entries(categories).map(([cat, stats]) => (
<div key={cat}>
<div className="flex items-center justify-between text-sm mb-1">
<span className="text-gray-700 capitalize">{cat}</span>
<span className="font-medium text-gray-900">{stats.rate}%</span>
</div>
<div className="h-2 bg-gray-100 rounded-full overflow-hidden">
<div className="h-full rounded-full transition-all" style={{ width: `${stats.rate}%`, backgroundColor: CAT_COLORS[cat] || '#9ca3af' }} />
</div>
</div>
))}
{Object.keys(categories).length === 0 && (
<p className="text-xs text-gray-400">Noch keine Daten vorhanden</p>
)}
</div>
</div>
{/* Device Breakdown */}
<div className="bg-white rounded-xl border border-gray-200 p-6">
<h3 className="font-semibold text-gray-900 mb-4">Geraete-Verteilung</h3>
<div className="space-y-3">
{[
{ key: 'desktop', label: 'Desktop', color: 'bg-blue-500' },
{ key: 'mobile', label: 'Mobile', color: 'bg-green-500' },
{ key: 'tablet', label: 'Tablet', color: 'bg-purple-500' },
].map(d => {
const count = devices[d.key as keyof DeviceStats]
const pct = deviceTotal > 0 ? Math.round(count / deviceTotal * 100) : 0
return (
<div key={d.key}>
<div className="flex items-center justify-between text-sm mb-1">
<span className="text-gray-700">{d.label}</span>
<span className="font-medium text-gray-900">{pct}% ({count})</span>
</div>
<div className="h-2 bg-gray-100 rounded-full overflow-hidden">
<div className={`h-full rounded-full ${d.color}`} style={{ width: `${pct}%` }} />
</div>
</div>
)
})}
{deviceTotal === 0 && (
<p className="text-xs text-gray-400">Noch keine Geraetedaten vorhanden</p>
)}
</div>
</div>
</div>
</div>
)
}