mirror of
https://github.com/alkimake/paperclip.git
synced 2026-06-16 10:50:38 +09:00
## Thinking Path > - Paperclip orchestrates AI agents for zero-human companies. > - Operators need to find work, documents, agents, projects, comments, and activity across a company without jumping through separate surfaces. > - The existing Command-K flow was useful for fast navigation but not enough for deeper company-wide discovery. > - Search also needs company-scoped backend contracts, query cost controls, and indexed document matching so it stays safe as company data grows. > - This pull request adds a full company search API and a dedicated board search page that Command-K can hand off to. > - The benefit is a single searchable control-plane surface with richer result context, recents, highlights, and test coverage across server and UI behavior. ## What Changed - Added a company-scoped search endpoint/service with query validation, rate limiting, text matching, fuzzy title matching, and result typing shared through `@paperclipai/shared`. - Added idempotent search migrations for document search indexes and fuzzy matching support. - Added the full `/companies/:companyKey/search` UI, search result row components, highlighted snippets, recent searches, and sidebar/Command-K handoff. - Added Storybook coverage for search surfaces and Vitest coverage for server search behavior, rate limiting, route generation, Command-K behavior, and the search page. - Addressed Greptile findings by renaming the no-match SQL helper, applying search pagination after cross-type merge sorting, and lazy-initializing the default search service so unrelated route-test mocks do not need to know about it. - Merged current `public-gh/master` and renumbered the search migrations behind upstream `0078_white_darwin`: search indexes are now `0079_company_search_document_indexes` and fuzzy matching is `0080_company_search_fuzzystrmatch`. ## Verification - `git fetch public-gh master` - `git diff --check public-gh/master...HEAD` - `git diff --name-only public-gh/master...HEAD | rg '^pnpm-lock\.yaml$' || true` produced no output before opening the PR. - `pnpm run preflight:workspace-links && pnpm exec vitest run server/src/__tests__/company-search-service.test.ts server/src/__tests__/company-search-rate-limit-routes.test.ts ui/src/pages/Search.test.tsx ui/src/components/CommandPalette.test.tsx ui/src/lib/company-routes.test.ts` passed: 5 files, 25 tests. - `pnpm --filter @paperclipai/shared typecheck && pnpm --filter @paperclipai/db typecheck && pnpm --filter @paperclipai/server typecheck && pnpm --filter @paperclipai/ui typecheck` passed. - `pnpm exec vitest run server/src/__tests__/company-search-service.test.ts server/src/__tests__/company-search-rate-limit-routes.test.ts && pnpm --filter @paperclipai/server typecheck` passed after Greptile pagination fixes. - `pnpm exec vitest run server/src/__tests__/issue-agent-mutation-ownership-routes.test.ts server/src/__tests__/company-search-rate-limit-routes.test.ts server/src/__tests__/company-search-service.test.ts && pnpm --filter @paperclipai/server typecheck` passed after the CI mock fix. - After resolving the migration conflict with current `public-gh/master`: `pnpm --filter @paperclipai/db typecheck && pnpm exec vitest run server/src/__tests__/company-search-service.test.ts server/src/__tests__/company-search-rate-limit-routes.test.ts && pnpm --filter @paperclipai/server typecheck` passed. - DB migration numbering check passed as part of `@paperclipai/db` typecheck. - UI states are covered by the added Storybook stories in `ui/storybook/stories/search.stories.tsx`. - GitHub reports the PR merge state as `CLEAN` on head `18e54fa8`. - GitHub PR checks are green on head `18e54fa8`: policy, verify, serialized server shards 1/4 through 4/4, e2e, canary dry run, Snyk, and Greptile Review. ## Risks - Search ranking and snippets are new user-facing behavior, so reviewers should check whether result ordering feels right on real company data. - Search touches broad company data, so company scoping and query cost/rate-limit behavior should be reviewed carefully. - The migrations add search indexes/extensions; they are idempotent with `IF NOT EXISTS` for users who may have applied an earlier branch migration number. > ROADMAP.md checked. This PR adds a focused board search surface and does not duplicate an open roadmap item. ## Model Used - OpenAI Codex, GPT-5 coding agent, tool-enabled shell/git/GitHub CLI session with medium reasoning effort. Existing branch commits were produced across prior agent sessions; this packaging pass verified, opened the PR, addressed Greptile findings, resolved migration conflicts after upstream PRs landed, and got PR checks green. ## Checklist - [x] I have included a thinking path that traces from project context to this change - [x] I have specified the model used (with version and capability details) - [x] I have checked ROADMAP.md and confirmed this PR does not duplicate planned core work - [x] I have run tests locally and they pass - [x] I have added or updated tests where applicable - [x] If this change affects the UI, I have included before/after screenshots - [x] I have updated relevant documentation to reflect my changes - [x] I have considered and documented any risks above - [x] I will address all Greptile and reviewer comments before requesting merge --------- Co-authored-by: Paperclip <noreply@paperclip.ing> Co-authored-by: Claude Opus 4.7 <noreply@anthropic.com>
618 lines
23 KiB
TypeScript
618 lines
23 KiB
TypeScript
import type { Meta, StoryObj } from "@storybook/react-vite";
|
|
import type { CompanySearchResult, CompanySearchResponse } from "@paperclipai/shared";
|
|
import { Badge } from "@/components/ui/badge";
|
|
import { IssueGroupHeader } from "@/components/IssueGroupHeader";
|
|
import { Input } from "@/components/ui/input";
|
|
import { PageTabBar, type PageTabItem } from "@/components/PageTabBar";
|
|
import { MatchSourceChip } from "@/components/search/MatchSourceChip";
|
|
import { SearchResultRow } from "@/components/search/SearchResultRow";
|
|
import { Tabs } from "@/components/ui/tabs";
|
|
import {
|
|
Bot,
|
|
CircleDot,
|
|
DollarSign,
|
|
Hexagon,
|
|
History,
|
|
Inbox,
|
|
LayoutDashboard,
|
|
Plus,
|
|
Search as SearchIcon,
|
|
SquarePen,
|
|
Target,
|
|
} from "lucide-react";
|
|
import {
|
|
Command,
|
|
CommandEmpty,
|
|
CommandGroup,
|
|
CommandInput,
|
|
CommandItem,
|
|
CommandList,
|
|
CommandSeparator,
|
|
} from "@/components/ui/command";
|
|
import { StatusBadge } from "@/components/StatusBadge";
|
|
import { storybookAgents, storybookProjects, storybookIssues } from "../fixtures/paperclipData";
|
|
|
|
const agentsById = new Map(storybookAgents.map((agent) => [agent.id, agent]));
|
|
|
|
type IssueResultOverrides = Omit<Partial<CompanySearchResult>, "issue"> & {
|
|
issue?: Partial<NonNullable<CompanySearchResult["issue"]>>;
|
|
};
|
|
|
|
function buildIssueResult(overrides: IssueResultOverrides): CompanySearchResult {
|
|
const baseIssue = {
|
|
id: overrides.issue?.id ?? "issue-1",
|
|
identifier: overrides.issue?.identifier ?? "PAP-3142",
|
|
title: overrides.issue?.title ?? "Auth middleware flakes on cold-start when session token is rotated",
|
|
status: overrides.issue?.status ?? "in_progress",
|
|
priority: overrides.issue?.priority ?? "high",
|
|
assigneeAgentId: overrides.issue?.assigneeAgentId ?? storybookAgents[0]?.id ?? null,
|
|
assigneeUserId: overrides.issue?.assigneeUserId ?? null,
|
|
projectId: overrides.issue?.projectId ?? storybookProjects[0]?.id ?? null,
|
|
updatedAt: overrides.issue?.updatedAt ?? new Date(Date.now() - 1000 * 60 * 60 * 3).toISOString(),
|
|
} satisfies NonNullable<CompanySearchResult["issue"]>;
|
|
return {
|
|
id: overrides.id ?? baseIssue.id,
|
|
type: "issue",
|
|
score: 100,
|
|
title: `${baseIssue.identifier} ${baseIssue.title}`,
|
|
href: `/PAP/issues/${baseIssue.identifier}`,
|
|
matchedFields: overrides.matchedFields ?? ["title"],
|
|
sourceLabel: overrides.sourceLabel ?? null,
|
|
snippet: overrides.snippet ?? null,
|
|
snippets: overrides.snippets ?? [],
|
|
issue: baseIssue,
|
|
updatedAt: baseIssue.updatedAt,
|
|
previewImageUrl: overrides.previewImageUrl ?? null,
|
|
};
|
|
}
|
|
|
|
const fixtureResults: CompanySearchResult[] = [
|
|
buildIssueResult({
|
|
id: "issue-1",
|
|
matchedFields: ["title", "comment"],
|
|
sourceLabel: "Comment",
|
|
snippet: "we hit another flake in the morning batch — auth middleware",
|
|
snippets: [
|
|
{
|
|
field: "title",
|
|
label: "Title",
|
|
text: "Auth middleware flakes on cold-start when session token is rotated",
|
|
highlights: [{ start: 0, end: 4 }],
|
|
},
|
|
{
|
|
field: "comment",
|
|
label: "Comment",
|
|
text: "we hit another flake in the morning batch — auth middleware ate the request",
|
|
highlights: [{ start: 16, end: 21 }, { start: 47, end: 51 }],
|
|
},
|
|
],
|
|
}),
|
|
buildIssueResult({
|
|
id: "issue-2",
|
|
issue: {
|
|
id: "issue-2",
|
|
identifier: "PAP-3091",
|
|
title: "Audit auth flake telemetry from last quarter",
|
|
status: "in_review",
|
|
assigneeAgentId: storybookAgents[1]?.id ?? null,
|
|
},
|
|
matchedFields: ["title", "document"],
|
|
sourceLabel: "Document",
|
|
snippet: "the deflake plan ranks auth regressions above latency tickets",
|
|
snippets: [
|
|
{
|
|
field: "title",
|
|
label: "Title",
|
|
text: "Audit auth flake telemetry from last quarter",
|
|
highlights: [{ start: 6, end: 10 }],
|
|
},
|
|
{
|
|
field: "document",
|
|
label: "PLAN",
|
|
text: "the deflake plan ranks auth regressions above latency tickets",
|
|
highlights: [{ start: 12, end: 16 }, { start: 26, end: 30 }],
|
|
},
|
|
],
|
|
previewImageUrl:
|
|
"data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 100 100'><rect width='100' height='100' fill='%23a78bfa'/><text x='50' y='55' font-size='14' fill='white' text-anchor='middle' font-family='sans-serif'>chart</text></svg>",
|
|
}),
|
|
buildIssueResult({
|
|
id: "issue-3",
|
|
issue: {
|
|
id: "issue-3",
|
|
identifier: "PAP-2748",
|
|
title: "Pin worker registration to a single auth backend",
|
|
status: "done",
|
|
assigneeAgentId: null,
|
|
},
|
|
matchedFields: ["title", "identifier"],
|
|
snippets: [
|
|
{
|
|
field: "title",
|
|
label: "Title",
|
|
text: "Pin worker registration to a single auth backend",
|
|
highlights: [{ start: 36, end: 40 }],
|
|
},
|
|
],
|
|
}),
|
|
];
|
|
|
|
const fixtureAgents: CompanySearchResult[] = storybookAgents.slice(0, 1).map((agent) => ({
|
|
id: agent.id,
|
|
type: "agent" as const,
|
|
score: 80,
|
|
title: agent.name,
|
|
href: `/PAP/agents/${agent.id}`,
|
|
matchedFields: ["agent"],
|
|
sourceLabel: "Agent",
|
|
snippet: agent.capabilities ?? null,
|
|
snippets: agent.capabilities
|
|
? [
|
|
{
|
|
field: "capabilities",
|
|
label: "Agent",
|
|
text: agent.capabilities,
|
|
highlights: [],
|
|
},
|
|
]
|
|
: [],
|
|
updatedAt: new Date().toISOString(),
|
|
previewImageUrl: null,
|
|
}));
|
|
|
|
const fixtureProjects: CompanySearchResult[] = storybookProjects.slice(0, 1).map((project) => ({
|
|
id: project.id,
|
|
type: "project" as const,
|
|
score: 70,
|
|
title: project.name,
|
|
href: `/PAP/projects/${project.id}`,
|
|
matchedFields: ["project"],
|
|
sourceLabel: "Project",
|
|
snippet: project.description ?? null,
|
|
snippets: project.description
|
|
? [
|
|
{
|
|
field: "description",
|
|
label: "Project",
|
|
text: project.description,
|
|
highlights: [],
|
|
},
|
|
]
|
|
: [],
|
|
updatedAt: new Date().toISOString(),
|
|
previewImageUrl: null,
|
|
}));
|
|
|
|
const fixtureResponse: CompanySearchResponse = {
|
|
query: "auth flake",
|
|
normalizedQuery: "auth flake",
|
|
scope: "all",
|
|
limit: 20,
|
|
offset: 0,
|
|
results: [...fixtureResults, ...fixtureAgents, ...fixtureProjects],
|
|
countsByType: {
|
|
issue: fixtureResults.length,
|
|
agent: fixtureAgents.length,
|
|
project: fixtureProjects.length,
|
|
},
|
|
hasMore: false,
|
|
};
|
|
|
|
function ScopeTabsPreview({
|
|
active,
|
|
response,
|
|
}: {
|
|
active: "all" | "issues" | "comments" | "documents" | "agents" | "projects";
|
|
response: CompanySearchResponse;
|
|
}) {
|
|
const total =
|
|
(response.countsByType.issue ?? 0) +
|
|
(response.countsByType.agent ?? 0) +
|
|
(response.countsByType.project ?? 0);
|
|
const items: PageTabItem[] = [
|
|
{ value: "all", label: <ScopeTabLabel label="All" count={total} /> },
|
|
{ value: "issues", label: <ScopeTabLabel label="Issues" count={response.countsByType.issue} /> },
|
|
{ value: "comments", label: <ScopeTabLabel label="Comments" count={response.results.filter((result) => result.matchedFields.includes("comment")).length} /> },
|
|
{ value: "documents", label: <ScopeTabLabel label="Documents" count={response.results.filter((result) => result.matchedFields.includes("document")).length} /> },
|
|
{ value: "agents", label: <ScopeTabLabel label="Agents" count={response.countsByType.agent} /> },
|
|
{ value: "projects", label: <ScopeTabLabel label="Projects" count={response.countsByType.project} /> },
|
|
];
|
|
return (
|
|
<Tabs value={active}>
|
|
<PageTabBar items={items} value={active} align="start" />
|
|
</Tabs>
|
|
);
|
|
}
|
|
|
|
function ScopeTabLabel({ label, count }: { label: string; count: number }) {
|
|
return (
|
|
<span className="flex items-center">
|
|
{label}
|
|
<Badge variant="outline" className="ml-1.5 px-1.5 py-0 text-[10px] tabular-nums font-normal">
|
|
{count}
|
|
</Badge>
|
|
</span>
|
|
);
|
|
}
|
|
|
|
function SearchPagePreview({
|
|
response,
|
|
state,
|
|
query,
|
|
}: {
|
|
response: CompanySearchResponse;
|
|
state: "results" | "empty" | "loading" | "initial";
|
|
query: string;
|
|
}) {
|
|
return (
|
|
<div className="flex h-full min-h-0 flex-col rounded-md border border-border bg-background">
|
|
<div className="border-b border-border px-4 py-3">
|
|
<div className="relative">
|
|
<SearchIcon className="pointer-events-none absolute left-3 top-1/2 h-4 w-4 -translate-y-1/2 text-muted-foreground" />
|
|
<Input
|
|
value={query}
|
|
readOnly
|
|
placeholder="Search issues, comments, documents, agents, projects…"
|
|
className="h-10 pl-9 pr-20 text-sm"
|
|
aria-label="Search query"
|
|
/>
|
|
<kbd className="pointer-events-none absolute right-3 top-1/2 -translate-y-1/2 rounded border border-border bg-muted px-1.5 py-0.5 text-[10px] font-medium text-muted-foreground">
|
|
⌘K
|
|
</kbd>
|
|
</div>
|
|
</div>
|
|
<div className="border-b border-border px-2 sm:px-4">
|
|
<ScopeTabsPreview active="all" response={response} />
|
|
</div>
|
|
|
|
{state === "results" ? (
|
|
<div className="flex w-full max-w-[960px] flex-col px-2 sm:px-4">
|
|
<div className="flex items-center justify-between py-2 text-[11px] uppercase tracking-wide text-muted-foreground">
|
|
<span>{response.results.length} results · sorted by relevance</span>
|
|
</div>
|
|
<section aria-label="Issues" className="flex flex-col">
|
|
<IssueGroupHeader
|
|
label="Issues"
|
|
trailing={
|
|
<span className="text-xs font-normal tabular-nums text-muted-foreground">
|
|
{fixtureResults.length}
|
|
</span>
|
|
}
|
|
className="pt-2 pb-1 text-[11px] tracking-wider text-muted-foreground"
|
|
/>
|
|
<div className="flex flex-col gap-y-1">
|
|
{fixtureResults.map((result) => (
|
|
<SearchResultRow
|
|
key={result.id}
|
|
result={result}
|
|
agentsById={agentsById}
|
|
/>
|
|
))}
|
|
</div>
|
|
</section>
|
|
<section aria-label="Agents" className="mt-6 flex flex-col">
|
|
<IssueGroupHeader
|
|
label="Agents"
|
|
trailing={
|
|
<span className="text-xs font-normal tabular-nums text-muted-foreground">
|
|
{fixtureAgents.length}
|
|
</span>
|
|
}
|
|
className="pt-2 pb-1 text-[11px] tracking-wider text-muted-foreground"
|
|
/>
|
|
<div className="flex flex-col gap-y-1">
|
|
{fixtureAgents.map((result) => (
|
|
<SearchResultRow key={result.id} result={result} />
|
|
))}
|
|
</div>
|
|
</section>
|
|
<section aria-label="Projects" className="mt-6 flex flex-col">
|
|
<IssueGroupHeader
|
|
label="Projects"
|
|
trailing={
|
|
<span className="text-xs font-normal tabular-nums text-muted-foreground">
|
|
{fixtureProjects.length}
|
|
</span>
|
|
}
|
|
className="pt-2 pb-1 text-[11px] tracking-wider text-muted-foreground"
|
|
/>
|
|
<div className="flex flex-col gap-y-1">
|
|
{fixtureProjects.map((result) => (
|
|
<SearchResultRow key={result.id} result={result} />
|
|
))}
|
|
</div>
|
|
</section>
|
|
</div>
|
|
) : null}
|
|
|
|
{state === "empty" ? (
|
|
<div className="mx-auto flex w-full max-w-xl flex-col items-center justify-center gap-3 px-4 py-12 text-center">
|
|
<div className="text-base font-semibold">
|
|
No results for “{query}”
|
|
</div>
|
|
<p className="text-sm text-muted-foreground">
|
|
We couldn’t find a match in all scopes. Try widening the scope or rephrasing your query.
|
|
</p>
|
|
<ul className="mt-2 space-y-0.5 text-xs text-muted-foreground">
|
|
<li>Try fewer tokens or a single distinctive term.</li>
|
|
<li>Use an identifier shortcut like <code className="rounded bg-muted px-1 py-0.5">PAP-123</code>.</li>
|
|
</ul>
|
|
</div>
|
|
) : null}
|
|
|
|
{state === "loading" ? (
|
|
<div className="flex flex-col gap-2 px-2 py-3 sm:px-4">
|
|
<div className="px-3 text-xs text-muted-foreground">Searching for “{query}”…</div>
|
|
<div className="flex flex-col">
|
|
{Array.from({ length: 5 }).map((_, index) => (
|
|
<div key={index} className="flex items-start gap-3 px-3 py-2">
|
|
<div className="mt-1 h-4 w-4 rounded-full bg-muted" />
|
|
<div className="flex flex-1 flex-col gap-1.5">
|
|
<div className="h-3 w-3/4 rounded bg-muted" />
|
|
<div className="h-3 w-1/2 rounded bg-muted/60" />
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
) : null}
|
|
|
|
{state === "initial" ? (
|
|
<div className="mx-auto flex w-full max-w-2xl flex-col gap-4 px-4 py-10 sm:px-6">
|
|
<div>
|
|
<h2 className="text-lg font-semibold">Type to search company memory.</h2>
|
|
<p className="mt-1 text-sm text-muted-foreground">
|
|
Issues, comments, plan documents, agents, projects — same surface, ranked by relevance.
|
|
</p>
|
|
</div>
|
|
<ul className="space-y-1 text-xs text-muted-foreground">
|
|
<li>
|
|
<span className="font-medium text-foreground">Identifier lookup:</span> type{" "}
|
|
<code className="rounded bg-muted px-1 py-0.5 text-[11px]">PAP-123</code> to jump straight to an issue.
|
|
</li>
|
|
<li>
|
|
<span className="font-medium text-foreground">Quoted phrases:</span> wrap a phrase in quotes to match the
|
|
exact sequence.
|
|
</li>
|
|
<li>
|
|
<span className="font-medium text-foreground">⌘K:</span> reopens the command palette pre-seeded with your
|
|
current query.
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
) : null}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function CommandPaletteWithSearchAll({
|
|
query,
|
|
emptyResults = false,
|
|
}: {
|
|
query: string;
|
|
emptyResults?: boolean;
|
|
}) {
|
|
return (
|
|
<Command className="rounded-md border border-border bg-popover shadow-lg">
|
|
<CommandInput value={query} readOnly placeholder="Search issues, agents, projects..." />
|
|
<CommandList className="max-h-none">
|
|
{emptyResults ? (
|
|
<CommandEmpty>
|
|
<span>
|
|
No quick issue matches. Press{" "}
|
|
<kbd className="rounded border border-border bg-muted px-1 py-0.5 text-[10px]">↵</kbd>{" "}
|
|
to <span className="font-medium">search all</span> or keep typing to refine.
|
|
</span>
|
|
</CommandEmpty>
|
|
) : null}
|
|
<CommandGroup heading="Search">
|
|
<CommandItem
|
|
value="search-all"
|
|
className="bg-accent/40 border border-accent data-[selected=true]:bg-accent/60"
|
|
>
|
|
<SearchIcon className="mr-2 h-4 w-4" />
|
|
<span className="flex-1 truncate">
|
|
Search all for <span className="font-semibold">“{query}”</span>
|
|
</span>
|
|
<span className="ml-auto inline-flex items-center gap-1 text-xs text-muted-foreground">
|
|
<span>open full search</span>
|
|
<kbd className="rounded border border-border bg-background px-1 py-0.5 text-[10px]">↵</kbd>
|
|
</span>
|
|
</CommandItem>
|
|
</CommandGroup>
|
|
<CommandSeparator />
|
|
<CommandGroup heading="Actions">
|
|
<CommandItem>
|
|
<SquarePen className="mr-2 h-4 w-4" />
|
|
Create new issue
|
|
<span className="ml-auto text-xs text-muted-foreground">C</span>
|
|
</CommandItem>
|
|
<CommandItem>
|
|
<Plus className="mr-2 h-4 w-4" />
|
|
Create new agent
|
|
</CommandItem>
|
|
</CommandGroup>
|
|
<CommandSeparator />
|
|
<CommandGroup heading="Pages">
|
|
<CommandItem>
|
|
<LayoutDashboard className="mr-2 h-4 w-4" />
|
|
Dashboard
|
|
</CommandItem>
|
|
<CommandItem>
|
|
<Inbox className="mr-2 h-4 w-4" />
|
|
Inbox
|
|
</CommandItem>
|
|
<CommandItem>
|
|
<CircleDot className="mr-2 h-4 w-4" />
|
|
Issues
|
|
</CommandItem>
|
|
<CommandItem>
|
|
<Target className="mr-2 h-4 w-4" />
|
|
Goals
|
|
</CommandItem>
|
|
<CommandItem>
|
|
<Bot className="mr-2 h-4 w-4" />
|
|
Agents
|
|
</CommandItem>
|
|
<CommandItem>
|
|
<DollarSign className="mr-2 h-4 w-4" />
|
|
Costs
|
|
</CommandItem>
|
|
<CommandItem>
|
|
<History className="mr-2 h-4 w-4" />
|
|
Activity
|
|
</CommandItem>
|
|
</CommandGroup>
|
|
{!emptyResults ? (
|
|
<>
|
|
<CommandSeparator />
|
|
<CommandGroup heading="Issues">
|
|
{storybookIssues.slice(0, 3).map((issue) => (
|
|
<CommandItem key={issue.id}>
|
|
<CircleDot className="mr-2 h-4 w-4" />
|
|
<span className="mr-2 font-mono text-xs text-muted-foreground">{issue.identifier}</span>
|
|
<span className="flex-1 truncate">{issue.title}</span>
|
|
<StatusBadge status={issue.status} />
|
|
</CommandItem>
|
|
))}
|
|
</CommandGroup>
|
|
</>
|
|
) : null}
|
|
<CommandSeparator />
|
|
<CommandGroup heading="Projects">
|
|
{storybookProjects.slice(0, 2).map((project) => (
|
|
<CommandItem key={project.id}>
|
|
<Hexagon className="mr-2 h-4 w-4" />
|
|
{project.name}
|
|
</CommandItem>
|
|
))}
|
|
</CommandGroup>
|
|
</CommandList>
|
|
</Command>
|
|
);
|
|
}
|
|
|
|
function SearchStories() {
|
|
return (
|
|
<div className="paperclip-story">
|
|
<main className="paperclip-story__inner max-w-[1320px] space-y-6">
|
|
<section className="paperclip-story__frame p-6">
|
|
<div className="paperclip-story__label">Search</div>
|
|
<h1 className="mt-2 text-3xl font-semibold tracking-tight">Full search page and Command K handoff</h1>
|
|
<p className="mt-3 max-w-3xl text-sm leading-6 text-muted-foreground">
|
|
Snippet-forward results, scope tabs, match-source chips, and the supporting empty / loading / initial
|
|
states. Cmd K palette renders the persistent “Search all for…” row when a query is non-empty.
|
|
</p>
|
|
</section>
|
|
|
|
<section className="paperclip-story__frame overflow-hidden">
|
|
<div className="paperclip-story__title-block">
|
|
<div className="paperclip-story__label">/search</div>
|
|
<h2 className="mt-1 text-lg font-semibold">Results, query “auth flake”</h2>
|
|
</div>
|
|
<SearchPagePreview response={fixtureResponse} state="results" query="auth flake" />
|
|
</section>
|
|
|
|
<section className="paperclip-story__frame overflow-hidden">
|
|
<div className="paperclip-story__title-block">
|
|
<div className="paperclip-story__label">/search</div>
|
|
<h2 className="mt-1 text-lg font-semibold">Initial state — no query</h2>
|
|
</div>
|
|
<SearchPagePreview response={fixtureResponse} state="initial" query="" />
|
|
</section>
|
|
|
|
<section className="paperclip-story__frame overflow-hidden">
|
|
<div className="paperclip-story__title-block">
|
|
<div className="paperclip-story__label">/search</div>
|
|
<h2 className="mt-1 text-lg font-semibold">Loading skeleton</h2>
|
|
</div>
|
|
<SearchPagePreview response={fixtureResponse} state="loading" query="auth flake" />
|
|
</section>
|
|
|
|
<section className="paperclip-story__frame overflow-hidden">
|
|
<div className="paperclip-story__title-block">
|
|
<div className="paperclip-story__label">/search</div>
|
|
<h2 className="mt-1 text-lg font-semibold">No results state</h2>
|
|
</div>
|
|
<SearchPagePreview response={{ ...fixtureResponse, results: [], countsByType: { issue: 0, agent: 0, project: 0 } }} state="empty" query="ghostbuster" />
|
|
</section>
|
|
|
|
<section className="paperclip-story__frame overflow-hidden p-4">
|
|
<div className="paperclip-story__title-block">
|
|
<div className="paperclip-story__label">Match-source chips</div>
|
|
<h2 className="mt-1 text-lg font-semibold">Type-coded chip variants</h2>
|
|
</div>
|
|
<div className="flex flex-wrap items-center gap-2 p-2">
|
|
<MatchSourceChip kind="title" />
|
|
<MatchSourceChip kind="identifier" />
|
|
<MatchSourceChip kind="comment" count={3} />
|
|
<MatchSourceChip kind="document" />
|
|
<MatchSourceChip kind="document" count={2} label="Doc" />
|
|
</div>
|
|
</section>
|
|
|
|
<section className="paperclip-story__frame overflow-hidden p-4">
|
|
<div className="paperclip-story__title-block">
|
|
<div className="paperclip-story__label">Cmd+K palette</div>
|
|
<h2 className="mt-1 text-lg font-semibold">Search-all row with quick results</h2>
|
|
</div>
|
|
<div className="grid gap-4 lg:grid-cols-2">
|
|
<div>
|
|
<div className="mb-2 text-[11px] uppercase tracking-wide text-muted-foreground">
|
|
With quick issue matches
|
|
</div>
|
|
<CommandPaletteWithSearchAll query="auth flake" />
|
|
</div>
|
|
<div>
|
|
<div className="mb-2 text-[11px] uppercase tracking-wide text-muted-foreground">
|
|
Empty results — Enter routes to /search
|
|
</div>
|
|
<CommandPaletteWithSearchAll query="ghostbuster" emptyResults />
|
|
</div>
|
|
</div>
|
|
</section>
|
|
|
|
<section className="paperclip-story__frame overflow-hidden p-4">
|
|
<div className="paperclip-story__title-block">
|
|
<div className="paperclip-story__label">Search result row</div>
|
|
<h2 className="mt-1 text-lg font-semibold">Issue, agent, project rows</h2>
|
|
</div>
|
|
<div className="flex w-full max-w-[960px] flex-col gap-y-1">
|
|
{fixtureResults.map((result) => (
|
|
<SearchResultRow
|
|
key={result.id}
|
|
result={result}
|
|
agentsById={agentsById}
|
|
/>
|
|
))}
|
|
{fixtureAgents.map((result) => (
|
|
<SearchResultRow key={result.id} result={result} />
|
|
))}
|
|
{fixtureProjects.map((result) => (
|
|
<SearchResultRow key={result.id} result={result} />
|
|
))}
|
|
</div>
|
|
</section>
|
|
</main>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const meta = {
|
|
title: "Product/Search & Command K",
|
|
component: SearchStories,
|
|
parameters: {
|
|
docs: {
|
|
description: {
|
|
component:
|
|
"Full search page surfaces and Command K Search-all handoff. Reuses StatusIcon, StatusBadge, Identity, IssueGroupHeader, and PageTabBar; adds MatchSourceChip + SearchResultRow.",
|
|
},
|
|
},
|
|
},
|
|
} satisfies Meta<typeof SearchStories>;
|
|
|
|
export default meta;
|
|
|
|
type Story = StoryObj<typeof meta>;
|
|
|
|
export const SearchSurfaces: Story = {};
|