Overhaul UI with shadcn components and new pages
Add shadcn/ui components (badge, button, card, input, select,
separator). Add company context provider. New pages: Activity,
Approvals, Companies, Costs, Org chart. Restyle existing pages
(Dashboard, Agents, Issues, Goals, Projects) with shadcn components
and dark theme. Update layout, sidebar navigation, and routing.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-17 09:07:32 -06:00
|
|
|
import { type ClassValue, clsx } from "clsx";
|
|
|
|
|
import { twMerge } from "tailwind-merge";
|
2026-03-03 08:45:26 -06:00
|
|
|
import { deriveAgentUrlKey, deriveProjectUrlKey } from "@paperclipai/shared";
|
2026-03-14 22:00:12 -05:00
|
|
|
import type { BillingType, FinanceDirection, FinanceEventKind } from "@paperclipai/shared";
|
Overhaul UI with shadcn components and new pages
Add shadcn/ui components (badge, button, card, input, select,
separator). Add company context provider. New pages: Activity,
Approvals, Companies, Costs, Org chart. Restyle existing pages
(Dashboard, Agents, Issues, Goals, Projects) with shadcn components
and dark theme. Update layout, sidebar navigation, and routing.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-17 09:07:32 -06:00
|
|
|
|
|
|
|
|
export function cn(...inputs: ClassValue[]) {
|
|
|
|
|
return twMerge(clsx(inputs));
|
2026-02-16 13:32:04 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function formatCents(cents: number): string {
|
|
|
|
|
return `$${(cents / 100).toFixed(2)}`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function formatDate(date: Date | string): string {
|
|
|
|
|
return new Date(date).toLocaleDateString("en-US", {
|
|
|
|
|
month: "short",
|
|
|
|
|
day: "numeric",
|
|
|
|
|
year: "numeric",
|
|
|
|
|
});
|
|
|
|
|
}
|
Build out agent management UI: detail page, create dialog, list view
Add NewAgentDialog for creating agents with adapter config. Expand
AgentDetail page with tabbed view (overview, runs, config, logs),
run history timeline, and live status. Enhance Agents list page with
richer cards and filtering. Update AgentProperties panel, API client,
query keys, and utility helpers.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-17 12:33:04 -06:00
|
|
|
|
2026-02-20 16:19:54 -06:00
|
|
|
export function formatDateTime(date: Date | string): string {
|
|
|
|
|
return new Date(date).toLocaleString("en-US", {
|
|
|
|
|
month: "short",
|
|
|
|
|
day: "numeric",
|
|
|
|
|
year: "numeric",
|
|
|
|
|
hour: "numeric",
|
|
|
|
|
minute: "2-digit",
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
Build out agent management UI: detail page, create dialog, list view
Add NewAgentDialog for creating agents with adapter config. Expand
AgentDetail page with tabbed view (overview, runs, config, logs),
run history timeline, and live status. Enhance Agents list page with
richer cards and filtering. Update AgentProperties panel, API client,
query keys, and utility helpers.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-17 12:33:04 -06:00
|
|
|
export function relativeTime(date: Date | string): string {
|
|
|
|
|
const now = Date.now();
|
|
|
|
|
const then = new Date(date).getTime();
|
|
|
|
|
const diffSec = Math.round((now - then) / 1000);
|
|
|
|
|
if (diffSec < 60) return "just now";
|
|
|
|
|
const diffMin = Math.round(diffSec / 60);
|
|
|
|
|
if (diffMin < 60) return `${diffMin}m ago`;
|
|
|
|
|
const diffHr = Math.round(diffMin / 60);
|
|
|
|
|
if (diffHr < 24) return `${diffHr}h ago`;
|
|
|
|
|
const diffDay = Math.round(diffHr / 24);
|
|
|
|
|
if (diffDay < 30) return `${diffDay}d ago`;
|
|
|
|
|
return formatDate(date);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function formatTokens(n: number): string {
|
|
|
|
|
if (n >= 1_000_000) return `${(n / 1_000_000).toFixed(1)}M`;
|
|
|
|
|
if (n >= 1_000) return `${(n / 1_000).toFixed(1)}k`;
|
|
|
|
|
return String(n);
|
|
|
|
|
}
|
2026-02-20 16:04:05 -06:00
|
|
|
|
2026-03-08 03:35:23 +05:30
|
|
|
/** Map a raw provider slug to a display-friendly name. */
|
|
|
|
|
export function providerDisplayName(provider: string): string {
|
|
|
|
|
const map: Record<string, string> = {
|
|
|
|
|
anthropic: "Anthropic",
|
|
|
|
|
openai: "OpenAI",
|
2026-03-14 22:00:12 -05:00
|
|
|
openrouter: "OpenRouter",
|
|
|
|
|
chatgpt: "ChatGPT",
|
2026-03-08 03:35:23 +05:30
|
|
|
google: "Google",
|
|
|
|
|
cursor: "Cursor",
|
|
|
|
|
jetbrains: "JetBrains AI",
|
|
|
|
|
};
|
|
|
|
|
return map[provider.toLowerCase()] ?? provider;
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-14 22:00:12 -05:00
|
|
|
export function billingTypeDisplayName(billingType: BillingType): string {
|
|
|
|
|
const map: Record<BillingType, string> = {
|
|
|
|
|
metered_api: "Metered API",
|
|
|
|
|
subscription_included: "Subscription",
|
|
|
|
|
subscription_overage: "Subscription overage",
|
|
|
|
|
credits: "Credits",
|
|
|
|
|
fixed: "Fixed",
|
|
|
|
|
unknown: "Unknown",
|
|
|
|
|
};
|
|
|
|
|
return map[billingType];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function quotaSourceDisplayName(source: string): string {
|
|
|
|
|
const map: Record<string, string> = {
|
|
|
|
|
"anthropic-oauth": "Anthropic OAuth",
|
|
|
|
|
"claude-cli": "Claude CLI",
|
|
|
|
|
"codex-rpc": "Codex app server",
|
|
|
|
|
"codex-wham": "ChatGPT WHAM",
|
|
|
|
|
};
|
|
|
|
|
return map[source] ?? source;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function coerceBillingType(value: unknown): BillingType | null {
|
|
|
|
|
if (
|
|
|
|
|
value === "metered_api" ||
|
|
|
|
|
value === "subscription_included" ||
|
|
|
|
|
value === "subscription_overage" ||
|
|
|
|
|
value === "credits" ||
|
|
|
|
|
value === "fixed" ||
|
|
|
|
|
value === "unknown"
|
|
|
|
|
) {
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function readRunCostUsd(payload: Record<string, unknown> | null): number {
|
|
|
|
|
if (!payload) return 0;
|
|
|
|
|
for (const key of ["costUsd", "cost_usd", "total_cost_usd"] as const) {
|
|
|
|
|
const value = payload[key];
|
|
|
|
|
if (typeof value === "number" && Number.isFinite(value)) return value;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function visibleRunCostUsd(
|
|
|
|
|
usage: Record<string, unknown> | null,
|
|
|
|
|
result: Record<string, unknown> | null = null,
|
|
|
|
|
): number {
|
|
|
|
|
const billingType = coerceBillingType(usage?.billingType) ?? coerceBillingType(result?.billingType);
|
|
|
|
|
if (billingType === "subscription_included") return 0;
|
|
|
|
|
return readRunCostUsd(usage) || readRunCostUsd(result);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function financeEventKindDisplayName(eventKind: FinanceEventKind): string {
|
|
|
|
|
const map: Record<FinanceEventKind, string> = {
|
|
|
|
|
inference_charge: "Inference charge",
|
|
|
|
|
platform_fee: "Platform fee",
|
|
|
|
|
credit_purchase: "Credit purchase",
|
|
|
|
|
credit_refund: "Credit refund",
|
|
|
|
|
credit_expiry: "Credit expiry",
|
|
|
|
|
byok_fee: "BYOK fee",
|
|
|
|
|
gateway_overhead: "Gateway overhead",
|
|
|
|
|
log_storage_charge: "Log storage",
|
|
|
|
|
logpush_charge: "Logpush",
|
|
|
|
|
provisioned_capacity_charge: "Provisioned capacity",
|
|
|
|
|
training_charge: "Training",
|
|
|
|
|
custom_model_import_charge: "Custom model import",
|
|
|
|
|
custom_model_storage_charge: "Custom model storage",
|
|
|
|
|
manual_adjustment: "Manual adjustment",
|
|
|
|
|
};
|
|
|
|
|
return map[eventKind];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function financeDirectionDisplayName(direction: FinanceDirection): string {
|
|
|
|
|
return direction === "credit" ? "Credit" : "Debit";
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-20 16:04:05 -06:00
|
|
|
/** Build an issue URL using the human-readable identifier when available. */
|
|
|
|
|
export function issueUrl(issue: { id: string; identifier?: string | null }): string {
|
|
|
|
|
return `/issues/${issue.identifier ?? issue.id}`;
|
|
|
|
|
}
|
2026-03-02 16:44:03 -06:00
|
|
|
|
|
|
|
|
/** Build an agent route URL using the short URL key when available. */
|
|
|
|
|
export function agentRouteRef(agent: { id: string; urlKey?: string | null; name?: string | null }): string {
|
|
|
|
|
return agent.urlKey ?? deriveAgentUrlKey(agent.name, agent.id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Build an agent URL using the short URL key when available. */
|
|
|
|
|
export function agentUrl(agent: { id: string; urlKey?: string | null; name?: string | null }): string {
|
|
|
|
|
return `/agents/${agentRouteRef(agent)}`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Build a project route reference using the short URL key when available. */
|
|
|
|
|
export function projectRouteRef(project: { id: string; urlKey?: string | null; name?: string | null }): string {
|
|
|
|
|
return project.urlKey ?? deriveProjectUrlKey(project.name, project.id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Build a project URL using the short URL key when available. */
|
|
|
|
|
export function projectUrl(project: { id: string; urlKey?: string | null; name?: string | null }): string {
|
|
|
|
|
return `/projects/${projectRouteRef(project)}`;
|
|
|
|
|
}
|
2026-03-28 09:51:58 -05:00
|
|
|
|
|
|
|
|
/** Build a project workspace URL scoped under its project. */
|
|
|
|
|
export function projectWorkspaceUrl(
|
|
|
|
|
project: { id: string; urlKey?: string | null; name?: string | null },
|
|
|
|
|
workspaceId: string,
|
|
|
|
|
): string {
|
|
|
|
|
return `${projectUrl(project)}/workspaces/${workspaceId}`;
|
|
|
|
|
}
|