2de893f624
## Thinking Path > - Paperclip orchestrates AI agents for zero-human companies. > - The board UI is the main operator surface, so its component and workflow coverage needs to stay reviewable as the product grows. > - This branch adds Storybook as a dedicated UI reference surface for core Paperclip screens and interaction patterns. > - That work spans Storybook infrastructure, app-level provider wiring, and a large fixture set that can render real control-plane states without a live backend. > - The branch also expands coverage across agents, budgets, issues, chat, dialogs, navigation, projects, and data visualization so future UI changes have a concrete visual baseline. > - This pull request packages that Storybook work on top of the latest `master`, excludes the lockfile from the final diff per repo policy, and fixes one fixture contract drift caught during verification. > - The benefit is a single reviewable PR that adds broad UI documentation and regression-surfacing coverage without losing the existing branch work. ## What Changed - Added Storybook 10 wiring for the UI package, including root scripts, UI package scripts, Storybook config, preview wrappers, Tailwind entrypoints, and setup docs. - Added a large fixture-backed data source for Storybook so complex board states can render without a live server. - Added story suites covering foundations, status language, control-plane surfaces, overview, UX labs, agent management, budget and finance, forms and editors, issue management, navigation and layout, chat and comments, data visualization, dialogs and modals, and projects/goals/workspaces. - Adjusted several UI components for Storybook parity so dialogs, menus, keyboard shortcuts, budget markers, markdown editing, and related surfaces render correctly in isolation. - Rebasing work for PR assembly: replayed the branch onto current `master`, removed `pnpm-lock.yaml` from the final PR diff, and aligned the dashboard fixture with the current `DashboardSummary.runActivity` API contract. ## Verification - `pnpm --filter @paperclipai/ui typecheck` - `pnpm --filter @paperclipai/ui build-storybook` - Manual diff audit after rebase: verified the PR no longer includes `pnpm-lock.yaml` and now cleanly targets current `master`. - Before/after UI note: before this branch there was no dedicated Storybook surface for these Paperclip views; after this branch the local Storybook build includes the new overview and domain story suites in `ui/storybook-static`. ## Risks - Large static fixture files can drift from shared types as dashboard and UI contracts evolve; this PR already needed one fixture correction for `runActivity`. - Storybook bundle output includes some large chunks, so future growth may need chunking work if build performance becomes an issue. - Several component tweaks were made for isolated rendering parity, so reviewers should spot-check key board surfaces against the live app behavior. ## Model Used - OpenAI Codex, GPT-5-based coding agent in the Paperclip harness; exact serving model ID is not exposed in-runtime to the agent. - Tool-assisted workflow with terminal execution, git operations, local typecheck/build verification, and GitHub CLI PR creation. - Context window/reasoning mode not surfaced by the harness. ## 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 - [ ] 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>
517 lines
20 KiB
TypeScript
517 lines
20 KiB
TypeScript
import { useMemo, useState, type ReactNode } from "react";
|
|
import type { Meta, StoryObj } from "@storybook/react-vite";
|
|
import { useQueryClient } from "@tanstack/react-query";
|
|
import type { Goal, Project } from "@paperclipai/shared";
|
|
import { Archive, Boxes, FolderGit2, GitBranch, Network, Play, RotateCcw, Square } from "lucide-react";
|
|
import { GoalProperties } from "@/components/GoalProperties";
|
|
import { GoalTree } from "@/components/GoalTree";
|
|
import { ProjectProperties, type ProjectConfigFieldKey, type ProjectFieldSaveState } from "@/components/ProjectProperties";
|
|
import { ProjectWorkspacesContent } from "@/components/ProjectWorkspacesContent";
|
|
import { ProjectWorkspaceSummaryCard } from "@/components/ProjectWorkspaceSummaryCard";
|
|
import {
|
|
WorkspaceRuntimeControls,
|
|
buildWorkspaceRuntimeControlSections,
|
|
type WorkspaceRuntimeControlRequest,
|
|
} from "@/components/WorkspaceRuntimeControls";
|
|
import { WorktreeBanner } from "@/components/WorktreeBanner";
|
|
import { Badge } from "@/components/ui/badge";
|
|
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card";
|
|
import { queryKeys } from "@/lib/queryKeys";
|
|
import { buildProjectWorkspaceSummaries } from "@/lib/project-workspaces-tab";
|
|
import {
|
|
storybookAgents,
|
|
storybookAuthSession,
|
|
storybookCompanies,
|
|
storybookExecutionWorkspaces,
|
|
storybookGoals,
|
|
storybookIssues,
|
|
storybookProjectWorkspaces,
|
|
storybookProjects,
|
|
} from "../fixtures/paperclipData";
|
|
|
|
const COMPANY_ID = "company-storybook";
|
|
const boardProject = storybookProjects.find((project) => project.id === "project-board-ui") ?? storybookProjects[0]!;
|
|
const archivedProject =
|
|
storybookProjects.find((project) => project.id === "project-archived-import")
|
|
?? storybookProjects[storybookProjects.length - 1]!;
|
|
|
|
const goalProgress = new Map<string, number>([
|
|
["goal-company", 62],
|
|
["goal-board-ux", 74],
|
|
["goal-agent-runtime", 48],
|
|
["goal-storybook", 88],
|
|
["goal-budget-safety", 100],
|
|
["goal-archived-import", 18],
|
|
]);
|
|
|
|
function Section({
|
|
eyebrow,
|
|
title,
|
|
children,
|
|
}: {
|
|
eyebrow: string;
|
|
title: string;
|
|
children: ReactNode;
|
|
}) {
|
|
return (
|
|
<section className="paperclip-story__frame overflow-hidden">
|
|
<div className="border-b border-border px-5 py-4">
|
|
<div className="paperclip-story__label">{eyebrow}</div>
|
|
<h2 className="mt-1 text-xl font-semibold">{title}</h2>
|
|
</div>
|
|
<div className="p-5">{children}</div>
|
|
</section>
|
|
);
|
|
}
|
|
|
|
function hydrateStorybookQueries(queryClient: ReturnType<typeof useQueryClient>) {
|
|
queryClient.setQueryData(queryKeys.auth.session, storybookAuthSession);
|
|
queryClient.setQueryData(queryKeys.companies.all, storybookCompanies);
|
|
queryClient.setQueryData(queryKeys.agents.list(COMPANY_ID), storybookAgents);
|
|
queryClient.setQueryData(queryKeys.projects.list(COMPANY_ID), storybookProjects);
|
|
queryClient.setQueryData(queryKeys.projects.detail(boardProject.id), boardProject);
|
|
queryClient.setQueryData(queryKeys.projects.detail(boardProject.urlKey), boardProject);
|
|
queryClient.setQueryData(queryKeys.projects.detail(archivedProject.id), archivedProject);
|
|
queryClient.setQueryData(queryKeys.goals.list(COMPANY_ID), storybookGoals);
|
|
for (const goal of storybookGoals) {
|
|
queryClient.setQueryData(queryKeys.goals.detail(goal.id), goal);
|
|
}
|
|
queryClient.setQueryData(queryKeys.issues.list(COMPANY_ID), storybookIssues);
|
|
queryClient.setQueryData(queryKeys.issues.listByProject(COMPANY_ID, boardProject.id), storybookIssues);
|
|
queryClient.setQueryData(queryKeys.secrets.list(COMPANY_ID), []);
|
|
queryClient.setQueryData(queryKeys.instance.experimentalSettings, {
|
|
enableIsolatedWorkspaces: true,
|
|
enableRoutineTriggers: true,
|
|
});
|
|
queryClient.setQueryData(queryKeys.executionWorkspaces.list(COMPANY_ID), storybookExecutionWorkspaces);
|
|
queryClient.setQueryData(
|
|
queryKeys.executionWorkspaces.list(COMPANY_ID, { projectId: boardProject.id }),
|
|
storybookExecutionWorkspaces,
|
|
);
|
|
queryClient.setQueryData(
|
|
queryKeys.executionWorkspaces.summaryList(COMPANY_ID),
|
|
storybookExecutionWorkspaces.map((workspace) => ({
|
|
id: workspace.id,
|
|
name: workspace.name,
|
|
mode: workspace.mode,
|
|
projectWorkspaceId: workspace.projectWorkspaceId,
|
|
})),
|
|
);
|
|
}
|
|
|
|
function StorybookData({ children }: { children: ReactNode }) {
|
|
const queryClient = useQueryClient();
|
|
const [ready] = useState(() => {
|
|
hydrateStorybookQueries(queryClient);
|
|
return true;
|
|
});
|
|
|
|
return ready ? children : null;
|
|
}
|
|
|
|
function stateForProjectField(field: ProjectConfigFieldKey): ProjectFieldSaveState {
|
|
if (field === "env" || field === "execution_workspace_branch_template") return "saved";
|
|
if (field === "execution_workspace_worktree_parent_dir") return "saving";
|
|
return "idle";
|
|
}
|
|
|
|
function ProjectPropertiesMatrix() {
|
|
const editableProject: Project = useMemo(
|
|
() => ({
|
|
...boardProject,
|
|
env: {
|
|
STORYBOOK_REVIEW: { type: "plain", value: "enabled" },
|
|
OPENAI_API_KEY: { type: "secret_ref", secretId: "secret-openai", version: "latest" },
|
|
},
|
|
}),
|
|
[],
|
|
);
|
|
|
|
return (
|
|
<div className="grid gap-5 xl:grid-cols-[minmax(0,1fr)_420px]">
|
|
<div className="rounded-lg border border-border bg-background p-4">
|
|
<ProjectProperties
|
|
project={editableProject}
|
|
onFieldUpdate={() => undefined}
|
|
getFieldSaveState={stateForProjectField}
|
|
onArchive={() => undefined}
|
|
/>
|
|
</div>
|
|
<div className="space-y-4">
|
|
<div className="rounded-lg border border-border bg-background p-4">
|
|
<div className="mb-3 flex items-center justify-between gap-3">
|
|
<div>
|
|
<div className="text-sm font-medium">{archivedProject.name}</div>
|
|
<div className="text-xs text-muted-foreground">Archived, no workspace configured</div>
|
|
</div>
|
|
<Badge variant="outline" className="gap-1">
|
|
<Archive className="h-3 w-3" />
|
|
archived
|
|
</Badge>
|
|
</div>
|
|
<ProjectProperties
|
|
project={archivedProject}
|
|
onFieldUpdate={() => undefined}
|
|
onArchive={() => undefined}
|
|
archivePending={false}
|
|
/>
|
|
</div>
|
|
<div className="grid gap-3 sm:grid-cols-3 xl:grid-cols-1">
|
|
{[
|
|
{ label: "Goals linked", value: boardProject.goalIds.length, icon: Network },
|
|
{ label: "Workspaces", value: boardProject.workspaces.length, icon: Boxes },
|
|
{ label: "Runtime services", value: boardProject.primaryWorkspace?.runtimeServices?.length ?? 0, icon: Play },
|
|
].map((item) => {
|
|
const Icon = item.icon;
|
|
return (
|
|
<div key={item.label} className="rounded-lg border border-border bg-background p-4">
|
|
<Icon className="h-4 w-4 text-muted-foreground" />
|
|
<div className="mt-3 text-2xl font-semibold">{item.value}</div>
|
|
<div className="text-xs text-muted-foreground">{item.label}</div>
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function WorkspacesMatrix() {
|
|
const summaries = buildProjectWorkspaceSummaries({
|
|
project: boardProject,
|
|
issues: storybookIssues.filter((issue) => issue.projectId === boardProject.id),
|
|
executionWorkspaces: storybookExecutionWorkspaces,
|
|
});
|
|
const localSummary = summaries.find((summary) => summary.kind === "project_workspace" && summary.workspaceId === "workspace-board-ui");
|
|
const remoteSummary = summaries.find((summary) => summary.workspaceId === "workspace-docs-remote");
|
|
const cleanupSummary = summaries.find((summary) => summary.executionWorkspaceStatus === "cleanup_failed");
|
|
const featuredSummaries = [localSummary, remoteSummary, cleanupSummary].filter(
|
|
(summary): summary is NonNullable<typeof summary> => Boolean(summary),
|
|
);
|
|
|
|
return (
|
|
<div className="space-y-5">
|
|
<ProjectWorkspacesContent
|
|
companyId={COMPANY_ID}
|
|
projectId={boardProject.id}
|
|
projectRef={boardProject.urlKey}
|
|
summaries={summaries}
|
|
/>
|
|
<div className="grid gap-4 xl:grid-cols-3">
|
|
{featuredSummaries.map((summary) => (
|
|
<ProjectWorkspaceSummaryCard
|
|
key={summary.key}
|
|
projectRef={boardProject.urlKey}
|
|
summary={summary}
|
|
runtimeActionKey={summary.runningServiceCount > 0 ? `${summary.key}:stop` : null}
|
|
runtimeActionPending={summary.runningServiceCount > 0}
|
|
onRuntimeAction={() => undefined}
|
|
onCloseWorkspace={() => undefined}
|
|
/>
|
|
))}
|
|
</div>
|
|
<div className="rounded-lg border border-dashed border-border p-5 text-sm text-muted-foreground">
|
|
<ProjectWorkspacesContent
|
|
companyId={COMPANY_ID}
|
|
projectId={archivedProject.id}
|
|
projectRef={archivedProject.urlKey}
|
|
summaries={[]}
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function GoalProgressRow({ goal }: { goal: Goal }) {
|
|
const progress = goalProgress.get(goal.id) ?? 0;
|
|
const childCount = storybookGoals.filter((candidate) => candidate.parentId === goal.id).length;
|
|
|
|
return (
|
|
<div className="rounded-lg border border-border bg-background p-3">
|
|
<div className="flex items-start justify-between gap-3">
|
|
<div className="min-w-0">
|
|
<div className="truncate text-sm font-medium">{goal.title}</div>
|
|
<div className="mt-1 text-xs text-muted-foreground">
|
|
{goal.level} · {childCount} child goal{childCount === 1 ? "" : "s"}
|
|
</div>
|
|
</div>
|
|
<span className="font-mono text-xs text-muted-foreground">{progress}%</span>
|
|
</div>
|
|
<div className="mt-3 h-2 overflow-hidden rounded-full bg-muted" aria-label={`${progress}% complete`}>
|
|
<div className="h-full rounded-full bg-primary" style={{ width: `${progress}%` }} />
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function GoalPropertiesMatrix() {
|
|
const selectedGoal = storybookGoals.find((goal) => goal.id === "goal-board-ux") ?? storybookGoals[0]!;
|
|
const childGoals = storybookGoals.filter((goal) => goal.parentId === selectedGoal.id);
|
|
const linkedProjects = storybookProjects.filter((project) => project.goalIds.includes(selectedGoal.id));
|
|
|
|
return (
|
|
<div className="grid gap-5 lg:grid-cols-[minmax(0,1fr)_340px]">
|
|
<div className="space-y-4">
|
|
<div className="rounded-lg border border-border bg-background p-5">
|
|
<div className="flex flex-wrap items-start justify-between gap-4">
|
|
<div>
|
|
<div className="paperclip-story__label">Goal detail composition</div>
|
|
<h3 className="mt-2 text-xl font-semibold">{selectedGoal.title}</h3>
|
|
<p className="mt-2 max-w-2xl text-sm leading-6 text-muted-foreground">{selectedGoal.description}</p>
|
|
</div>
|
|
<Badge variant="outline">{selectedGoal.status}</Badge>
|
|
</div>
|
|
<div className="mt-5 grid gap-3 md:grid-cols-3">
|
|
<GoalProgressRow goal={selectedGoal} />
|
|
<div className="rounded-lg border border-border bg-background p-3">
|
|
<div className="text-2xl font-semibold">{childGoals.length}</div>
|
|
<div className="text-xs text-muted-foreground">Child goals</div>
|
|
</div>
|
|
<div className="rounded-lg border border-border bg-background p-3">
|
|
<div className="text-2xl font-semibold">{linkedProjects.length}</div>
|
|
<div className="text-xs text-muted-foreground">Linked projects</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div className="grid gap-3 md:grid-cols-2">
|
|
{childGoals.map((goal) => (
|
|
<GoalProgressRow key={goal.id} goal={goal} />
|
|
))}
|
|
</div>
|
|
</div>
|
|
<div className="rounded-lg border border-border bg-background p-4">
|
|
<GoalProperties goal={selectedGoal} onUpdate={() => undefined} />
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function GoalTreeMatrix() {
|
|
const [selectedGoal, setSelectedGoal] = useState<Goal | null>(storybookGoals[1] ?? null);
|
|
|
|
return (
|
|
<div className="grid gap-5 lg:grid-cols-[minmax(0,1fr)_320px]">
|
|
<div className="overflow-hidden rounded-lg border border-border bg-background">
|
|
<GoalTree
|
|
goals={storybookGoals}
|
|
onSelect={setSelectedGoal}
|
|
/>
|
|
</div>
|
|
<div className="rounded-lg border border-border bg-background p-4">
|
|
<div className="paperclip-story__label">Selected goal</div>
|
|
{selectedGoal ? (
|
|
<div className="mt-3 space-y-3">
|
|
<div>
|
|
<div className="text-sm font-medium">{selectedGoal.title}</div>
|
|
<div className="mt-1 text-xs text-muted-foreground">{selectedGoal.description}</div>
|
|
</div>
|
|
<GoalProgressRow goal={selectedGoal} />
|
|
</div>
|
|
) : (
|
|
<p className="mt-3 text-sm text-muted-foreground">Select a goal row to inspect its progress state.</p>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function RuntimeControlsMatrix() {
|
|
const primaryWorkspace = storybookProjectWorkspaces[0]!;
|
|
const remoteWorkspace = storybookProjectWorkspaces.find((workspace) => workspace.id === "workspace-docs-remote")!;
|
|
const runningSections = buildWorkspaceRuntimeControlSections({
|
|
runtimeConfig: primaryWorkspace.runtimeConfig?.workspaceRuntime,
|
|
runtimeServices: primaryWorkspace.runtimeServices,
|
|
canStartServices: true,
|
|
canRunJobs: true,
|
|
});
|
|
const stoppedSections = buildWorkspaceRuntimeControlSections({
|
|
runtimeConfig: remoteWorkspace.runtimeConfig?.workspaceRuntime,
|
|
runtimeServices: remoteWorkspace.runtimeServices,
|
|
canStartServices: true,
|
|
canRunJobs: true,
|
|
});
|
|
const disabledSections = buildWorkspaceRuntimeControlSections({
|
|
runtimeConfig: {
|
|
commands: [
|
|
{ id: "web", name: "Web app", kind: "service", command: "pnpm dev" },
|
|
{ id: "migrate", name: "Migrate database", kind: "job", command: "pnpm db:migrate" },
|
|
],
|
|
},
|
|
runtimeServices: [],
|
|
canStartServices: false,
|
|
canRunJobs: false,
|
|
});
|
|
const pendingRequest: WorkspaceRuntimeControlRequest = {
|
|
action: "restart",
|
|
workspaceCommandId: "storybook",
|
|
runtimeServiceId: "service-storybook",
|
|
serviceIndex: 0,
|
|
};
|
|
|
|
return (
|
|
<div className="grid gap-5 xl:grid-cols-3">
|
|
<Card className="shadow-none">
|
|
<CardHeader>
|
|
<CardTitle className="flex items-center gap-2">
|
|
<Square className="h-4 w-4" />
|
|
Running services
|
|
</CardTitle>
|
|
<CardDescription>Stop and restart actions with a pending request spinner.</CardDescription>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<WorkspaceRuntimeControls
|
|
sections={runningSections}
|
|
isPending
|
|
pendingRequest={pendingRequest}
|
|
onAction={() => undefined}
|
|
/>
|
|
</CardContent>
|
|
</Card>
|
|
<Card className="shadow-none">
|
|
<CardHeader>
|
|
<CardTitle className="flex items-center gap-2">
|
|
<Play className="h-4 w-4" />
|
|
Stopped remote preview
|
|
</CardTitle>
|
|
<CardDescription>Startable remote workspace service with URL history.</CardDescription>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<WorkspaceRuntimeControls sections={stoppedSections} onAction={() => undefined} />
|
|
</CardContent>
|
|
</Card>
|
|
<Card className="shadow-none">
|
|
<CardHeader>
|
|
<CardTitle className="flex items-center gap-2">
|
|
<RotateCcw className="h-4 w-4" />
|
|
Missing prerequisites
|
|
</CardTitle>
|
|
<CardDescription>Disabled runtime controls when no workspace path is available.</CardDescription>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<WorkspaceRuntimeControls
|
|
sections={disabledSections}
|
|
disabledHint="Add a workspace path before starting runtime services."
|
|
square
|
|
onAction={() => undefined}
|
|
/>
|
|
</CardContent>
|
|
</Card>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function setWorktreeMeta(name: string, content: string) {
|
|
if (typeof document === "undefined") return;
|
|
let element = document.querySelector(`meta[name="${name}"]`);
|
|
if (!element) {
|
|
element = document.createElement("meta");
|
|
element.setAttribute("name", name);
|
|
document.head.appendChild(element);
|
|
}
|
|
element.setAttribute("content", content);
|
|
}
|
|
|
|
function WorktreeBannerMatrix() {
|
|
setWorktreeMeta("paperclip-worktree-enabled", "true");
|
|
setWorktreeMeta("paperclip-worktree-name", "PAP-1675-projects-goals-workspaces");
|
|
setWorktreeMeta("paperclip-worktree-color", "#0f766e");
|
|
setWorktreeMeta("paperclip-worktree-text-color", "#ecfeff");
|
|
|
|
return (
|
|
<div className="space-y-4">
|
|
<div className="overflow-hidden rounded-lg border border-border bg-background">
|
|
<WorktreeBanner />
|
|
</div>
|
|
<div className="grid gap-3 md:grid-cols-3">
|
|
{[
|
|
{ label: "Branch", value: "PAP-1675-projects-goals-workspaces", icon: GitBranch },
|
|
{ label: "Workspace", value: "Project Storybook worktree", icon: FolderGit2 },
|
|
{ label: "Context", value: "visible before layout chrome", icon: Boxes },
|
|
].map((item) => {
|
|
const Icon = item.icon;
|
|
return (
|
|
<div key={item.label} className="rounded-lg border border-border bg-background p-4">
|
|
<Icon className="h-4 w-4 text-muted-foreground" />
|
|
<div className="mt-3 text-xs uppercase tracking-[0.14em] text-muted-foreground">{item.label}</div>
|
|
<div className="mt-1 break-all font-mono text-xs">{item.value}</div>
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function ProjectsGoalsWorkspacesStories() {
|
|
return (
|
|
<StorybookData>
|
|
<div className="paperclip-story">
|
|
<main className="paperclip-story__inner space-y-6">
|
|
<section className="paperclip-story__frame p-6">
|
|
<div className="flex flex-wrap items-start justify-between gap-5">
|
|
<div>
|
|
<div className="paperclip-story__label">Projects, goals, and workspaces</div>
|
|
<h1 className="mt-2 text-3xl font-semibold tracking-tight">Hierarchical planning and runtime surfaces</h1>
|
|
<p className="mt-3 max-w-3xl text-sm leading-6 text-muted-foreground">
|
|
Fixture-backed project and goal stories cover editable project properties, local and remote workspace
|
|
cards, cleanup failures, goal hierarchy states, and runtime command controls.
|
|
</p>
|
|
</div>
|
|
<div className="flex flex-wrap gap-2">
|
|
<Badge variant="outline">active</Badge>
|
|
<Badge variant="outline">archived</Badge>
|
|
<Badge variant="outline">local workspace</Badge>
|
|
<Badge variant="outline">remote workspace</Badge>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
|
|
<Section eyebrow="ProjectProperties" title="Full project detail panels with codebase, goals, env, and archive states">
|
|
<ProjectPropertiesMatrix />
|
|
</Section>
|
|
|
|
<Section eyebrow="ProjectWorkspacesContent" title="Workspace list with local, remote, cleanup-failed, and empty states">
|
|
<WorkspacesMatrix />
|
|
</Section>
|
|
|
|
<Section eyebrow="GoalProperties" title="Goal detail panel with progress and child goal context">
|
|
<GoalPropertiesMatrix />
|
|
</Section>
|
|
|
|
<Section eyebrow="GoalTree" title="Hierarchical goal tree with expand/collapse and progress sidecar">
|
|
<GoalTreeMatrix />
|
|
</Section>
|
|
|
|
<Section eyebrow="WorkspaceRuntimeControls" title="Runtime start, stop, restart, and disabled command states">
|
|
<RuntimeControlsMatrix />
|
|
</Section>
|
|
|
|
<Section eyebrow="WorktreeBanner" title="Worktree context banner with branch identity">
|
|
<WorktreeBannerMatrix />
|
|
</Section>
|
|
</main>
|
|
</div>
|
|
</StorybookData>
|
|
);
|
|
}
|
|
|
|
const meta = {
|
|
title: "Product/Projects Goals Workspaces",
|
|
component: ProjectsGoalsWorkspacesStories,
|
|
parameters: {
|
|
docs: {
|
|
description: {
|
|
component:
|
|
"Projects, goals, and workspaces stories cover project properties, workspace cards/lists, goal hierarchy panels, runtime controls, and worktree branding states.",
|
|
},
|
|
},
|
|
},
|
|
} satisfies Meta<typeof ProjectsGoalsWorkspacesStories>;
|
|
|
|
export default meta;
|
|
|
|
type Story = StoryObj<typeof meta>;
|
|
|
|
export const SurfaceMatrix: Story = {};
|