Third phase of the DESIGN.md migration. Removes every raw Tailwind
color palette utility (bg-red-*, text-amber-*, border-blue-*, etc.)
from component source and replaces them with the semantic tokens
introduced in phases 1 and 2.
Scope:
- 84 files touched under ui/src/
- ~420 raw palette utility instances replaced
- 23 hardcoded hex fallbacks replaced with var(--token) refs
- Zero raw palette utilities remain in component source
(verified with rg '(bg|text|border|ring)-(red|blue|green|amber|
yellow|cyan|violet|purple|pink|slate|zinc|neutral|sky|teal|
emerald|indigo|rose|orange|fuchsia)-[0-9]+' ui/src)
Mapping rules applied:
- red-* -> destructive
- amber-/yellow-/orange-* -> warning
- green-/emerald-* -> success
- blue-/cyan-/sky-* -> primary (info/in-progress) or muted-foreground
- slate-/gray-/zinc-/neutral-* -> muted / muted-foreground / border
- violet-/purple-/pink-/indigo-/rose-/teal-* -> collapsed to
primary or muted (most were one-off decorative choices, not
role-bearing). Role-bearing uses go through lib/agent-role-colors
which was rewritten in phase 2.
- Opacity modifiers preserved (/10, /15, /20, etc.)
- dark: variant duplicates removed (theme tokens auto-switch)
Hardcoded hex fallbacks fixed:
- #6366f1 (indigo) -> var(--primary) / var(--volt)
- #64748b (slate) -> var(--muted-foreground) / var(--silver)
- #4f46e5 (indigo) -> var(--primary)
- #89b4fa (old Catppuccin blue) -> var(--primary) / #faff69
- OrgChart status dots (#22d3ee/#4ade80/#facc15/#f87171/#a3a3a3)
-> var(--primary) / var(--success) / var(--warning) /
var(--destructive) / var(--muted-foreground) per status
- VoiceWaveform fallback #89b4fa -> #faff69 (volt)
Legitimate hex values left untouched (12 total):
- lib/color-contrast.ts WCAG reference constants
- lib/worktree-branding.ts contrast fallback references
- lib/mention-chips.ts runtime-generated SVG fills
- context/ThemeContext.tsx theme metadata brand hexes
- components/ThemeSeedInput.tsx user-facing hex picker
Ambiguous decisions (flagged for visual QA):
- AgentDetail.tsx invocation-source badges (timer/assignment/
on_demand) collapsed to primary/muted — visual distinction
is reduced, labels still differ. Consider chart-role slots
if differentiation matters.
- AgentDetail.tsx mixed-opacity amber banners: bg-warning/60
against new warning base reads heavier than original amber-50
base.
- Live-state dots in KanbanBoard/AgentDetail: bg-blue-* ->
bg-primary — will glow volt in dark mode, probably desirable.
Verification:
- npx tsc --noEmit in ui/ — zero errors introduced. Pre-existing
errors in AgentConfigForm, command.tsx, useKeyboardShortcuts,
usePiperTts, useVadRecorder, PersonalAssistant remain, all
unrelated to color work.
- Dev server on :6100 returns 200.
Not changed in this commit:
- ui/src/lib/company-routes.ts — separate routing fix for broken
Assistant/ContentStudio/Convert links, committed next.
- Test files — a few will need assertion updates but are out of
phase 3 scope.
Phase 4 follow-ups (rounded-xl/2xl collapse, soft shadow removal,
gradient removal) noted in .planning/AUDIT-RADIUS-SHADOWS.md.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
263 lines
9.9 KiB
TypeScript
263 lines
9.9 KiB
TypeScript
import type { HeartbeatRun } from "@paperclipai/shared";
|
|
|
|
/* ---- Utilities ---- */
|
|
|
|
export function getLast14Days(): string[] {
|
|
return Array.from({ length: 14 }, (_, i) => {
|
|
const d = new Date();
|
|
d.setDate(d.getDate() - (13 - i));
|
|
return d.toISOString().slice(0, 10);
|
|
});
|
|
}
|
|
|
|
function formatDayLabel(dateStr: string): string {
|
|
const d = new Date(dateStr + "T12:00:00");
|
|
return `${d.getMonth() + 1}/${d.getDate()}`;
|
|
}
|
|
|
|
/* ---- Sub-components ---- */
|
|
|
|
function DateLabels({ days }: { days: string[] }) {
|
|
return (
|
|
<div className="flex gap-[3px] mt-1.5">
|
|
{days.map((day, i) => (
|
|
<div key={day} className="flex-1 text-center">
|
|
{(i === 0 || i === 6 || i === 13) ? (
|
|
<span className="text-[9px] text-muted-foreground tabular-nums">{formatDayLabel(day)}</span>
|
|
) : null}
|
|
</div>
|
|
))}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function ChartLegend({ items }: { items: { color: string; label: string }[] }) {
|
|
return (
|
|
<div className="flex flex-wrap gap-x-2.5 gap-y-0.5 mt-2">
|
|
{items.map(item => (
|
|
<span key={item.label} className="flex items-center gap-1 text-[9px] text-muted-foreground">
|
|
<span className="h-1.5 w-1.5 rounded-full shrink-0" style={{ backgroundColor: item.color }} />
|
|
{item.label}
|
|
</span>
|
|
))}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export function ChartCard({ title, subtitle, children }: { title: string; subtitle?: string; children: React.ReactNode }) {
|
|
return (
|
|
<div className="border border-border rounded-lg p-4 space-y-3">
|
|
<div>
|
|
<h3 className="text-xs font-medium text-muted-foreground">{title}</h3>
|
|
{subtitle && <span className="text-[10px] text-muted-foreground/60">{subtitle}</span>}
|
|
</div>
|
|
{children}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
/* ---- Chart Components ---- */
|
|
|
|
export function RunActivityChart({ runs }: { runs: HeartbeatRun[] }) {
|
|
const days = getLast14Days();
|
|
|
|
const grouped = new Map<string, { succeeded: number; failed: number; other: number }>();
|
|
for (const day of days) grouped.set(day, { succeeded: 0, failed: 0, other: 0 });
|
|
for (const run of runs) {
|
|
const day = new Date(run.createdAt).toISOString().slice(0, 10);
|
|
const entry = grouped.get(day);
|
|
if (!entry) continue;
|
|
if (run.status === "succeeded") entry.succeeded++;
|
|
else if (run.status === "failed" || run.status === "timed_out") entry.failed++;
|
|
else entry.other++;
|
|
}
|
|
|
|
const maxValue = Math.max(...Array.from(grouped.values()).map(v => v.succeeded + v.failed + v.other), 1);
|
|
const hasData = Array.from(grouped.values()).some(v => v.succeeded + v.failed + v.other > 0);
|
|
|
|
if (!hasData) return <p className="text-xs text-muted-foreground">No runs yet</p>;
|
|
|
|
return (
|
|
<div>
|
|
<div className="flex items-end gap-[3px] h-20">
|
|
{days.map(day => {
|
|
const entry = grouped.get(day)!;
|
|
const total = entry.succeeded + entry.failed + entry.other;
|
|
const heightPct = (total / maxValue) * 100;
|
|
return (
|
|
<div key={day} className="flex-1 h-full flex flex-col justify-end" title={`${day}: ${total} runs`}>
|
|
{total > 0 ? (
|
|
<div className="flex flex-col-reverse gap-px overflow-hidden" style={{ height: `${heightPct}%`, minHeight: 2 }}>
|
|
{entry.succeeded > 0 && <div className="bg-success" style={{ flex: entry.succeeded }} />}
|
|
{entry.failed > 0 && <div className="bg-destructive" style={{ flex: entry.failed }} />}
|
|
{entry.other > 0 && <div className="bg-muted" style={{ flex: entry.other }} />}
|
|
</div>
|
|
) : (
|
|
<div className="bg-muted/30 rounded-sm" style={{ height: 2 }} />
|
|
)}
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
<DateLabels days={days} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const priorityColors: Record<string, string> = {
|
|
critical: "var(--destructive)",
|
|
high: "var(--chart-4)",
|
|
medium: "var(--chart-1)",
|
|
low: "var(--muted-foreground)",
|
|
};
|
|
|
|
const priorityOrder = ["critical", "high", "medium", "low"] as const;
|
|
|
|
export function PriorityChart({ issues }: { issues: { priority: string; createdAt: Date }[] }) {
|
|
const days = getLast14Days();
|
|
const grouped = new Map<string, Record<string, number>>();
|
|
for (const day of days) grouped.set(day, { critical: 0, high: 0, medium: 0, low: 0 });
|
|
for (const issue of issues) {
|
|
const day = new Date(issue.createdAt).toISOString().slice(0, 10);
|
|
const entry = grouped.get(day);
|
|
if (!entry) continue;
|
|
if (issue.priority in entry) entry[issue.priority]++;
|
|
}
|
|
|
|
const maxValue = Math.max(...Array.from(grouped.values()).map(v => Object.values(v).reduce((a, b) => a + b, 0)), 1);
|
|
const hasData = Array.from(grouped.values()).some(v => Object.values(v).reduce((a, b) => a + b, 0) > 0);
|
|
|
|
if (!hasData) return <p className="text-xs text-muted-foreground">No issues</p>;
|
|
|
|
return (
|
|
<div>
|
|
<div className="flex items-end gap-[3px] h-20">
|
|
{days.map(day => {
|
|
const entry = grouped.get(day)!;
|
|
const total = Object.values(entry).reduce((a, b) => a + b, 0);
|
|
const heightPct = (total / maxValue) * 100;
|
|
return (
|
|
<div key={day} className="flex-1 h-full flex flex-col justify-end" title={`${day}: ${total} issues`}>
|
|
{total > 0 ? (
|
|
<div className="flex flex-col-reverse gap-px overflow-hidden" style={{ height: `${heightPct}%`, minHeight: 2 }}>
|
|
{priorityOrder.map(p => entry[p] > 0 ? (
|
|
<div key={p} style={{ flex: entry[p], backgroundColor: priorityColors[p] }} />
|
|
) : null)}
|
|
</div>
|
|
) : (
|
|
<div className="bg-muted/30 rounded-sm" style={{ height: 2 }} />
|
|
)}
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
<DateLabels days={days} />
|
|
<ChartLegend items={priorityOrder.map(p => ({ color: priorityColors[p], label: p.charAt(0).toUpperCase() + p.slice(1) }))} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const statusColors: Record<string, string> = {
|
|
todo: "var(--chart-1)",
|
|
in_progress: "var(--chart-3)",
|
|
in_review: "var(--chart-4)",
|
|
done: "var(--chart-2)",
|
|
blocked: "var(--destructive)",
|
|
cancelled: "var(--muted-foreground)",
|
|
backlog: "var(--muted-foreground)",
|
|
};
|
|
|
|
const statusLabels: Record<string, string> = {
|
|
todo: "To Do",
|
|
in_progress: "In Progress",
|
|
in_review: "In Review",
|
|
done: "Done",
|
|
blocked: "Blocked",
|
|
cancelled: "Cancelled",
|
|
backlog: "Backlog",
|
|
};
|
|
|
|
export function IssueStatusChart({ issues }: { issues: { status: string; createdAt: Date }[] }) {
|
|
const days = getLast14Days();
|
|
const allStatuses = new Set<string>();
|
|
const grouped = new Map<string, Record<string, number>>();
|
|
for (const day of days) grouped.set(day, {});
|
|
for (const issue of issues) {
|
|
const day = new Date(issue.createdAt).toISOString().slice(0, 10);
|
|
const entry = grouped.get(day);
|
|
if (!entry) continue;
|
|
entry[issue.status] = (entry[issue.status] ?? 0) + 1;
|
|
allStatuses.add(issue.status);
|
|
}
|
|
|
|
const statusOrder = ["todo", "in_progress", "in_review", "done", "blocked", "cancelled", "backlog"].filter(s => allStatuses.has(s));
|
|
const maxValue = Math.max(...Array.from(grouped.values()).map(v => Object.values(v).reduce((a, b) => a + b, 0)), 1);
|
|
const hasData = allStatuses.size > 0;
|
|
|
|
if (!hasData) return <p className="text-xs text-muted-foreground">No issues</p>;
|
|
|
|
return (
|
|
<div>
|
|
<div className="flex items-end gap-[3px] h-20">
|
|
{days.map(day => {
|
|
const entry = grouped.get(day)!;
|
|
const total = Object.values(entry).reduce((a, b) => a + b, 0);
|
|
const heightPct = (total / maxValue) * 100;
|
|
return (
|
|
<div key={day} className="flex-1 h-full flex flex-col justify-end" title={`${day}: ${total} issues`}>
|
|
{total > 0 ? (
|
|
<div className="flex flex-col-reverse gap-px overflow-hidden" style={{ height: `${heightPct}%`, minHeight: 2 }}>
|
|
{statusOrder.map(s => (entry[s] ?? 0) > 0 ? (
|
|
<div key={s} style={{ flex: entry[s], backgroundColor: statusColors[s] ?? "var(--muted-foreground)" }} />
|
|
) : null)}
|
|
</div>
|
|
) : (
|
|
<div className="bg-muted/30 rounded-sm" style={{ height: 2 }} />
|
|
)}
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
<DateLabels days={days} />
|
|
<ChartLegend items={statusOrder.map(s => ({ color: statusColors[s] ?? "var(--muted-foreground)", label: statusLabels[s] ?? s }))} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export function SuccessRateChart({ runs }: { runs: HeartbeatRun[] }) {
|
|
const days = getLast14Days();
|
|
const grouped = new Map<string, { succeeded: number; total: number }>();
|
|
for (const day of days) grouped.set(day, { succeeded: 0, total: 0 });
|
|
for (const run of runs) {
|
|
const day = new Date(run.createdAt).toISOString().slice(0, 10);
|
|
const entry = grouped.get(day);
|
|
if (!entry) continue;
|
|
entry.total++;
|
|
if (run.status === "succeeded") entry.succeeded++;
|
|
}
|
|
|
|
const hasData = Array.from(grouped.values()).some(v => v.total > 0);
|
|
if (!hasData) return <p className="text-xs text-muted-foreground">No runs yet</p>;
|
|
|
|
return (
|
|
<div>
|
|
<div className="flex items-end gap-[3px] h-20">
|
|
{days.map(day => {
|
|
const entry = grouped.get(day)!;
|
|
const rate = entry.total > 0 ? entry.succeeded / entry.total : 0;
|
|
const color = entry.total === 0 ? undefined : rate >= 0.8 ? "var(--chart-2)" : rate >= 0.5 ? "var(--chart-4)" : "var(--destructive)";
|
|
return (
|
|
<div key={day} className="flex-1 h-full flex flex-col justify-end" title={`${day}: ${entry.total > 0 ? Math.round(rate * 100) : 0}% (${entry.succeeded}/${entry.total})`}>
|
|
{entry.total > 0 ? (
|
|
<div style={{ height: `${rate * 100}%`, minHeight: 2, backgroundColor: color }} />
|
|
) : (
|
|
<div className="bg-muted/30 rounded-sm" style={{ height: 2 }} />
|
|
)}
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
<DateLabels days={days} />
|
|
</div>
|
|
);
|
|
}
|