feat(tracker-web): redesign fleet jobs list — stage chips, filters, hide-shipped

- Clickable stage chips with live counts (color-coded) replace the plain dropdown;
  click to filter, click again to clear.
- Search box (key / repo / id) + 'Hide shipped' checkbox (persisted) + 'N of M shown'.
- Redesigned table: color-coded stage badges, priority emphasis, a Repo column
  (PR target), newest-first sort, bordered/zebra layout.
- All filtering is client-side over a 100-job window (instant + accurate counts);
  list/links/New-Job form unchanged. FleetJob gains repo/baseBranch.

Generated with [Devin](https://cli.devin.ai/docs)

Co-Authored-By: Devin <158243242+devin-ai-integration[bot]@users.noreply.github.com>
This commit is contained in:
saravanakumardb1 2026-05-31 16:58:12 -07:00
parent 2bd97791c9
commit 6709862c1a
2 changed files with 141 additions and 43 deletions

View File

@ -8,7 +8,6 @@ import { useAuth } from '@/lib/auth-context';
import { listJobs, submitJob, type FleetJob } from '@/lib/fleet-client';
const STAGES = [
'',
'queued',
'blocked',
'assigned',
@ -18,7 +17,28 @@ const STAGES = [
'shipped',
'failed',
'dead_letter',
];
] as const;
// Semantic badge styling per stage (token-backed Tailwind colors).
const STAGE_STYLE: Record<string, string> = {
queued: 'bg-muted text-muted-foreground',
blocked: 'bg-amber-500/15 text-amber-700 dark:text-amber-400',
assigned: 'bg-sky-500/15 text-sky-700 dark:text-sky-400',
building: 'bg-blue-500/15 text-blue-700 dark:text-blue-400',
review: 'bg-amber-500/15 text-amber-700 dark:text-amber-400',
testing: 'bg-purple-500/15 text-purple-700 dark:text-purple-400',
shipped: 'bg-green-600/15 text-green-700 dark:text-green-400',
failed: 'bg-destructive/15 text-destructive',
dead_letter: 'bg-destructive/15 text-destructive',
};
const stageStyle = (s: string) => STAGE_STYLE[s] ?? 'bg-muted text-muted-foreground';
const stageLabel = (s: string) => s.replace(/_/g, ' ');
const PRIORITY_STYLE: Record<string, string> = {
critical: 'text-destructive font-semibold',
high: 'text-amber-700 dark:text-amber-400',
medium: 'text-muted-foreground',
low: 'text-muted-foreground',
};
const POLL_INTERVAL = 30_000;
// MVP: PR-mode target repos (local checkouts under the factory's repo base).
@ -59,6 +79,8 @@ export default function FleetJobsPage() {
const { token } = useAuth();
const [jobs, setJobs] = useState<FleetJob[]>([]);
const [stage, setStage] = useState('');
const [hideShipped, setHideShipped] = useState(false);
const [search, setSearch] = useState('');
const [loading, setLoading] = useState(true);
// New-job form state
@ -73,18 +95,18 @@ export default function FleetJobsPage() {
const [submitting, setSubmitting] = useState(false);
const [submitMsg, setSubmitMsg] = useState<{ ok: boolean; text: string } | null>(null);
// Fetch the full recent window once; stage / hide-shipped / search are applied
// client-side so the stage counts stay accurate and filtering is instant.
const refresh = useCallback(async () => {
try {
const params: Record<string, string> = { limit: '50' };
if (stage) params.stage = stage;
const res = await listJobs(params as never);
const res = await listJobs({ limit: '100' } as never);
setJobs(res.jobs);
} catch {
/* degrade */
} finally {
setLoading(false);
}
}, [stage]);
}, []);
useEffect(() => {
if (!token) return;
@ -93,6 +115,33 @@ export default function FleetJobsPage() {
return () => clearInterval(id);
}, [token, refresh]);
// Restore the hide-shipped preference (client-only; avoids hydration mismatch).
useEffect(() => {
setHideShipped(localStorage.getItem('fleet_hide_shipped') === '1');
}, []);
const toggleHideShipped = useCallback((next: boolean) => {
setHideShipped(next);
localStorage.setItem('fleet_hide_shipped', next ? '1' : '0');
}, []);
// Stage counts (over everything fetched) + the filtered, sorted view.
const stageCounts = jobs.reduce<Record<string, number>>((acc, j) => {
acc[j.stage] = (acc[j.stage] ?? 0) + 1;
return acc;
}, {});
const q = search.trim().toLowerCase();
const visible = jobs
.filter(j => {
if (stage && j.stage !== stage) return false;
if (hideShipped && j.stage === 'shipped') return false;
if (q) {
const hay = `${j.idempotencyKey} ${j.repo ?? ''} ${j.id}`.toLowerCase();
if (!hay.includes(q)) return false;
}
return true;
})
.sort((a, b) => Date.parse(b.createdAt) - Date.parse(a.createdAt));
const handleSubmit = useCallback(async () => {
if (!body.trim()) {
setSubmitMsg({ ok: false, text: 'Job body is required.' });
@ -291,64 +340,111 @@ export default function FleetJobsPage() {
</div>
{/* Filters */}
<div className="flex gap-3 items-center">
<label htmlFor="stage-filter" className="text-sm font-medium">
Stage:
</label>
<select
id="stage-filter"
value={stage}
onChange={e => setStage(e.target.value)}
className="rounded border px-2 py-1 text-sm bg-background"
aria-label="Filter by stage"
>
{STAGES.map(s => (
<option key={s} value={s}>
{s || 'All'}
</option>
<div className="space-y-3">
{/* Stage chips with live counts (click to filter) */}
<div className="flex flex-wrap items-center gap-1.5">
<button
type="button"
onClick={() => setStage('')}
className={`rounded-full border px-2.5 py-1 text-xs font-medium ${
stage === '' ? 'bg-primary text-primary-foreground border-primary' : 'hover:bg-muted'
}`}
>
All <span className="opacity-70">{jobs.length}</span>
</button>
{STAGES.filter(s => stageCounts[s]).map(s => (
<button
key={s}
type="button"
onClick={() => setStage(stage === s ? '' : s)}
className={`rounded-full px-2.5 py-1 text-xs font-medium ${stageStyle(s)} ${
stage === s ? 'ring-2 ring-primary ring-offset-1 ring-offset-background' : ''
}`}
>
{stageLabel(s)} <span className="opacity-70">{stageCounts[s]}</span>
</button>
))}
</select>
</div>
{/* Search + hide-shipped */}
<div className="flex flex-wrap items-center gap-4">
<input
type="search"
value={search}
onChange={e => setSearch(e.target.value)}
placeholder="Search key / repo / id…"
className="w-64 rounded border bg-background px-2 py-1 text-sm"
aria-label="Search jobs"
/>
<label className="flex items-center gap-2 text-sm">
<input
type="checkbox"
checked={hideShipped}
onChange={e => toggleHideShipped(e.target.checked)}
aria-label="Hide shipped jobs"
/>
Hide shipped
</label>
<span className="text-xs text-muted-foreground">
{visible.length} of {jobs.length} shown
</span>
</div>
</div>
{/* Table */}
{loading ? (
<p className="text-muted-foreground">Loading jobs...</p>
) : jobs.length === 0 ? (
<p className="text-muted-foreground">No jobs match the current filter.</p>
) : visible.length === 0 ? (
<p className="text-muted-foreground">
{jobs.length === 0 ? 'No jobs yet — submit one above.' : 'No jobs match the filters.'}
</p>
) : (
<div className="overflow-x-auto">
<div className="overflow-x-auto rounded-lg border">
<table className="w-full text-sm" aria-label="Fleet jobs table">
<thead>
<tr className="border-b text-left text-muted-foreground">
<th className="pb-2 pr-4">Idempotency Key</th>
<th className="pb-2 pr-4">Stage</th>
<th className="pb-2 pr-4">Priority</th>
<th className="pb-2 pr-4">Kind</th>
<th className="pb-2 pr-4">Attempts</th>
<th className="pb-2">Created</th>
<tr className="border-b bg-muted/40 text-left text-xs uppercase tracking-wide text-muted-foreground">
<th className="px-3 py-2">Job</th>
<th className="px-3 py-2">Stage</th>
<th className="px-3 py-2">Priority</th>
<th className="px-3 py-2">Repo</th>
<th className="px-3 py-2 text-right">Attempts</th>
<th className="px-3 py-2 text-right">Created</th>
</tr>
</thead>
<tbody>
{jobs.map(j => (
<tr key={j.id} className="border-b last:border-0 hover:bg-muted/50 cursor-pointer">
<td className="py-2 pr-4">
{visible.map(j => (
<tr key={j.id} className="border-b last:border-0 hover:bg-muted/40">
<td className="px-3 py-2">
<Link
href={`/dashboard/fleet/jobs/${j.id}`}
className="hover:underline font-mono text-xs"
className="font-mono text-xs hover:underline"
aria-label={`View job ${j.idempotencyKey}`}
>
{j.idempotencyKey}
</Link>
{j.kind !== 'leaf' && (
<span className="ml-2 text-[10px] text-muted-foreground">{j.kind}</span>
)}
</td>
<td className="py-2 pr-4">
<span className="inline-flex items-center rounded-full px-2 py-0.5 text-xs font-medium bg-muted">
{j.stage}
<td className="px-3 py-2">
<span
className={`inline-flex items-center rounded-full px-2 py-0.5 text-xs font-medium ${stageStyle(j.stage)}`}
>
{stageLabel(j.stage)}
</span>
</td>
<td className="py-2 pr-4 capitalize">{j.priority}</td>
<td className="py-2 pr-4">{j.kind}</td>
<td className="py-2 pr-4">{j.attempts}</td>
<td className="py-2 text-xs text-muted-foreground">
<td className={`px-3 py-2 capitalize ${PRIORITY_STYLE[j.priority] ?? ''}`}>
{j.priority}
</td>
<td className="px-3 py-2 font-mono text-xs">
{j.repo ? (
<span title={`${j.repo}@${j.baseBranch ?? 'main'}`}>{j.repo}</span>
) : (
<span className="text-muted-foreground"></span>
)}
</td>
<td className="px-3 py-2 text-right tabular-nums">{j.attempts}</td>
<td className="px-3 py-2 text-right text-xs text-muted-foreground whitespace-nowrap">
{new Date(j.createdAt).toLocaleString()}
</td>
</tr>

View File

@ -25,6 +25,8 @@ export interface FleetJob {
updatedAt: string;
reviewPolicy?: ReviewPolicy;
reviewDecisions?: ReviewDecision[];
repo?: string;
baseBranch?: string;
}
export interface FleetFactory {