orbital-command 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (325) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +396 -0
  3. package/bin/orbital.js +362 -0
  4. package/dist/assets/WorkflowVisualizer-BZ21PIIF.js +84 -0
  5. package/dist/assets/WorkflowVisualizer-BZV40eAE.css +1 -0
  6. package/dist/assets/charts-D__PA1zp.js +72 -0
  7. package/dist/assets/index-D1G6i0nS.css +1 -0
  8. package/dist/assets/index-DpItvKpf.js +419 -0
  9. package/dist/assets/ui-BvF022GT.js +53 -0
  10. package/dist/assets/vendor-Dzv9lrRc.js +59 -0
  11. package/dist/index.html +19 -0
  12. package/dist/scanner-sweep.png +0 -0
  13. package/dist/server/server/adapters/index.js +34 -0
  14. package/dist/server/server/adapters/iterm2-adapter.js +29 -0
  15. package/dist/server/server/adapters/subprocess-adapter.js +21 -0
  16. package/dist/server/server/adapters/terminal-adapter.js +1 -0
  17. package/dist/server/server/config.js +156 -0
  18. package/dist/server/server/database.js +90 -0
  19. package/dist/server/server/index.js +372 -0
  20. package/dist/server/server/init.js +811 -0
  21. package/dist/server/server/parsers/event-parser.js +64 -0
  22. package/dist/server/server/parsers/scope-parser.js +188 -0
  23. package/dist/server/server/routes/config-routes.js +163 -0
  24. package/dist/server/server/routes/data-routes.js +461 -0
  25. package/dist/server/server/routes/dispatch-routes.js +215 -0
  26. package/dist/server/server/routes/git-routes.js +92 -0
  27. package/dist/server/server/routes/scope-routes.js +215 -0
  28. package/dist/server/server/routes/sprint-routes.js +116 -0
  29. package/dist/server/server/routes/version-routes.js +130 -0
  30. package/dist/server/server/routes/workflow-routes.js +185 -0
  31. package/dist/server/server/schema.js +90 -0
  32. package/dist/server/server/services/batch-orchestrator.js +253 -0
  33. package/dist/server/server/services/claude-session-service.js +352 -0
  34. package/dist/server/server/services/config-service.js +132 -0
  35. package/dist/server/server/services/deploy-service.js +51 -0
  36. package/dist/server/server/services/event-service.js +63 -0
  37. package/dist/server/server/services/gate-service.js +83 -0
  38. package/dist/server/server/services/git-service.js +309 -0
  39. package/dist/server/server/services/github-service.js +145 -0
  40. package/dist/server/server/services/readiness-service.js +184 -0
  41. package/dist/server/server/services/scope-cache.js +72 -0
  42. package/dist/server/server/services/scope-service.js +424 -0
  43. package/dist/server/server/services/sprint-orchestrator.js +312 -0
  44. package/dist/server/server/services/sprint-service.js +293 -0
  45. package/dist/server/server/services/workflow-service.js +397 -0
  46. package/dist/server/server/utils/cc-hooks-parser.js +49 -0
  47. package/dist/server/server/utils/dispatch-utils.js +305 -0
  48. package/dist/server/server/utils/logger.js +86 -0
  49. package/dist/server/server/utils/terminal-launcher.js +388 -0
  50. package/dist/server/server/utils/worktree-manager.js +98 -0
  51. package/dist/server/server/watchers/event-watcher.js +81 -0
  52. package/dist/server/server/watchers/scope-watcher.js +33 -0
  53. package/dist/server/shared/api-types.js +5 -0
  54. package/dist/server/shared/default-workflow.json +616 -0
  55. package/dist/server/shared/workflow-config.js +44 -0
  56. package/dist/server/shared/workflow-engine.js +353 -0
  57. package/index.html +15 -0
  58. package/package.json +110 -0
  59. package/postcss.config.js +6 -0
  60. package/schemas/orbital.config.schema.json +83 -0
  61. package/scripts/postinstall.js +24 -0
  62. package/scripts/start.sh +20 -0
  63. package/server/adapters/index.ts +41 -0
  64. package/server/adapters/iterm2-adapter.ts +37 -0
  65. package/server/adapters/subprocess-adapter.ts +25 -0
  66. package/server/adapters/terminal-adapter.ts +24 -0
  67. package/server/config.ts +234 -0
  68. package/server/database.ts +107 -0
  69. package/server/index.ts +452 -0
  70. package/server/init.ts +891 -0
  71. package/server/parsers/event-parser.ts +74 -0
  72. package/server/parsers/scope-parser.ts +240 -0
  73. package/server/routes/config-routes.ts +182 -0
  74. package/server/routes/data-routes.ts +548 -0
  75. package/server/routes/dispatch-routes.ts +275 -0
  76. package/server/routes/git-routes.ts +112 -0
  77. package/server/routes/scope-routes.ts +262 -0
  78. package/server/routes/sprint-routes.ts +142 -0
  79. package/server/routes/version-routes.ts +156 -0
  80. package/server/routes/workflow-routes.ts +198 -0
  81. package/server/schema.ts +90 -0
  82. package/server/services/batch-orchestrator.ts +286 -0
  83. package/server/services/claude-session-service.ts +441 -0
  84. package/server/services/config-service.ts +151 -0
  85. package/server/services/deploy-service.ts +98 -0
  86. package/server/services/event-service.ts +98 -0
  87. package/server/services/gate-service.ts +126 -0
  88. package/server/services/git-service.ts +391 -0
  89. package/server/services/github-service.ts +183 -0
  90. package/server/services/readiness-service.ts +250 -0
  91. package/server/services/scope-cache.ts +81 -0
  92. package/server/services/scope-service.ts +476 -0
  93. package/server/services/sprint-orchestrator.ts +361 -0
  94. package/server/services/sprint-service.ts +415 -0
  95. package/server/services/workflow-service.ts +461 -0
  96. package/server/utils/cc-hooks-parser.ts +70 -0
  97. package/server/utils/dispatch-utils.ts +395 -0
  98. package/server/utils/logger.ts +109 -0
  99. package/server/utils/terminal-launcher.ts +462 -0
  100. package/server/utils/worktree-manager.ts +104 -0
  101. package/server/watchers/event-watcher.ts +100 -0
  102. package/server/watchers/scope-watcher.ts +38 -0
  103. package/shared/api-types.ts +20 -0
  104. package/shared/default-workflow.json +616 -0
  105. package/shared/workflow-config.ts +170 -0
  106. package/shared/workflow-engine.ts +427 -0
  107. package/src/App.tsx +33 -0
  108. package/src/components/AgentBadge.tsx +40 -0
  109. package/src/components/BatchPreflightModal.tsx +115 -0
  110. package/src/components/CardDisplayToggle.tsx +74 -0
  111. package/src/components/ColumnHeaderActions.tsx +55 -0
  112. package/src/components/ColumnMenu.tsx +99 -0
  113. package/src/components/DeployHistory.tsx +141 -0
  114. package/src/components/DispatchModal.tsx +164 -0
  115. package/src/components/DispatchPopover.tsx +139 -0
  116. package/src/components/DragOverlay.tsx +25 -0
  117. package/src/components/DriftSidebar.tsx +140 -0
  118. package/src/components/EnvironmentStrip.tsx +88 -0
  119. package/src/components/ErrorBoundary.tsx +62 -0
  120. package/src/components/FilterChip.tsx +105 -0
  121. package/src/components/GateIndicator.tsx +33 -0
  122. package/src/components/IdeaDetailModal.tsx +190 -0
  123. package/src/components/IdeaFormDialog.tsx +113 -0
  124. package/src/components/KanbanColumn.tsx +201 -0
  125. package/src/components/MarkdownRenderer.tsx +114 -0
  126. package/src/components/NeonGrid.tsx +128 -0
  127. package/src/components/PromotionQueue.tsx +89 -0
  128. package/src/components/ScopeCard.tsx +234 -0
  129. package/src/components/ScopeDetailModal.tsx +255 -0
  130. package/src/components/ScopeFilterBar.tsx +152 -0
  131. package/src/components/SearchInput.tsx +102 -0
  132. package/src/components/SessionPanel.tsx +335 -0
  133. package/src/components/SprintContainer.tsx +303 -0
  134. package/src/components/SprintDependencyDialog.tsx +78 -0
  135. package/src/components/SprintPreflightModal.tsx +138 -0
  136. package/src/components/StatusBar.tsx +168 -0
  137. package/src/components/SwimCell.tsx +67 -0
  138. package/src/components/SwimLaneRow.tsx +94 -0
  139. package/src/components/SwimlaneBoardView.tsx +108 -0
  140. package/src/components/VersionBadge.tsx +139 -0
  141. package/src/components/ViewModeSelector.tsx +114 -0
  142. package/src/components/config/AgentChip.tsx +53 -0
  143. package/src/components/config/AgentCreateDialog.tsx +321 -0
  144. package/src/components/config/AgentEditor.tsx +175 -0
  145. package/src/components/config/DirectoryTree.tsx +582 -0
  146. package/src/components/config/FileEditor.tsx +550 -0
  147. package/src/components/config/HookChip.tsx +50 -0
  148. package/src/components/config/StageCard.tsx +198 -0
  149. package/src/components/config/TransitionZone.tsx +173 -0
  150. package/src/components/config/UnifiedWorkflowPipeline.tsx +216 -0
  151. package/src/components/config/WorkflowPipeline.tsx +161 -0
  152. package/src/components/source-control/BranchList.tsx +93 -0
  153. package/src/components/source-control/BranchPanel.tsx +105 -0
  154. package/src/components/source-control/CommitLog.tsx +100 -0
  155. package/src/components/source-control/CommitRow.tsx +47 -0
  156. package/src/components/source-control/GitHubPanel.tsx +110 -0
  157. package/src/components/source-control/GitHubSetupGuide.tsx +52 -0
  158. package/src/components/source-control/GitOverviewBar.tsx +101 -0
  159. package/src/components/source-control/PullRequestList.tsx +69 -0
  160. package/src/components/source-control/WorktreeList.tsx +80 -0
  161. package/src/components/ui/badge.tsx +41 -0
  162. package/src/components/ui/button.tsx +55 -0
  163. package/src/components/ui/card.tsx +78 -0
  164. package/src/components/ui/dialog.tsx +94 -0
  165. package/src/components/ui/popover.tsx +33 -0
  166. package/src/components/ui/scroll-area.tsx +54 -0
  167. package/src/components/ui/separator.tsx +28 -0
  168. package/src/components/ui/tabs.tsx +52 -0
  169. package/src/components/ui/toggle-switch.tsx +35 -0
  170. package/src/components/ui/tooltip.tsx +27 -0
  171. package/src/components/workflow/AddEdgeDialog.tsx +217 -0
  172. package/src/components/workflow/AddListDialog.tsx +201 -0
  173. package/src/components/workflow/ChecklistEditor.tsx +239 -0
  174. package/src/components/workflow/CommandPrefixManager.tsx +118 -0
  175. package/src/components/workflow/ConfigSettingsPanel.tsx +189 -0
  176. package/src/components/workflow/DirectionSelector.tsx +133 -0
  177. package/src/components/workflow/DispatchConfigPanel.tsx +180 -0
  178. package/src/components/workflow/EdgeDetailPanel.tsx +236 -0
  179. package/src/components/workflow/EdgePropertyEditor.tsx +251 -0
  180. package/src/components/workflow/EditToolbar.tsx +138 -0
  181. package/src/components/workflow/HookDetailPanel.tsx +250 -0
  182. package/src/components/workflow/HookExecutionLog.tsx +24 -0
  183. package/src/components/workflow/HookSourceModal.tsx +129 -0
  184. package/src/components/workflow/HooksDashboard.tsx +363 -0
  185. package/src/components/workflow/ListPropertyEditor.tsx +251 -0
  186. package/src/components/workflow/MigrationPreviewDialog.tsx +237 -0
  187. package/src/components/workflow/MovementRulesPanel.tsx +188 -0
  188. package/src/components/workflow/NodeDetailPanel.tsx +245 -0
  189. package/src/components/workflow/PresetSelector.tsx +414 -0
  190. package/src/components/workflow/SkillCommandBuilder.tsx +174 -0
  191. package/src/components/workflow/WorkflowEdgeComponent.tsx +145 -0
  192. package/src/components/workflow/WorkflowNode.tsx +147 -0
  193. package/src/components/workflow/graphLayout.ts +186 -0
  194. package/src/components/workflow/mergeHooks.ts +85 -0
  195. package/src/components/workflow/useEditHistory.ts +88 -0
  196. package/src/components/workflow/useWorkflowEditor.ts +262 -0
  197. package/src/components/workflow/validateConfig.ts +70 -0
  198. package/src/hooks/useActiveDispatches.ts +198 -0
  199. package/src/hooks/useBoardSettings.ts +170 -0
  200. package/src/hooks/useCardDisplay.ts +57 -0
  201. package/src/hooks/useCcHooks.ts +24 -0
  202. package/src/hooks/useConfigTree.ts +51 -0
  203. package/src/hooks/useEnforcementRules.ts +46 -0
  204. package/src/hooks/useEvents.ts +59 -0
  205. package/src/hooks/useFileEditor.ts +165 -0
  206. package/src/hooks/useGates.ts +57 -0
  207. package/src/hooks/useIdeaActions.ts +53 -0
  208. package/src/hooks/useKanbanDnd.ts +410 -0
  209. package/src/hooks/useOrbitalConfig.ts +54 -0
  210. package/src/hooks/usePipeline.ts +47 -0
  211. package/src/hooks/usePipelineData.ts +338 -0
  212. package/src/hooks/useReconnect.ts +25 -0
  213. package/src/hooks/useScopeFilters.ts +125 -0
  214. package/src/hooks/useScopeSessions.ts +44 -0
  215. package/src/hooks/useScopes.ts +67 -0
  216. package/src/hooks/useSearch.ts +67 -0
  217. package/src/hooks/useSettings.tsx +187 -0
  218. package/src/hooks/useSocket.ts +25 -0
  219. package/src/hooks/useSourceControl.ts +105 -0
  220. package/src/hooks/useSprintPreflight.ts +55 -0
  221. package/src/hooks/useSprints.ts +154 -0
  222. package/src/hooks/useStatusBarHighlight.ts +18 -0
  223. package/src/hooks/useSwimlaneBoardSettings.ts +104 -0
  224. package/src/hooks/useTheme.ts +9 -0
  225. package/src/hooks/useTransitionReadiness.ts +53 -0
  226. package/src/hooks/useVersion.ts +155 -0
  227. package/src/hooks/useViolations.ts +65 -0
  228. package/src/hooks/useWorkflow.tsx +125 -0
  229. package/src/hooks/useZoomModifier.ts +19 -0
  230. package/src/index.css +797 -0
  231. package/src/layouts/DashboardLayout.tsx +113 -0
  232. package/src/lib/collisionDetection.ts +20 -0
  233. package/src/lib/scope-fields.ts +61 -0
  234. package/src/lib/swimlane.ts +146 -0
  235. package/src/lib/utils.ts +15 -0
  236. package/src/main.tsx +19 -0
  237. package/src/socket.ts +11 -0
  238. package/src/types/index.ts +497 -0
  239. package/src/views/AgentFeed.tsx +339 -0
  240. package/src/views/DeployPipeline.tsx +59 -0
  241. package/src/views/EnforcementView.tsx +378 -0
  242. package/src/views/PrimitivesConfig.tsx +500 -0
  243. package/src/views/QualityGates.tsx +1012 -0
  244. package/src/views/ScopeBoard.tsx +454 -0
  245. package/src/views/SessionTimeline.tsx +516 -0
  246. package/src/views/Settings.tsx +183 -0
  247. package/src/views/SourceControl.tsx +95 -0
  248. package/src/views/WorkflowVisualizer.tsx +382 -0
  249. package/tailwind.config.js +161 -0
  250. package/templates/agents/AUTO-INVOKE.md +180 -0
  251. package/templates/agents/CONFLICT-RESOLUTION.md +128 -0
  252. package/templates/agents/QUICK-REFERENCE.md +122 -0
  253. package/templates/agents/README.md +188 -0
  254. package/templates/agents/SKILL-TRIGGERS.md +100 -0
  255. package/templates/agents/blue-team/frontend-designer.md +424 -0
  256. package/templates/agents/green-team/architect.md +526 -0
  257. package/templates/agents/green-team/rules-enforcer.md +131 -0
  258. package/templates/agents/red-team/attacker-learned.md +24 -0
  259. package/templates/agents/red-team/attacker.md +486 -0
  260. package/templates/agents/red-team/chaos.md +548 -0
  261. package/templates/agents/reference/component-registry.md +82 -0
  262. package/templates/agents/workflows/full-mode.md +218 -0
  263. package/templates/agents/workflows/quick-mode.md +118 -0
  264. package/templates/agents/workflows/security-mode.md +283 -0
  265. package/templates/anti-patterns/dangerous-shortcuts.md +427 -0
  266. package/templates/config/agent-triggers.json +92 -0
  267. package/templates/hooks/agent-team-gate.sh +31 -0
  268. package/templates/hooks/agent-trigger.sh +97 -0
  269. package/templates/hooks/block-push.sh +66 -0
  270. package/templates/hooks/block-workarounds.sh +61 -0
  271. package/templates/hooks/blocker-check.sh +28 -0
  272. package/templates/hooks/completion-checklist.sh +28 -0
  273. package/templates/hooks/decision-capture.sh +15 -0
  274. package/templates/hooks/dependency-check.sh +27 -0
  275. package/templates/hooks/end-session.sh +31 -0
  276. package/templates/hooks/exploration-logger.sh +37 -0
  277. package/templates/hooks/files-changed-summary.sh +37 -0
  278. package/templates/hooks/get-session-id.sh +49 -0
  279. package/templates/hooks/git-commit-guard.sh +34 -0
  280. package/templates/hooks/init-session.sh +93 -0
  281. package/templates/hooks/orbital-emit.sh +79 -0
  282. package/templates/hooks/orbital-report-deploy.sh +78 -0
  283. package/templates/hooks/orbital-report-gates.sh +40 -0
  284. package/templates/hooks/orbital-report-violation.sh +36 -0
  285. package/templates/hooks/orbital-scope-update.sh +53 -0
  286. package/templates/hooks/phase-verify-reminder.sh +26 -0
  287. package/templates/hooks/review-gate-check.sh +82 -0
  288. package/templates/hooks/scope-commit-logger.sh +37 -0
  289. package/templates/hooks/scope-create-cleanup.sh +36 -0
  290. package/templates/hooks/scope-create-gate.sh +80 -0
  291. package/templates/hooks/scope-create-tracker.sh +17 -0
  292. package/templates/hooks/scope-file-sync.sh +53 -0
  293. package/templates/hooks/scope-gate.sh +35 -0
  294. package/templates/hooks/scope-helpers.sh +188 -0
  295. package/templates/hooks/scope-lifecycle-gate.sh +139 -0
  296. package/templates/hooks/scope-prepare.sh +244 -0
  297. package/templates/hooks/scope-transition.sh +172 -0
  298. package/templates/hooks/session-enforcer.sh +143 -0
  299. package/templates/hooks/time-tracker.sh +33 -0
  300. package/templates/lessons-learned.md +15 -0
  301. package/templates/orbital.config.json +35 -0
  302. package/templates/presets/development.json +42 -0
  303. package/templates/presets/gitflow.json +712 -0
  304. package/templates/presets/minimal.json +23 -0
  305. package/templates/quick/rules.md +218 -0
  306. package/templates/scopes/_template.md +255 -0
  307. package/templates/settings-hooks.json +98 -0
  308. package/templates/skills/git-commit/SKILL.md +85 -0
  309. package/templates/skills/git-dev/SKILL.md +99 -0
  310. package/templates/skills/git-hotfix/SKILL.md +223 -0
  311. package/templates/skills/git-main/SKILL.md +84 -0
  312. package/templates/skills/git-production/SKILL.md +165 -0
  313. package/templates/skills/git-staging/SKILL.md +112 -0
  314. package/templates/skills/scope-create/SKILL.md +81 -0
  315. package/templates/skills/scope-fix-review/SKILL.md +168 -0
  316. package/templates/skills/scope-implement/SKILL.md +110 -0
  317. package/templates/skills/scope-post-review/SKILL.md +144 -0
  318. package/templates/skills/scope-pre-review/SKILL.md +211 -0
  319. package/templates/skills/scope-verify/SKILL.md +201 -0
  320. package/templates/skills/session-init/SKILL.md +62 -0
  321. package/templates/skills/session-resume/SKILL.md +201 -0
  322. package/templates/skills/test-checks/SKILL.md +171 -0
  323. package/templates/skills/test-code-review/SKILL.md +252 -0
  324. package/tsconfig.json +25 -0
  325. package/vite.config.ts +38 -0
@@ -0,0 +1,101 @@
1
+ import {
2
+ GitBranch,
3
+ Circle,
4
+ ArrowUpDown,
5
+ GitFork,
6
+ Github,
7
+ } from 'lucide-react';
8
+ import { Badge } from '@/components/ui/badge';
9
+ import { Card, CardContent } from '@/components/ui/card';
10
+ import { Tooltip, TooltipTrigger, TooltipContent } from '@/components/ui/tooltip';
11
+ import type { GitOverview, GitHubStatus } from '@/types';
12
+
13
+ interface Props {
14
+ overview: GitOverview;
15
+ github: GitHubStatus | null;
16
+ }
17
+
18
+ export function GitOverviewBar({ overview, github }: Props) {
19
+ return (
20
+ <Card className="mb-6">
21
+ <CardContent className="flex flex-wrap items-center gap-4 py-3">
22
+ {/* Current branch */}
23
+ <div className="flex items-center gap-2">
24
+ <GitBranch className="h-4 w-4 text-primary" />
25
+ <code className="font-mono text-sm">{overview.currentBranch}</code>
26
+ {overview.dirty && (
27
+ <Tooltip>
28
+ <TooltipTrigger>
29
+ <Circle className="h-2.5 w-2.5 fill-warning-amber text-warning-amber" />
30
+ </TooltipTrigger>
31
+ <TooltipContent>Uncommitted changes</TooltipContent>
32
+ </Tooltip>
33
+ )}
34
+ </div>
35
+
36
+ {/* Branching mode badge */}
37
+ <Badge variant="outline" className="text-xs">
38
+ {overview.branchingMode}
39
+ </Badge>
40
+
41
+ {/* HEAD SHA */}
42
+ {overview.mainHead && (
43
+ <div className="flex items-center gap-1.5 text-xs text-muted-foreground">
44
+ <span>HEAD</span>
45
+ <code className="font-mono text-xs">{overview.mainHead.sha.slice(0, 7)}</code>
46
+ <span className="max-w-[200px] truncate">{overview.mainHead.message}</span>
47
+ </div>
48
+ )}
49
+
50
+ {/* Ahead/Behind */}
51
+ {overview.aheadBehind && (overview.aheadBehind.ahead > 0 || overview.aheadBehind.behind > 0) && (
52
+ <div className="flex items-center gap-1 text-xs">
53
+ <ArrowUpDown className="h-3 w-3 text-muted-foreground" />
54
+ {overview.aheadBehind.ahead > 0 && (
55
+ <span className="text-bid-green">{overview.aheadBehind.ahead}↑</span>
56
+ )}
57
+ {overview.aheadBehind.behind > 0 && (
58
+ <span className="text-ask-red">{overview.aheadBehind.behind}↓</span>
59
+ )}
60
+ </div>
61
+ )}
62
+
63
+ {/* Worktree / feature branch count */}
64
+ {overview.branchingMode === 'worktree' && overview.worktreeCount > 0 && (
65
+ <div className="flex items-center gap-1 text-xs text-muted-foreground">
66
+ <GitFork className="h-3 w-3" />
67
+ <span>{overview.worktreeCount} worktree{overview.worktreeCount !== 1 ? 's' : ''}</span>
68
+ </div>
69
+ )}
70
+ {overview.branchingMode === 'trunk' && overview.featureBranchCount > 0 && (
71
+ <div className="flex items-center gap-1 text-xs text-muted-foreground">
72
+ <GitBranch className="h-3 w-3" />
73
+ <span>{overview.featureBranchCount} feature branch{overview.featureBranchCount !== 1 ? 'es' : ''}</span>
74
+ </div>
75
+ )}
76
+
77
+ {/* GitHub connection badge — pushed to the right */}
78
+ <div className="ml-auto flex items-center gap-1.5">
79
+ <Github className="h-3.5 w-3.5 text-muted-foreground" />
80
+ {github?.connected ? (
81
+ <Tooltip>
82
+ <TooltipTrigger>
83
+ <Badge variant="secondary" className="text-xs gap-1">
84
+ <Circle className="h-1.5 w-1.5 fill-bid-green text-bid-green" />
85
+ {github.repo?.fullName ?? 'Connected'}
86
+ </Badge>
87
+ </TooltipTrigger>
88
+ <TooltipContent>
89
+ {github.authUser ? `Signed in as ${github.authUser}` : 'Connected to GitHub'}
90
+ </TooltipContent>
91
+ </Tooltip>
92
+ ) : (
93
+ <Badge variant="outline" className="text-xs text-muted-foreground">
94
+ Not connected
95
+ </Badge>
96
+ )}
97
+ </div>
98
+ </CardContent>
99
+ </Card>
100
+ );
101
+ }
@@ -0,0 +1,69 @@
1
+ import { formatDistanceToNow } from 'date-fns';
2
+ import { GitPullRequest, ExternalLink } from 'lucide-react';
3
+ import { Badge } from '@/components/ui/badge';
4
+ import type { PullRequestInfo } from '@/types';
5
+
6
+ interface Props {
7
+ prs: PullRequestInfo[];
8
+ }
9
+
10
+ export function PullRequestList({ prs }: Props) {
11
+ if (prs.length === 0) {
12
+ return (
13
+ <div className="py-4 text-center">
14
+ <GitPullRequest className="mx-auto mb-2 h-8 w-8 text-muted-foreground/50" />
15
+ <p className="text-xs text-muted-foreground">No open pull requests.</p>
16
+ </div>
17
+ );
18
+ }
19
+
20
+ return (
21
+ <div className="space-y-0.5">
22
+ {prs.map(pr => (
23
+ <a
24
+ key={pr.number}
25
+ href={pr.url}
26
+ target="_blank"
27
+ rel="noopener noreferrer"
28
+ className="flex items-center gap-3 rounded px-2.5 py-1.5 transition-colors hover:bg-surface-light group"
29
+ >
30
+ <GitPullRequest className="h-4 w-4 shrink-0 text-bid-green" />
31
+
32
+ <span className="shrink-0 font-mono text-xs text-muted-foreground">
33
+ #{pr.number}
34
+ </span>
35
+
36
+ <span className="min-w-0 flex-1 truncate text-sm">
37
+ {pr.title}
38
+ </span>
39
+
40
+ {/* Branch badges */}
41
+ <Badge variant="outline" className="shrink-0 text-xs font-normal">
42
+ {pr.branch}
43
+ </Badge>
44
+ <span className="text-xs text-muted-foreground">→</span>
45
+ <Badge variant="outline" className="shrink-0 text-xs font-normal">
46
+ {pr.baseBranch}
47
+ </Badge>
48
+
49
+ {/* Scope IDs */}
50
+ {pr.scopeIds.map(id => (
51
+ <Badge key={id} variant="secondary" className="shrink-0 text-xs">
52
+ #{id}
53
+ </Badge>
54
+ ))}
55
+
56
+ <span className="shrink-0 text-xs text-muted-foreground">
57
+ {pr.author}
58
+ </span>
59
+
60
+ <span className="shrink-0 text-xs text-muted-foreground/60">
61
+ {formatDistanceToNow(new Date(pr.createdAt), { addSuffix: true })}
62
+ </span>
63
+
64
+ <ExternalLink className="h-3 w-3 shrink-0 text-muted-foreground/40 opacity-0 group-hover:opacity-100 transition-opacity" />
65
+ </a>
66
+ ))}
67
+ </div>
68
+ );
69
+ }
@@ -0,0 +1,80 @@
1
+ import { GitFork, Circle, ArrowUp, ArrowDown } from 'lucide-react';
2
+ import { Badge } from '@/components/ui/badge';
3
+ import { cn } from '@/lib/utils';
4
+ import type { WorktreeDetail } from '@/types';
5
+
6
+ interface Props {
7
+ worktrees: WorktreeDetail[];
8
+ }
9
+
10
+ export function WorktreeList({ worktrees }: Props) {
11
+ if (worktrees.length === 0) {
12
+ return (
13
+ <div className="py-6 text-center">
14
+ <GitFork className="mx-auto mb-2 h-8 w-8 text-muted-foreground/50" />
15
+ <p className="text-xs text-muted-foreground">No active worktrees.</p>
16
+ </div>
17
+ );
18
+ }
19
+
20
+ return (
21
+ <div className="space-y-1">
22
+ {worktrees.map(wt => (
23
+ <div
24
+ key={wt.path}
25
+ className="rounded border border-border/50 px-3 py-2 transition-colors hover:bg-surface-light"
26
+ >
27
+ <div className="flex items-center gap-2">
28
+ <GitFork className="h-3.5 w-3.5 shrink-0 text-primary" />
29
+ <code className="min-w-0 flex-1 truncate font-mono text-xs">
30
+ {wt.branch}
31
+ </code>
32
+ {wt.dirty && (
33
+ <Circle className="h-2 w-2 shrink-0 fill-warning-amber text-warning-amber" />
34
+ )}
35
+ <code className="shrink-0 font-mono text-xs text-muted-foreground/60">
36
+ {wt.head}
37
+ </code>
38
+ </div>
39
+
40
+ {/* Scope info */}
41
+ {wt.scopeId && (
42
+ <div className="mt-1 flex items-center gap-2">
43
+ <Badge variant="secondary" className="text-xs">
44
+ #{wt.scopeId}
45
+ </Badge>
46
+ {wt.scopeTitle && (
47
+ <span className="min-w-0 truncate text-xs text-muted-foreground">
48
+ {wt.scopeTitle}
49
+ </span>
50
+ )}
51
+ {wt.scopeStatus && (
52
+ <Badge variant="outline" className={cn('shrink-0 text-xs capitalize')}>
53
+ {wt.scopeStatus}
54
+ </Badge>
55
+ )}
56
+ </div>
57
+ )}
58
+
59
+ {/* Ahead/behind */}
60
+ {wt.aheadBehind && (wt.aheadBehind.ahead > 0 || wt.aheadBehind.behind > 0) && (
61
+ <div className="mt-1 flex items-center gap-2 text-xs">
62
+ {wt.aheadBehind.ahead > 0 && (
63
+ <span className="flex items-center gap-0.5 text-bid-green">
64
+ <ArrowUp className="h-2.5 w-2.5" />
65
+ {wt.aheadBehind.ahead} ahead
66
+ </span>
67
+ )}
68
+ {wt.aheadBehind.behind > 0 && (
69
+ <span className="flex items-center gap-0.5 text-ask-red">
70
+ <ArrowDown className="h-2.5 w-2.5" />
71
+ {wt.aheadBehind.behind} behind
72
+ </span>
73
+ )}
74
+ </div>
75
+ )}
76
+ </div>
77
+ ))}
78
+ </div>
79
+ );
80
+ }
@@ -0,0 +1,41 @@
1
+ import * as React from 'react';
2
+ import { cva, type VariantProps } from 'class-variance-authority';
3
+ import { cn } from '@/lib/utils';
4
+
5
+ const badgeVariants = cva(
6
+ 'inline-flex items-center rounded border px-1.5 py-0.5 text-xxs font-normal transition-colors focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2',
7
+ {
8
+ variants: {
9
+ variant: {
10
+ default:
11
+ 'border-transparent bg-primary text-primary-foreground hover:bg-primary/80',
12
+ secondary:
13
+ 'border-transparent bg-secondary text-secondary-foreground hover:bg-secondary/80',
14
+ destructive:
15
+ 'border-transparent bg-destructive text-destructive-foreground hover:bg-destructive/80',
16
+ outline: 'text-foreground',
17
+ success:
18
+ 'border-transparent bg-[#00c85322] text-[#00c853] glow-green',
19
+ warning:
20
+ 'border-transparent bg-[#ffab0022] text-[#ffab00] glow-amber',
21
+ error:
22
+ 'border-transparent bg-[#ff174422] text-[#ff1744] glow-red',
23
+ },
24
+ },
25
+ defaultVariants: {
26
+ variant: 'default',
27
+ },
28
+ }
29
+ );
30
+
31
+ export interface BadgeProps
32
+ extends React.HTMLAttributes<HTMLDivElement>,
33
+ VariantProps<typeof badgeVariants> {}
34
+
35
+ function Badge({ className, variant, ...props }: BadgeProps) {
36
+ return (
37
+ <div className={cn(badgeVariants({ variant }), className)} {...props} />
38
+ );
39
+ }
40
+
41
+ export { Badge, badgeVariants };
@@ -0,0 +1,55 @@
1
+ import * as React from 'react';
2
+ import { Slot } from '@radix-ui/react-slot';
3
+ import { cva, type VariantProps } from 'class-variance-authority';
4
+ import { cn } from '@/lib/utils';
5
+
6
+ const buttonVariants = cva(
7
+ 'inline-flex items-center justify-center whitespace-nowrap rounded text-xs font-normal ring-offset-background transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50',
8
+ {
9
+ variants: {
10
+ variant: {
11
+ default: 'bg-primary text-primary-foreground hover:bg-primary/90 glow-blue',
12
+ destructive:
13
+ 'bg-destructive text-destructive-foreground hover:bg-destructive/90',
14
+ outline:
15
+ 'border border-border bg-background hover:bg-surface-light hover:text-accent-foreground',
16
+ secondary:
17
+ 'bg-secondary text-secondary-foreground hover:bg-secondary/80',
18
+ ghost: 'hover:bg-surface-light hover:text-accent-foreground',
19
+ link: 'text-primary underline-offset-4 hover:underline',
20
+ },
21
+ size: {
22
+ default: 'h-8 px-3 py-1.5',
23
+ sm: 'h-7 rounded px-2.5',
24
+ lg: 'h-9 rounded px-5',
25
+ icon: 'h-8 w-8',
26
+ },
27
+ },
28
+ defaultVariants: {
29
+ variant: 'default',
30
+ size: 'default',
31
+ },
32
+ }
33
+ );
34
+
35
+ export interface ButtonProps
36
+ extends React.ButtonHTMLAttributes<HTMLButtonElement>,
37
+ VariantProps<typeof buttonVariants> {
38
+ asChild?: boolean;
39
+ }
40
+
41
+ const Button = React.forwardRef<HTMLButtonElement, ButtonProps>(
42
+ ({ className, variant, size, asChild = false, ...props }, ref) => {
43
+ const Comp = asChild ? Slot : 'button';
44
+ return (
45
+ <Comp
46
+ className={cn(buttonVariants({ variant, size, className }))}
47
+ ref={ref}
48
+ {...props}
49
+ />
50
+ );
51
+ }
52
+ );
53
+ Button.displayName = 'Button';
54
+
55
+ export { Button, buttonVariants };
@@ -0,0 +1,78 @@
1
+ import * as React from 'react';
2
+ import { cn } from '@/lib/utils';
3
+
4
+ const Card = React.forwardRef<
5
+ HTMLDivElement,
6
+ React.HTMLAttributes<HTMLDivElement>
7
+ >(({ className, ...props }, ref) => (
8
+ <div
9
+ ref={ref}
10
+ className={cn(
11
+ 'card-glass rounded border bg-card text-card-foreground shadow-none',
12
+ className
13
+ )}
14
+ {...props}
15
+ />
16
+ ));
17
+ Card.displayName = 'Card';
18
+
19
+ const CardHeader = React.forwardRef<
20
+ HTMLDivElement,
21
+ React.HTMLAttributes<HTMLDivElement>
22
+ >(({ className, ...props }, ref) => (
23
+ <div
24
+ ref={ref}
25
+ className={cn('flex flex-col space-y-0.5 px-3 py-2', className)}
26
+ {...props}
27
+ />
28
+ ));
29
+ CardHeader.displayName = 'CardHeader';
30
+
31
+ const CardTitle = React.forwardRef<
32
+ HTMLHeadingElement,
33
+ React.HTMLAttributes<HTMLHeadingElement>
34
+ >(({ className, ...props }, ref) => (
35
+ <h3
36
+ ref={ref}
37
+ className={cn(
38
+ 'text-sm font-light uppercase tracking-wider text-muted-foreground',
39
+ className
40
+ )}
41
+ {...props}
42
+ />
43
+ ));
44
+ CardTitle.displayName = 'CardTitle';
45
+
46
+ const CardDescription = React.forwardRef<
47
+ HTMLParagraphElement,
48
+ React.HTMLAttributes<HTMLParagraphElement>
49
+ >(({ className, ...props }, ref) => (
50
+ <p
51
+ ref={ref}
52
+ className={cn('text-xs text-muted-foreground', className)}
53
+ {...props}
54
+ />
55
+ ));
56
+ CardDescription.displayName = 'CardDescription';
57
+
58
+ const CardContent = React.forwardRef<
59
+ HTMLDivElement,
60
+ React.HTMLAttributes<HTMLDivElement>
61
+ >(({ className, ...props }, ref) => (
62
+ <div ref={ref} className={cn('p-3 pt-0', className)} {...props} />
63
+ ));
64
+ CardContent.displayName = 'CardContent';
65
+
66
+ const CardFooter = React.forwardRef<
67
+ HTMLDivElement,
68
+ React.HTMLAttributes<HTMLDivElement>
69
+ >(({ className, ...props }, ref) => (
70
+ <div
71
+ ref={ref}
72
+ className={cn('flex items-center p-3 pt-0', className)}
73
+ {...props}
74
+ />
75
+ ));
76
+ CardFooter.displayName = 'CardFooter';
77
+
78
+ export { Card, CardHeader, CardFooter, CardTitle, CardDescription, CardContent };
@@ -0,0 +1,94 @@
1
+ import * as React from 'react';
2
+ import * as DialogPrimitive from '@radix-ui/react-dialog';
3
+ import { X } from 'lucide-react';
4
+ import { cn } from '@/lib/utils';
5
+
6
+ const Dialog = DialogPrimitive.Root;
7
+ const DialogTrigger = DialogPrimitive.Trigger;
8
+ const DialogPortal = DialogPrimitive.Portal;
9
+ const DialogClose = DialogPrimitive.Close;
10
+
11
+ const DialogOverlay = React.forwardRef<
12
+ React.ComponentRef<typeof DialogPrimitive.Overlay>,
13
+ React.ComponentPropsWithoutRef<typeof DialogPrimitive.Overlay>
14
+ >(({ className, ...props }, ref) => (
15
+ <DialogPrimitive.Overlay
16
+ ref={ref}
17
+ className={cn(
18
+ 'fixed inset-0 z-50 bg-black/70 dialog-overlay-glass data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0',
19
+ className,
20
+ )}
21
+ {...props}
22
+ />
23
+ ));
24
+ DialogOverlay.displayName = DialogPrimitive.Overlay.displayName;
25
+
26
+ const DialogContent = React.forwardRef<
27
+ React.ComponentRef<typeof DialogPrimitive.Content>,
28
+ React.ComponentPropsWithoutRef<typeof DialogPrimitive.Content>
29
+ >(({ className, children, ...props }, ref) => (
30
+ <DialogPortal>
31
+ <DialogOverlay />
32
+ <DialogPrimitive.Content
33
+ ref={ref}
34
+ className={cn(
35
+ 'fixed left-1/2 top-1/2 z-50 -translate-x-1/2 -translate-y-1/2',
36
+ 'w-full border bg-background shadow-lg duration-200',
37
+ 'data-[state=open]:animate-in data-[state=closed]:animate-out',
38
+ 'data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0',
39
+ 'data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95',
40
+ 'data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%]',
41
+ 'data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%]',
42
+ 'card-glass rounded border-border bg-[#12121a]',
43
+ className,
44
+ )}
45
+ {...props}
46
+ >
47
+ {children}
48
+ <DialogPrimitive.Close className="absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-accent data-[state=open]:text-muted-foreground">
49
+ <X className="h-4 w-4" />
50
+ <span className="sr-only">Close</span>
51
+ </DialogPrimitive.Close>
52
+ </DialogPrimitive.Content>
53
+ </DialogPortal>
54
+ ));
55
+ DialogContent.displayName = DialogPrimitive.Content.displayName;
56
+
57
+ function DialogHeader({ className, ...props }: React.HTMLAttributes<HTMLDivElement>) {
58
+ return (
59
+ <div
60
+ className={cn('flex flex-col space-y-1.5 text-center sm:text-left', className)}
61
+ {...props}
62
+ />
63
+ );
64
+ }
65
+
66
+ function DialogTitle({ className, ...props }: React.ComponentPropsWithoutRef<typeof DialogPrimitive.Title>) {
67
+ return (
68
+ <DialogPrimitive.Title
69
+ className={cn('text-sm font-normal leading-none tracking-tight', className)}
70
+ {...props}
71
+ />
72
+ );
73
+ }
74
+
75
+ function DialogDescription({ className, ...props }: React.ComponentPropsWithoutRef<typeof DialogPrimitive.Description>) {
76
+ return (
77
+ <DialogPrimitive.Description
78
+ className={cn('text-sm text-muted-foreground', className)}
79
+ {...props}
80
+ />
81
+ );
82
+ }
83
+
84
+ export {
85
+ Dialog,
86
+ DialogPortal,
87
+ DialogOverlay,
88
+ DialogClose,
89
+ DialogTrigger,
90
+ DialogContent,
91
+ DialogHeader,
92
+ DialogTitle,
93
+ DialogDescription,
94
+ };
@@ -0,0 +1,33 @@
1
+ import * as React from 'react';
2
+ import * as PopoverPrimitive from '@radix-ui/react-popover';
3
+ import { cn } from '@/lib/utils';
4
+
5
+ const Popover = PopoverPrimitive.Root;
6
+ const PopoverTrigger = PopoverPrimitive.Trigger;
7
+ const PopoverAnchor = PopoverPrimitive.Anchor;
8
+
9
+ const PopoverContent = React.forwardRef<
10
+ React.ComponentRef<typeof PopoverPrimitive.Content>,
11
+ React.ComponentPropsWithoutRef<typeof PopoverPrimitive.Content>
12
+ >(({ className, align = 'start', sideOffset = 4, ...props }, ref) => (
13
+ <PopoverPrimitive.Portal>
14
+ <PopoverPrimitive.Content
15
+ ref={ref}
16
+ align={align}
17
+ sideOffset={sideOffset}
18
+ className={cn(
19
+ 'z-50 min-w-[180px] rounded border border-border bg-popover p-2 text-popover-foreground shadow-md outline-none',
20
+ 'data-[state=open]:animate-in data-[state=closed]:animate-out',
21
+ 'data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0',
22
+ 'data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95',
23
+ 'data-[side=bottom]:slide-in-from-top-2',
24
+ 'data-[side=top]:slide-in-from-bottom-2',
25
+ className
26
+ )}
27
+ {...props}
28
+ />
29
+ </PopoverPrimitive.Portal>
30
+ ));
31
+ PopoverContent.displayName = PopoverPrimitive.Content.displayName;
32
+
33
+ export { Popover, PopoverTrigger, PopoverContent, PopoverAnchor };
@@ -0,0 +1,54 @@
1
+ import * as React from 'react';
2
+ import * as ScrollAreaPrimitive from '@radix-ui/react-scroll-area';
3
+ import { cn } from '@/lib/utils';
4
+
5
+ interface ScrollAreaProps extends React.ComponentPropsWithoutRef<typeof ScrollAreaPrimitive.Root> {
6
+ orientation?: 'vertical' | 'horizontal' | 'both';
7
+ }
8
+
9
+ const ScrollArea = React.forwardRef<
10
+ React.ComponentRef<typeof ScrollAreaPrimitive.Root>,
11
+ ScrollAreaProps
12
+ >(({ className, children, orientation = 'vertical', ...props }, ref) => (
13
+ <ScrollAreaPrimitive.Root
14
+ ref={ref}
15
+ className={cn('relative overflow-hidden', className)}
16
+ {...props}
17
+ >
18
+ <ScrollAreaPrimitive.Viewport className="h-full w-full rounded-[inherit]">
19
+ {children}
20
+ </ScrollAreaPrimitive.Viewport>
21
+ {(orientation === 'vertical' || orientation === 'both') && (
22
+ <ScrollBar orientation="vertical" />
23
+ )}
24
+ {(orientation === 'horizontal' || orientation === 'both') && (
25
+ <ScrollBar orientation="horizontal" />
26
+ )}
27
+ <ScrollAreaPrimitive.Corner />
28
+ </ScrollAreaPrimitive.Root>
29
+ ));
30
+ ScrollArea.displayName = ScrollAreaPrimitive.Root.displayName;
31
+
32
+ const ScrollBar = React.forwardRef<
33
+ React.ComponentRef<typeof ScrollAreaPrimitive.ScrollAreaScrollbar>,
34
+ React.ComponentPropsWithoutRef<typeof ScrollAreaPrimitive.ScrollAreaScrollbar>
35
+ >(({ className, orientation = 'vertical', ...props }, ref) => (
36
+ <ScrollAreaPrimitive.ScrollAreaScrollbar
37
+ ref={ref}
38
+ orientation={orientation}
39
+ className={cn(
40
+ 'flex touch-none select-none transition-colors',
41
+ orientation === 'vertical' &&
42
+ 'h-full w-1.5 border-l border-l-transparent p-[1px]',
43
+ orientation === 'horizontal' &&
44
+ 'h-1.5 flex-col border-t border-t-transparent p-[1px]',
45
+ className
46
+ )}
47
+ {...props}
48
+ >
49
+ <ScrollAreaPrimitive.ScrollAreaThumb className="relative flex-1 rounded-full bg-border" />
50
+ </ScrollAreaPrimitive.ScrollAreaScrollbar>
51
+ ));
52
+ ScrollBar.displayName = ScrollAreaPrimitive.ScrollAreaScrollbar.displayName;
53
+
54
+ export { ScrollArea, ScrollBar };
@@ -0,0 +1,28 @@
1
+ import * as React from 'react';
2
+ import * as SeparatorPrimitive from '@radix-ui/react-separator';
3
+ import { cn } from '@/lib/utils';
4
+
5
+ const Separator = React.forwardRef<
6
+ React.ComponentRef<typeof SeparatorPrimitive.Root>,
7
+ React.ComponentPropsWithoutRef<typeof SeparatorPrimitive.Root>
8
+ >(
9
+ (
10
+ { className, orientation = 'horizontal', decorative = true, ...props },
11
+ ref
12
+ ) => (
13
+ <SeparatorPrimitive.Root
14
+ ref={ref}
15
+ decorative={decorative}
16
+ orientation={orientation}
17
+ className={cn(
18
+ 'shrink-0 bg-border',
19
+ orientation === 'horizontal' ? 'h-[1px] w-full' : 'h-full w-[1px]',
20
+ className
21
+ )}
22
+ {...props}
23
+ />
24
+ )
25
+ );
26
+ Separator.displayName = SeparatorPrimitive.Root.displayName;
27
+
28
+ export { Separator };