feat(agents): Enhance UI for agent and board management with improved styling and layout

This commit is contained in:
Abhimanyu Saharan
2026-02-04 21:27:58 +05:30
parent bf2a9452dc
commit 1a1efc2d5a
3 changed files with 215 additions and 203 deletions

View File

@@ -299,13 +299,13 @@ export default function AgentsPage() {
> >
<Link <Link
href={`/agents/${row.original.id}`} href={`/agents/${row.original.id}`}
className="inline-flex h-8 items-center justify-center rounded-lg border border-[color:var(--border)] px-3 text-xs font-medium text-muted transition hover:border-[color:var(--accent)] hover:text-[color:var(--accent)]" className="inline-flex h-8 items-center justify-center rounded-md border border-slate-200 px-3 text-xs font-medium text-slate-600 transition hover:border-slate-300 hover:text-slate-900"
> >
View View
</Link> </Link>
<Link <Link
href={`/agents/${row.original.id}/edit`} href={`/agents/${row.original.id}/edit`}
className="inline-flex h-8 items-center justify-center rounded-lg border border-[color:var(--border)] px-3 text-xs font-medium text-muted transition hover:border-[color:var(--accent)] hover:text-[color:var(--accent)]" className="inline-flex h-8 items-center justify-center rounded-md border border-slate-200 px-3 text-xs font-medium text-slate-600 transition hover:border-slate-300 hover:text-slate-900"
> >
Edit Edit
</Link> </Link>
@@ -336,24 +336,29 @@ export default function AgentsPage() {
return ( return (
<DashboardShell> <DashboardShell>
<SignedOut> <SignedOut>
<div className="flex h-full flex-col items-center justify-center gap-4 rounded-2xl surface-panel p-10 text-center lg:col-span-2"> <div className="col-span-2 flex min-h-[calc(100vh-64px)] items-center justify-center bg-slate-50 p-10 text-center">
<p className="text-sm text-muted">Sign in to view agents.</p> <div className="rounded-xl border border-slate-200 bg-white px-8 py-6 shadow-sm">
<p className="text-sm text-slate-600">Sign in to view agents.</p>
<SignInButton <SignInButton
mode="modal" mode="modal"
forceRedirectUrl="/agents" forceRedirectUrl="/agents"
signUpForceRedirectUrl="/agents" signUpForceRedirectUrl="/agents"
> >
<Button>Sign in</Button> <Button className="mt-4">Sign in</Button>
</SignInButton> </SignInButton>
</div> </div>
</div>
</SignedOut> </SignedOut>
<SignedIn> <SignedIn>
<DashboardSidebar /> <DashboardSidebar />
<div className="flex h-full flex-col gap-6 rounded-2xl surface-panel p-8"> <main className="flex-1 overflow-y-auto bg-slate-50">
<div className="flex flex-wrap items-center justify-between gap-3"> <div className="border-b border-slate-200 bg-white px-8 py-6">
<div className="flex flex-wrap items-center justify-between gap-4">
<div> <div>
<h2 className="text-lg font-semibold text-strong">Agents</h2> <h2 className="font-heading text-2xl font-semibold text-slate-900 tracking-tight">
<p className="text-sm text-muted"> Agents
</h2>
<p className="mt-1 text-sm text-slate-500">
{agents.length} agent{agents.length === 1 ? "" : "s"} total. {agents.length} agent{agents.length === 1 ? "" : "s"} total.
</p> </p>
</div> </div>
@@ -370,28 +375,30 @@ export default function AgentsPage() {
</Button> </Button>
</div> </div>
</div> </div>
</div>
<div className="p-8">
{error ? ( {error ? (
<div className="rounded-lg border border-[color:var(--border)] bg-[color:var(--surface-muted)] p-3 text-xs text-muted"> <div className="rounded-lg border border-slate-200 bg-white p-3 text-sm text-slate-600 shadow-sm">
{error} {error}
</div> </div>
) : null} ) : null}
{agents.length === 0 && !isLoading ? ( {agents.length === 0 && !isLoading ? (
<div className="flex flex-1 flex-col items-center justify-center gap-2 rounded-2xl border border-dashed border-[color:var(--border)] bg-[color:var(--surface-muted)] p-6 text-center text-sm text-muted"> <div className="flex flex-1 flex-col items-center justify-center gap-2 rounded-xl border border-dashed border-slate-200 bg-white/70 p-10 text-center text-sm text-slate-500">
No agents yet. Create your first agent to get started. No agents yet. Create your first agent to get started.
</div> </div>
) : ( ) : (
<div className="overflow-hidden rounded-2xl border border-[color:var(--border)] bg-[color:var(--surface)]"> <div className="overflow-hidden rounded-xl border border-slate-200 bg-white shadow-sm">
<div className="overflow-x-auto"> <div className="overflow-x-auto">
<table className="min-w-full divide-y divide-[color:var(--border)] text-sm"> <table className="min-w-full divide-y divide-slate-200 text-sm">
<thead className="bg-[color:var(--surface-muted)]"> <thead className="bg-slate-50">
{table.getHeaderGroups().map((headerGroup) => ( {table.getHeaderGroups().map((headerGroup) => (
<tr key={headerGroup.id}> <tr key={headerGroup.id}>
{headerGroup.headers.map((header) => ( {headerGroup.headers.map((header) => (
<th <th
key={header.id} key={header.id}
className="px-4 py-3 text-left text-[11px] font-semibold uppercase tracking-[0.22em] text-quiet" className="px-4 py-3 text-left text-[11px] font-semibold uppercase tracking-wider text-slate-500"
> >
{header.isPlaceholder {header.isPlaceholder
? null ? null
@@ -404,11 +411,11 @@ export default function AgentsPage() {
</tr> </tr>
))} ))}
</thead> </thead>
<tbody className="divide-y divide-[color:var(--border)] bg-[color:var(--surface)]"> <tbody className="divide-y divide-slate-200 bg-white">
{table.getRowModel().rows.map((row) => ( {table.getRowModel().rows.map((row) => (
<tr <tr
key={row.id} key={row.id}
className="cursor-pointer transition hover:bg-[color:var(--surface-muted)]" className="cursor-pointer transition hover:bg-slate-50"
onClick={() => router.push(`/agents/${row.original.id}`)} onClick={() => router.push(`/agents/${row.original.id}`)}
> >
{row.getVisibleCells().map((cell) => ( {row.getVisibleCells().map((cell) => (
@@ -427,13 +434,13 @@ export default function AgentsPage() {
</div> </div>
)} )}
<div className="rounded-2xl border border-[color:var(--border)] bg-[color:var(--surface)] p-4 text-sm text-muted"> <div className="mt-6 rounded-xl border border-slate-200 bg-white p-4 text-sm text-slate-600 shadow-sm">
<div className="flex flex-wrap items-center justify-between gap-3"> <div className="flex flex-wrap items-center justify-between gap-3">
<div> <div>
<p className="text-xs font-semibold uppercase tracking-[0.2em] text-quiet"> <p className="text-xs font-semibold uppercase tracking-wider text-slate-500">
Gateway status Gateway status
</p> </p>
<p className="mt-1 text-sm text-strong"> <p className="mt-1 text-sm font-semibold text-slate-900">
{gatewayStatus?.gateway_url ?? "Gateway URL not set"} {gatewayStatus?.gateway_url ?? "Gateway URL not set"}
</p> </p>
</div> </div>
@@ -455,23 +462,20 @@ export default function AgentsPage() {
</SelectContent> </SelectContent>
</Select> </Select>
<StatusPill status={gatewayStatus?.connected ? "online" : "offline"} /> <StatusPill status={gatewayStatus?.connected ? "online" : "offline"} />
<span className="text-xs text-quiet"> <span className="text-xs text-slate-500">
{gatewayStatus?.sessions_count ?? 0} sessions {gatewayStatus?.sessions_count ?? 0} sessions
</span> </span>
</div> </div>
</div> </div>
{gatewayStatus?.error ? ( {gatewayStatus?.error ? (
<p className="mt-3 text-xs text-[color:var(--danger)]"> <p className="mt-3 text-xs text-red-600">{gatewayStatus.error}</p>
{gatewayStatus.error}
</p>
) : null} ) : null}
{gatewayError ? ( {gatewayError ? (
<p className="mt-3 text-xs text-[color:var(--danger)]"> <p className="mt-3 text-xs text-red-600">{gatewayError}</p>
{gatewayError}
</p>
) : null} ) : null}
</div> </div>
</div> </div>
</main>
</SignedIn> </SignedIn>
<Dialog <Dialog

View File

@@ -122,13 +122,13 @@ export default function BoardsPage() {
> >
<Link <Link
href={`/boards/${row.original.id}`} href={`/boards/${row.original.id}`}
className="inline-flex h-8 items-center justify-center rounded-lg border border-[color:var(--border)] px-3 text-xs font-medium text-muted transition hover:border-[color:var(--accent)] hover:text-[color:var(--accent)]" className="inline-flex h-8 items-center justify-center rounded-md border border-slate-200 px-3 text-xs font-medium text-slate-600 transition hover:border-slate-300 hover:text-slate-900"
> >
Open Open
</Link> </Link>
<Link <Link
href={`/boards/${row.original.id}/edit`} href={`/boards/${row.original.id}/edit`}
className="inline-flex h-8 items-center justify-center rounded-lg border border-[color:var(--border)] px-3 text-xs font-medium text-muted transition hover:border-[color:var(--accent)] hover:text-[color:var(--accent)]" className="inline-flex h-8 items-center justify-center rounded-md border border-slate-200 px-3 text-xs font-medium text-slate-600 transition hover:border-slate-300 hover:text-slate-900"
> >
Edit Edit
</Link> </Link>
@@ -155,24 +155,29 @@ export default function BoardsPage() {
return ( return (
<DashboardShell> <DashboardShell>
<SignedOut> <SignedOut>
<div className="flex h-full flex-col items-center justify-center gap-4 rounded-2xl surface-panel p-10 text-center lg:col-span-2"> <div className="col-span-2 flex min-h-[calc(100vh-64px)] items-center justify-center bg-slate-50 p-10 text-center">
<p className="text-sm text-muted">Sign in to view boards.</p> <div className="rounded-xl border border-slate-200 bg-white px-8 py-6 shadow-sm">
<p className="text-sm text-slate-600">Sign in to view boards.</p>
<SignInButton <SignInButton
mode="modal" mode="modal"
forceRedirectUrl="/boards" forceRedirectUrl="/boards"
signUpForceRedirectUrl="/boards" signUpForceRedirectUrl="/boards"
> >
<Button>Sign in</Button> <Button className="mt-4">Sign in</Button>
</SignInButton> </SignInButton>
</div> </div>
</div>
</SignedOut> </SignedOut>
<SignedIn> <SignedIn>
<DashboardSidebar /> <DashboardSidebar />
<div className="flex h-full flex-col gap-6 rounded-2xl surface-panel p-8"> <main className="flex-1 overflow-y-auto bg-slate-50">
<div className="flex flex-wrap items-center justify-between gap-3"> <div className="border-b border-slate-200 bg-white px-8 py-6">
<div className="flex flex-wrap items-center justify-between gap-4">
<div> <div>
<h2 className="text-lg font-semibold text-strong">Boards</h2> <h2 className="font-heading text-2xl font-semibold text-slate-900 tracking-tight">
<p className="text-sm text-muted"> Boards
</h2>
<p className="mt-1 text-sm text-slate-500">
{sortedBoards.length} board {sortedBoards.length} board
{sortedBoards.length === 1 ? "" : "s"} total. {sortedBoards.length === 1 ? "" : "s"} total.
</p> </p>
@@ -181,27 +186,29 @@ export default function BoardsPage() {
New board New board
</Button> </Button>
</div> </div>
</div>
<div className="p-8">
{error && ( {error && (
<div className="rounded-lg border border-[color:var(--border)] bg-[color:var(--surface-muted)] p-3 text-xs text-muted"> <div className="rounded-lg border border-slate-200 bg-white p-3 text-sm text-slate-600 shadow-sm">
{error} {error}
</div> </div>
)} )}
{sortedBoards.length === 0 && !isLoading ? ( {sortedBoards.length === 0 && !isLoading ? (
<div className="flex flex-1 flex-col items-center justify-center gap-2 rounded-2xl border border-dashed border-[color:var(--border)] bg-[color:var(--surface-muted)] p-6 text-center text-sm text-muted"> <div className="flex flex-1 flex-col items-center justify-center gap-2 rounded-xl border border-dashed border-slate-200 bg-white/70 p-10 text-center text-sm text-slate-500">
No boards yet. Create your first board to get started. No boards yet. Create your first board to get started.
</div> </div>
) : ( ) : (
<div className="overflow-hidden rounded-2xl border border-[color:var(--border)] bg-[color:var(--surface)]"> <div className="overflow-hidden rounded-xl border border-slate-200 bg-white shadow-sm">
<table className="min-w-full divide-y divide-[color:var(--border)] text-sm"> <table className="min-w-full divide-y divide-slate-200 text-sm">
<thead className="bg-[color:var(--surface-muted)]"> <thead className="bg-slate-50">
{table.getHeaderGroups().map((headerGroup) => ( {table.getHeaderGroups().map((headerGroup) => (
<tr key={headerGroup.id}> <tr key={headerGroup.id}>
{headerGroup.headers.map((header) => ( {headerGroup.headers.map((header) => (
<th <th
key={header.id} key={header.id}
className="px-4 py-3 text-left text-[11px] font-semibold uppercase tracking-[0.22em] text-quiet" className="px-4 py-3 text-left text-[11px] font-semibold uppercase tracking-wider text-slate-500"
> >
{header.isPlaceholder {header.isPlaceholder
? null ? null
@@ -214,11 +221,11 @@ export default function BoardsPage() {
</tr> </tr>
))} ))}
</thead> </thead>
<tbody className="divide-y divide-[color:var(--border)] bg-[color:var(--surface)]"> <tbody className="divide-y divide-slate-200 bg-white">
{table.getRowModel().rows.map((row) => ( {table.getRowModel().rows.map((row) => (
<tr <tr
key={row.id} key={row.id}
className="cursor-pointer transition hover:bg-[color:var(--surface-muted)]" className="cursor-pointer transition hover:bg-slate-50"
onClick={() => router.push(`/boards/${row.original.id}`)} onClick={() => router.push(`/boards/${row.original.id}`)}
> >
{row.getVisibleCells().map((cell) => ( {row.getVisibleCells().map((cell) => (
@@ -236,6 +243,7 @@ export default function BoardsPage() {
</div> </div>
)} )}
</div> </div>
</main>
</SignedIn> </SignedIn>
<Dialog <Dialog

View File

@@ -182,7 +182,7 @@ function KpiCard({
progress?: number; progress?: number;
}) { }) {
return ( return (
<div className="stat-card rounded-xl border border-slate-200 bg-white p-6 shadow-sm transition"> <div className="rounded-xl border border-slate-200 bg-white p-6 shadow-sm transition hover:-translate-y-0.5 hover:shadow-md">
<div className="mb-4 flex items-center justify-between"> <div className="mb-4 flex items-center justify-between">
<p className="text-xs font-semibold uppercase tracking-wider text-slate-500"> <p className="text-xs font-semibold uppercase tracking-wider text-slate-500">
{label} {label}