Files

213 lines
8.1 KiB
TypeScript

import Link from "next/link";
import { Mic2, Plus, Sparkles, ArrowRight, Mic, Gauge, Crown, Infinity as InfinityIcon } from "lucide-react";
import { requireAuth } from "@/lib/auth/guards";
import { getEffectivePlan } from "@/lib/billing/subscription";
import { getUsageSummary } from "@/lib/usage/meter";
import { prisma } from "@/lib/db";
import { UNLIMITED, type UsageMetric } from "@/lib/billing/plans";
import { PageHeader } from "@/components/app/page-header";
import { EpisodeStatusBadge } from "@/components/app/episode-status-badge";
import { StatCard } from "@/components/admin/ui/stat-card";
import { Button } from "@/components/ui/button";
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
import { Badge } from "@/components/ui/badge";
import { Progress } from "@/components/ui/progress";
import { EmptyState } from "@/components/ui/empty-state";
const METRIC_LABELS: Record<UsageMetric, string> = {
script: "Scripts",
audio: "Audio generations",
art: "Cover art",
repurpose: "Repurposed content",
};
export default async function DashboardPage() {
const session = await requireAuth();
const { plan, key, subjectId } = await getEffectivePlan(
session.user.id,
session.session.activeOrganizationId
);
// Episodes from the last 8 weeks for the sparkline + a 30-day rolling count.
const now = new Date();
const eightWeeksAgo = new Date(now);
eightWeeksAgo.setDate(eightWeeksAgo.getDate() - 7 * 8);
const [episodeCount, recent, recentForSpark, usage] = await Promise.all([
prisma.episode.count({ where: { userId: session.user.id } }),
prisma.episode.findMany({
where: { userId: session.user.id },
orderBy: { createdAt: "desc" },
take: 5,
select: { id: true, title: true, status: true, format: true, createdAt: true },
}),
prisma.episode.findMany({
where: { userId: session.user.id, createdAt: { gte: eightWeeksAgo } },
select: { createdAt: true },
}),
getUsageSummary(subjectId),
]);
// Bucket episodes into 8 weekly counts (oldest → newest) for the sparkline.
const weeklySpark = Array.from({ length: 8 }, (_, i) => {
const start = new Date(eightWeeksAgo);
start.setDate(start.getDate() + i * 7);
const end = new Date(start);
end.setDate(end.getDate() + 7);
return recentForSpark.filter((e) => e.createdAt >= start && e.createdAt < end).length;
});
const thisWeek = weeklySpark[weeklySpark.length - 1];
// Tightest metered limit: the metric closest to its cap (excluding unlimited).
const metrics = (Object.keys(METRIC_LABELS) as UsageMetric[])
.map((m) => {
const limit = plan.limits[m];
const used = usage[m];
const unlimited = limit === UNLIMITED;
const pct = unlimited ? 0 : Math.min(100, Math.round((used / Math.max(1, limit)) * 100));
return { metric: m, used, limit, unlimited, pct };
});
const tightest =
metrics
.filter((m) => !m.unlimited)
.sort((a, b) => b.pct - a.pct)[0] ?? metrics[0];
const tightestAtLimit = !tightest.unlimited && tightest.used >= tightest.limit;
const firstName = session.user.name.split(" ")[0];
return (
<>
<PageHeader
title={`Welcome back, ${firstName}`}
description="Spin up a fully produced episode in a couple of minutes."
action={
<Button asChild>
<Link href="/episodes/new">
<Plus className="h-4 w-4" /> New episode
</Link>
</Button>
}
/>
<div className="grid gap-4 lg:grid-cols-3">
<StatCard
label="Episodes created"
value={String(episodeCount)}
icon={Mic}
spark={weeklySpark.some((v) => v > 0) ? weeklySpark : undefined}
hint={`${thisWeek} this week`}
/>
{/* Live usage meter — tightest metered metric */}
<Card>
<CardContent className="space-y-3 p-5">
<div className="flex items-center justify-between">
<span className="text-sm font-medium text-muted-foreground">
{METRIC_LABELS[tightest.metric]} this month
</span>
<Gauge className="h-4 w-4 text-muted-foreground" />
</div>
<div className="flex items-baseline justify-between gap-3">
<p className="font-display text-3xl font-extrabold tracking-tight">{tightest.used}</p>
<span className="flex items-center gap-1 text-sm text-muted-foreground">
{tightest.unlimited ? (
<>
<InfinityIcon className="h-4 w-4" /> Unlimited
</>
) : (
<>of {tightest.limit}</>
)}
</span>
</div>
{!tightest.unlimited ? (
<Progress
value={tightest.pct}
indicatorClassName={tightestAtLimit ? "bg-warning" : undefined}
/>
) : (
<p className="text-xs text-muted-foreground">No limits on this metric.</p>
)}
<Button asChild size="sm" variant="ghost" className="-ml-2 h-8 px-2 text-brand">
<Link href="/usage">
View all usage <ArrowRight className="h-3.5 w-3.5" />
</Link>
</Button>
</CardContent>
</Card>
{/* Plan card with inline upgrade */}
<Card>
<CardContent className="space-y-3 p-5">
<div className="flex items-center justify-between">
<span className="text-sm font-medium text-muted-foreground">Current plan</span>
<Crown className="h-4 w-4 text-muted-foreground" />
</div>
<div className="flex items-center gap-2">
<p className="font-display text-3xl font-extrabold capitalize tracking-tight">
{plan.name}
</p>
{key !== "free" && <Badge variant="brand">Active</Badge>}
</div>
<p className="text-xs text-muted-foreground">{plan.tagline}</p>
{key !== "agency" && (
<Button asChild size="sm" variant={key === "free" ? "default" : "outline"}>
<Link href="/billing">
<Sparkles className="h-3.5 w-3.5" /> Upgrade
</Link>
</Button>
)}
</CardContent>
</Card>
</div>
<Card className="mt-6">
<CardHeader className="flex flex-row items-center justify-between space-y-0">
<CardTitle>Recent episodes</CardTitle>
{recent.length > 0 && (
<Button asChild variant="ghost" size="sm">
<Link href="/episodes">View all</Link>
</Button>
)}
</CardHeader>
<CardContent>
{recent.length === 0 ? (
<EmptyState
bordered={false}
icon={Mic2}
title="No episodes yet"
description="Create your first AI-produced episode to get started."
action={
<Button asChild>
<Link href="/episodes/new">
<Sparkles className="h-4 w-4" /> Create your first episode
</Link>
</Button>
}
/>
) : (
<ul className="divide-y">
{recent.map((ep) => (
<li key={ep.id}>
<Link
href={`/episodes/${ep.id}`}
className="flex items-center justify-between gap-3 py-3 transition-colors hover:opacity-80"
>
<div className="min-w-0">
<p className="truncate font-medium">{ep.title}</p>
<p className="text-xs text-muted-foreground">
{ep.format.replace("_", "-").toLowerCase()} ·{" "}
{ep.createdAt.toLocaleDateString()}
</p>
</div>
<EpisodeStatusBadge status={ep.status} />
</Link>
</li>
))}
</ul>
)}
</CardContent>
</Card>
</>
);
}