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,161 @@
1
+ import { useMemo } from 'react';
2
+ import { useDroppable } from '@dnd-kit/core';
3
+ import { ArrowDown, Terminal, Zap, Users } from 'lucide-react';
4
+ import { Card } from '@/components/ui/card';
5
+ import { Badge } from '@/components/ui/badge';
6
+ import { ScrollArea } from '@/components/ui/scroll-area';
7
+ import { cn } from '@/lib/utils';
8
+ import { useWorkflow } from '@/hooks/useWorkflow';
9
+ import type { ConfigPrimitiveType } from '@/types';
10
+ import type { WorkflowList, WorkflowEdge } from '../../../shared/workflow-config';
11
+
12
+ interface WorkflowPipelineProps {
13
+ type: ConfigPrimitiveType;
14
+ }
15
+
16
+ /** A droppable zone on the pipeline where tree items can be dropped */
17
+ function PipelineDropZone({ id, children, className }: { id: string; children: React.ReactNode; className?: string }) {
18
+ const { isOver, setNodeRef } = useDroppable({ id });
19
+
20
+ return (
21
+ <div
22
+ ref={setNodeRef}
23
+ className={cn(
24
+ 'rounded border border-dashed border-transparent transition-colors',
25
+ isOver && 'border-accent-blue/40 bg-accent-blue/5',
26
+ className,
27
+ )}
28
+ >
29
+ {children}
30
+ </div>
31
+ );
32
+ }
33
+
34
+ /** A single workflow list card in the pipeline */
35
+ function ListCard({
36
+ list,
37
+ type,
38
+ outEdges,
39
+ }: {
40
+ list: WorkflowList;
41
+ type: ConfigPrimitiveType;
42
+ outEdges: WorkflowEdge[];
43
+ }) {
44
+ const hookBadges = type === 'hooks' && list.activeHooks?.length
45
+ ? list.activeHooks
46
+ : [];
47
+
48
+ return (
49
+ <PipelineDropZone id={`pipeline-list-${list.id}`}>
50
+ <Card className="overflow-hidden">
51
+ <div className="flex items-center gap-2 px-3 py-2">
52
+ <div
53
+ className="h-2.5 w-2.5 shrink-0 rounded-full"
54
+ style={{ backgroundColor: list.hex }}
55
+ />
56
+ <span className="text-xs font-medium text-foreground">{list.label}</span>
57
+ <span className="text-[10px] text-muted-foreground/50 font-mono">{list.id}</span>
58
+ </div>
59
+
60
+ {/* Hook badges on list (hooks mode) */}
61
+ {hookBadges.length > 0 && (
62
+ <div className="flex flex-wrap gap-1 border-t border-border px-3 py-1.5">
63
+ {hookBadges.map((hook) => (
64
+ <Badge key={hook} variant="secondary" className="text-[10px] gap-1 px-1.5 py-0">
65
+ <Zap className="h-2.5 w-2.5" />
66
+ {hook}
67
+ </Badge>
68
+ ))}
69
+ </div>
70
+ )}
71
+ </Card>
72
+
73
+ {/* Outgoing edges */}
74
+ {outEdges.map((edge) => (
75
+ <EdgeSlot key={`${edge.from}:${edge.to}`} edge={edge} type={type} />
76
+ ))}
77
+ </PipelineDropZone>
78
+ );
79
+ }
80
+
81
+ /** The connector between two lists, showing edge metadata */
82
+ function EdgeSlot({ edge, type }: { edge: WorkflowEdge; type: ConfigPrimitiveType }) {
83
+ const showCommand = type === 'skills' && edge.command;
84
+ const showAgents = type === 'agents' && edge.agents?.length;
85
+ const hasContent = showCommand || showAgents;
86
+
87
+ return (
88
+ <PipelineDropZone id={`pipeline-edge-${edge.from}:${edge.to}`} className="my-1">
89
+ <div className="flex items-center gap-2 px-4 py-1">
90
+ <ArrowDown className="h-3 w-3 shrink-0 text-muted-foreground/40" />
91
+ <span className="text-[10px] text-muted-foreground/60 truncate">
92
+ {edge.label}
93
+ </span>
94
+
95
+ {hasContent && (
96
+ <div className="ml-auto flex items-center gap-1">
97
+ {showCommand && (
98
+ <Badge variant="outline" className="text-[10px] gap-1 px-1.5 py-0 text-accent-blue border-accent-blue/30">
99
+ <Terminal className="h-2.5 w-2.5" />
100
+ {edge.command}
101
+ </Badge>
102
+ )}
103
+ {showAgents && edge.agents?.map((agent) => (
104
+ <Badge key={agent} variant="outline" className="text-[10px] gap-1 px-1.5 py-0 text-purple-400 border-purple-400/30">
105
+ <Users className="h-2.5 w-2.5" />
106
+ {agent}
107
+ </Badge>
108
+ ))}
109
+ </div>
110
+ )}
111
+ </div>
112
+ </PipelineDropZone>
113
+ );
114
+ }
115
+
116
+ export function WorkflowPipeline({ type }: WorkflowPipelineProps) {
117
+ const { engine } = useWorkflow();
118
+
119
+ const lists = useMemo(() => engine.getLists(), [engine]);
120
+ const edges = useMemo(() => engine.getAllEdges(), [engine]);
121
+
122
+ // Group edges by source list for rendering
123
+ const edgesByFrom = useMemo(() => {
124
+ const map = new Map<string, WorkflowEdge[]>();
125
+ for (const edge of edges) {
126
+ if (edge.direction !== 'forward') continue; // Only show forward edges in pipeline
127
+ const existing = map.get(edge.from);
128
+ if (existing) existing.push(edge);
129
+ else map.set(edge.from, [edge]);
130
+ }
131
+ return map;
132
+ }, [edges]);
133
+
134
+ return (
135
+ <div className="flex h-full flex-col">
136
+ {/* Header */}
137
+ <div className="flex items-center justify-between border-b border-border px-3 py-2">
138
+ <span className="text-xxs font-medium uppercase tracking-wider text-muted-foreground">
139
+ Workflow Pipeline
140
+ </span>
141
+ <Badge variant="secondary" className="text-[10px]">
142
+ {lists.length} stages
143
+ </Badge>
144
+ </div>
145
+
146
+ {/* Pipeline */}
147
+ <ScrollArea className="flex-1">
148
+ <div className="space-y-0 p-3">
149
+ {lists.map((list) => (
150
+ <ListCard
151
+ key={list.id}
152
+ list={list}
153
+ type={type}
154
+ outEdges={edgesByFrom.get(list.id) ?? []}
155
+ />
156
+ ))}
157
+ </div>
158
+ </ScrollArea>
159
+ </div>
160
+ );
161
+ }
@@ -0,0 +1,93 @@
1
+ import { formatDistanceToNow } from 'date-fns';
2
+ import { GitBranch, ArrowUp, ArrowDown } from 'lucide-react';
3
+ import { Badge } from '@/components/ui/badge';
4
+ import { cn } from '@/lib/utils';
5
+ import type { BranchInfoData } from '@/types';
6
+
7
+ interface Props {
8
+ branches: BranchInfoData[];
9
+ }
10
+
11
+ export function BranchList({ branches }: Props) {
12
+ // Sort: current first, then by date descending, stale at bottom
13
+ const sorted = [...branches]
14
+ .filter(b => !b.isRemote)
15
+ .sort((a, b) => {
16
+ if (a.isCurrent) return -1;
17
+ if (b.isCurrent) return 1;
18
+ if (a.isStale !== b.isStale) return a.isStale ? 1 : -1;
19
+ return new Date(b.headDate).getTime() - new Date(a.headDate).getTime();
20
+ });
21
+
22
+ if (sorted.length === 0) {
23
+ return (
24
+ <div className="py-6 text-center">
25
+ <GitBranch className="mx-auto mb-2 h-8 w-8 text-muted-foreground/50" />
26
+ <p className="text-xs text-muted-foreground">No branches found.</p>
27
+ </div>
28
+ );
29
+ }
30
+
31
+ return (
32
+ <div className="space-y-0.5">
33
+ {sorted.map(branch => (
34
+ <div
35
+ key={branch.name}
36
+ className={cn(
37
+ 'flex items-center gap-2 rounded px-2.5 py-1.5 transition-colors hover:bg-surface-light',
38
+ branch.isStale && 'opacity-50',
39
+ )}
40
+ >
41
+ <GitBranch className={cn(
42
+ 'h-3.5 w-3.5 shrink-0',
43
+ branch.isCurrent ? 'text-primary' : 'text-muted-foreground',
44
+ )} />
45
+
46
+ <span className={cn(
47
+ 'min-w-0 flex-1 truncate font-mono text-xs',
48
+ branch.isCurrent && 'text-foreground font-medium',
49
+ )}>
50
+ {branch.name}
51
+ </span>
52
+
53
+ {/* Scope badge */}
54
+ {branch.scopeId && (
55
+ <Badge variant="secondary" className="shrink-0 text-xs">
56
+ #{branch.scopeId}
57
+ </Badge>
58
+ )}
59
+
60
+ {/* Ahead/behind */}
61
+ {branch.aheadBehind && (
62
+ <div className="flex shrink-0 items-center gap-1 text-xs">
63
+ {branch.aheadBehind.ahead > 0 && (
64
+ <span className="flex items-center gap-0.5 text-bid-green">
65
+ <ArrowUp className="h-2.5 w-2.5" />
66
+ {branch.aheadBehind.ahead}
67
+ </span>
68
+ )}
69
+ {branch.aheadBehind.behind > 0 && (
70
+ <span className="flex items-center gap-0.5 text-ask-red">
71
+ <ArrowDown className="h-2.5 w-2.5" />
72
+ {branch.aheadBehind.behind}
73
+ </span>
74
+ )}
75
+ </div>
76
+ )}
77
+
78
+ {/* SHA */}
79
+ <code className="shrink-0 font-mono text-xs text-muted-foreground/60">
80
+ {branch.headSha}
81
+ </code>
82
+
83
+ {/* Last commit time */}
84
+ {branch.headDate && (
85
+ <span className="shrink-0 text-xs text-muted-foreground/60">
86
+ {formatDistanceToNow(new Date(branch.headDate), { addSuffix: true })}
87
+ </span>
88
+ )}
89
+ </div>
90
+ ))}
91
+ </div>
92
+ );
93
+ }
@@ -0,0 +1,105 @@
1
+ import { useState } from 'react';
2
+ import { GitBranch, GitFork, ArrowRight } from 'lucide-react';
3
+ import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
4
+ import { Badge } from '@/components/ui/badge';
5
+ import { ScrollArea } from '@/components/ui/scroll-area';
6
+ import { BranchList } from './BranchList';
7
+ import { WorktreeList } from './WorktreeList';
8
+ import type { BranchInfoData, WorktreeDetail, DriftPair } from '@/types';
9
+ import { cn } from '@/lib/utils';
10
+
11
+ interface Props {
12
+ branches: BranchInfoData[];
13
+ worktrees: WorktreeDetail[];
14
+ drift: DriftPair[];
15
+ branchingMode: 'trunk' | 'worktree';
16
+ }
17
+
18
+ type PanelTab = 'branches' | 'worktrees';
19
+
20
+ function driftColor(count: number): string {
21
+ if (count === 0) return 'text-bid-green';
22
+ if (count <= 5) return 'text-accent-blue';
23
+ if (count <= 20) return 'text-warning-amber';
24
+ return 'text-ask-red';
25
+ }
26
+
27
+ export function BranchPanel({ branches, worktrees, drift, branchingMode }: Props) {
28
+ const showWorktrees = branchingMode === 'worktree';
29
+ const [tab, setTab] = useState<PanelTab>('branches');
30
+
31
+ return (
32
+ <Card>
33
+ <CardHeader className="pb-2">
34
+ <div className="flex items-center justify-between">
35
+ <CardTitle className="flex items-center gap-2 text-base">
36
+ <GitBranch className="h-4 w-4 text-primary" />
37
+ {showWorktrees ? (tab === 'branches' ? 'Branches' : 'Worktrees') : 'Branches'}
38
+ <Badge variant="secondary">
39
+ {tab === 'branches' ? branches.filter(b => !b.isRemote).length : worktrees.length}
40
+ </Badge>
41
+ </CardTitle>
42
+ {showWorktrees && (
43
+ <div className="flex gap-1">
44
+ <button
45
+ onClick={() => setTab('branches')}
46
+ className={cn(
47
+ 'rounded-md px-2.5 py-1 text-xs transition-colors',
48
+ tab === 'branches'
49
+ ? 'bg-surface-light text-foreground'
50
+ : 'text-muted-foreground hover:text-foreground',
51
+ )}
52
+ >
53
+ <GitBranch className="inline h-3 w-3 mr-1" />
54
+ Branches
55
+ </button>
56
+ <button
57
+ onClick={() => setTab('worktrees')}
58
+ className={cn(
59
+ 'rounded-md px-2.5 py-1 text-xs transition-colors',
60
+ tab === 'worktrees'
61
+ ? 'bg-surface-light text-foreground'
62
+ : 'text-muted-foreground hover:text-foreground',
63
+ )}
64
+ >
65
+ <GitFork className="inline h-3 w-3 mr-1" />
66
+ Worktrees
67
+ </button>
68
+ </div>
69
+ )}
70
+ </div>
71
+ </CardHeader>
72
+ <CardContent>
73
+ <ScrollArea className="max-h-[400px]">
74
+ {tab === 'branches' ? (
75
+ <BranchList branches={branches} />
76
+ ) : (
77
+ <WorktreeList worktrees={worktrees} />
78
+ )}
79
+ </ScrollArea>
80
+
81
+ {/* Drift indicators */}
82
+ {drift.length > 0 && (
83
+ <div className="mt-4 border-t border-border pt-3">
84
+ <h4 className="mb-2 text-xs font-medium text-muted-foreground">Branch Drift</h4>
85
+ <div className="space-y-1.5">
86
+ {drift.map(d => (
87
+ <div key={`${d.from}-${d.to}`} className="flex items-center gap-2 text-xs">
88
+ <code className="font-mono text-muted-foreground">{d.from}</code>
89
+ <ArrowRight className="h-3 w-3 text-muted-foreground/50" />
90
+ <code className="font-mono text-muted-foreground">{d.to}</code>
91
+ <Badge
92
+ variant="outline"
93
+ className={cn('ml-auto text-xs', driftColor(d.count))}
94
+ >
95
+ {d.count === 0 ? 'in sync' : `${d.count} commit${d.count !== 1 ? 's' : ''}`}
96
+ </Badge>
97
+ </div>
98
+ ))}
99
+ </div>
100
+ </div>
101
+ )}
102
+ </CardContent>
103
+ </Card>
104
+ );
105
+ }
@@ -0,0 +1,100 @@
1
+ import { useState, useMemo } from 'react';
2
+ import { GitCommit } from 'lucide-react';
3
+ import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
4
+ import { Badge } from '@/components/ui/badge';
5
+ import { ScrollArea } from '@/components/ui/scroll-area';
6
+ import { CommitRow } from './CommitRow';
7
+ import type { CommitEntry, BranchInfoData } from '@/types';
8
+
9
+ interface Props {
10
+ commits: CommitEntry[];
11
+ branches: BranchInfoData[];
12
+ hasMore: boolean;
13
+ onLoadMore: () => void;
14
+ }
15
+
16
+ type FilterTab = 'all' | 'main' | 'feature';
17
+
18
+ export function CommitLog({ commits, branches, hasMore, onLoadMore }: Props) {
19
+ const [filter, setFilter] = useState<FilterTab>('all');
20
+
21
+ const mainBranches = useMemo(() => {
22
+ const names = new Set(['main', 'master', 'dev', 'develop', 'staging', 'production']);
23
+ for (const b of branches) {
24
+ if (!b.isRemote && !b.name.startsWith('feat/') && !b.name.startsWith('fix/')) {
25
+ names.add(b.name);
26
+ }
27
+ }
28
+ return names;
29
+ }, [branches]);
30
+
31
+ const filtered = useMemo(() => {
32
+ if (filter === 'all') return commits;
33
+ if (filter === 'main') {
34
+ return commits.filter(c => !c.branch || mainBranches.has(c.branch));
35
+ }
36
+ // feature
37
+ return commits.filter(c => c.branch && !mainBranches.has(c.branch));
38
+ }, [commits, filter, mainBranches]);
39
+
40
+ const tabs: { key: FilterTab; label: string }[] = [
41
+ { key: 'all', label: 'All' },
42
+ { key: 'main', label: 'Main' },
43
+ { key: 'feature', label: 'Feature' },
44
+ ];
45
+
46
+ return (
47
+ <Card className="lg:col-span-2">
48
+ <CardHeader className="pb-2">
49
+ <div className="flex items-center justify-between">
50
+ <CardTitle className="flex items-center gap-2 text-base">
51
+ <GitCommit className="h-4 w-4 text-primary" />
52
+ Commits
53
+ <Badge variant="secondary">{filtered.length}</Badge>
54
+ </CardTitle>
55
+ <div className="flex gap-1">
56
+ {tabs.map(tab => (
57
+ <button
58
+ key={tab.key}
59
+ onClick={() => setFilter(tab.key)}
60
+ className={`rounded-md px-2.5 py-1 text-xs transition-colors ${
61
+ filter === tab.key
62
+ ? 'bg-surface-light text-foreground'
63
+ : 'text-muted-foreground hover:text-foreground'
64
+ }`}
65
+ >
66
+ {tab.label}
67
+ </button>
68
+ ))}
69
+ </div>
70
+ </div>
71
+ </CardHeader>
72
+ <CardContent>
73
+ {filtered.length === 0 ? (
74
+ <div className="py-8 text-center">
75
+ <GitCommit className="mx-auto mb-3 h-10 w-10 text-muted-foreground/50" />
76
+ <p className="text-sm text-muted-foreground">No commits found.</p>
77
+ </div>
78
+ ) : (
79
+ <>
80
+ <ScrollArea className="max-h-[600px]">
81
+ <div className="space-y-0.5">
82
+ {filtered.map(commit => (
83
+ <CommitRow key={commit.sha} commit={commit} />
84
+ ))}
85
+ </div>
86
+ </ScrollArea>
87
+ {hasMore && (
88
+ <button
89
+ onClick={onLoadMore}
90
+ className="mt-3 w-full rounded border border-border py-2 text-xs text-muted-foreground transition-colors hover:bg-surface-light hover:text-foreground"
91
+ >
92
+ Load more commits
93
+ </button>
94
+ )}
95
+ </>
96
+ )}
97
+ </CardContent>
98
+ </Card>
99
+ );
100
+ }
@@ -0,0 +1,47 @@
1
+ import { formatDistanceToNow } from 'date-fns';
2
+ import { Badge } from '@/components/ui/badge';
3
+ import type { CommitEntry } from '@/types';
4
+
5
+ interface Props {
6
+ commit: CommitEntry;
7
+ }
8
+
9
+ export function CommitRow({ commit }: Props) {
10
+ return (
11
+ <div className="flex items-center gap-3 rounded px-2.5 py-1.5 transition-colors hover:bg-surface-light">
12
+ {/* SHA */}
13
+ <code className="shrink-0 font-mono text-xs text-primary">
14
+ {commit.shortSha}
15
+ </code>
16
+
17
+ {/* Message */}
18
+ <span className="min-w-0 flex-1 truncate text-sm">
19
+ {commit.message}
20
+ </span>
21
+
22
+ {/* Branch ref badge */}
23
+ {commit.branch && (
24
+ <Badge variant="outline" className="shrink-0 text-xs font-normal">
25
+ {commit.branch}
26
+ </Badge>
27
+ )}
28
+
29
+ {/* Scope link */}
30
+ {commit.scopeId && (
31
+ <Badge variant="secondary" className="shrink-0 text-xs">
32
+ #{commit.scopeId}
33
+ </Badge>
34
+ )}
35
+
36
+ {/* Author */}
37
+ <span className="shrink-0 text-xs text-muted-foreground">
38
+ {commit.author}
39
+ </span>
40
+
41
+ {/* Time */}
42
+ <span className="shrink-0 text-xs text-muted-foreground/60">
43
+ {formatDistanceToNow(new Date(commit.date), { addSuffix: true })}
44
+ </span>
45
+ </div>
46
+ );
47
+ }
@@ -0,0 +1,110 @@
1
+ import { useState, useEffect, useCallback } from 'react';
2
+ import { Github, ChevronDown, ChevronRight, Eye, Lock, Globe } from 'lucide-react';
3
+ import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
4
+ import { Badge } from '@/components/ui/badge';
5
+ import { GitHubSetupGuide } from './GitHubSetupGuide';
6
+ import { PullRequestList } from './PullRequestList';
7
+ import type { GitHubStatus, PullRequestInfo } from '@/types';
8
+
9
+ interface Props {
10
+ github: GitHubStatus | null;
11
+ }
12
+
13
+ export function GitHubPanel({ github }: Props) {
14
+ const [expanded, setExpanded] = useState(true);
15
+ const [prs, setPrs] = useState<PullRequestInfo[]>([]);
16
+
17
+ const fetchPRs = useCallback(async () => {
18
+ if (!github?.connected) return;
19
+ try {
20
+ const res = await fetch('/api/orbital/github/prs');
21
+ if (res.ok) setPrs(await res.json());
22
+ } catch { /* ok */ }
23
+ }, [github?.connected]);
24
+
25
+ useEffect(() => {
26
+ if (expanded && github?.connected) fetchPRs();
27
+ }, [expanded, github?.connected, fetchPRs]);
28
+
29
+ const VisibilityIcon = github?.repo?.visibility === 'public' ? Globe : Lock;
30
+
31
+ return (
32
+ <Card className="mt-6">
33
+ <CardHeader
34
+ className="cursor-pointer select-none"
35
+ onClick={() => setExpanded(!expanded)}
36
+ >
37
+ <CardTitle className="flex items-center gap-2 text-base">
38
+ {expanded ? (
39
+ <ChevronDown className="h-4 w-4 text-muted-foreground" />
40
+ ) : (
41
+ <ChevronRight className="h-4 w-4 text-muted-foreground" />
42
+ )}
43
+ <Github className="h-4 w-4 text-primary" />
44
+ GitHub
45
+ {github?.connected && github.repo && (
46
+ <Badge variant="secondary" className="ml-1 text-xs gap-1">
47
+ <VisibilityIcon className="h-2.5 w-2.5" />
48
+ {github.repo.fullName}
49
+ </Badge>
50
+ )}
51
+ {github?.openPRs ? (
52
+ <Badge variant="outline" className="text-xs">
53
+ {github.openPRs} open PR{github.openPRs !== 1 ? 's' : ''}
54
+ </Badge>
55
+ ) : null}
56
+ </CardTitle>
57
+ </CardHeader>
58
+
59
+ {expanded && (
60
+ <CardContent>
61
+ {!github || !github.connected ? (
62
+ <GitHubSetupGuide error={github?.error ?? null} />
63
+ ) : (
64
+ <div className="space-y-4">
65
+ {/* Repo info */}
66
+ {github.repo && (
67
+ <div className="grid grid-cols-2 gap-x-4 gap-y-1 text-xs">
68
+ <span className="text-muted-foreground">Repository</span>
69
+ <a
70
+ href={github.repo.url}
71
+ target="_blank"
72
+ rel="noopener noreferrer"
73
+ className="text-primary hover:underline"
74
+ >
75
+ {github.repo.fullName}
76
+ </a>
77
+
78
+ <span className="text-muted-foreground">Default branch</span>
79
+ <code className="font-mono">{github.repo.defaultBranch}</code>
80
+
81
+ <span className="text-muted-foreground">Visibility</span>
82
+ <span className="flex items-center gap-1 capitalize">
83
+ <VisibilityIcon className="h-3 w-3" />
84
+ {github.repo.visibility}
85
+ </span>
86
+
87
+ {github.authUser && (
88
+ <>
89
+ <span className="text-muted-foreground">Signed in as</span>
90
+ <span>{github.authUser}</span>
91
+ </>
92
+ )}
93
+ </div>
94
+ )}
95
+
96
+ {/* PRs */}
97
+ <div className="border-t border-border pt-3">
98
+ <h4 className="mb-2 flex items-center gap-2 text-xs font-medium text-muted-foreground">
99
+ <Eye className="h-3 w-3" />
100
+ Open Pull Requests
101
+ </h4>
102
+ <PullRequestList prs={prs} />
103
+ </div>
104
+ </div>
105
+ )}
106
+ </CardContent>
107
+ )}
108
+ </Card>
109
+ );
110
+ }
@@ -0,0 +1,52 @@
1
+ import { Terminal, ExternalLink } from 'lucide-react';
2
+
3
+ interface Props {
4
+ error: string | null;
5
+ }
6
+
7
+ export function GitHubSetupGuide({ error }: Props) {
8
+ const isNotInstalled = error?.includes('not installed');
9
+
10
+ return (
11
+ <div className="space-y-4">
12
+ <p className="text-sm text-muted-foreground">
13
+ {isNotInstalled
14
+ ? 'The GitHub CLI (gh) is required to connect your repository.'
15
+ : 'Authenticate with GitHub to see PRs, repo info, and more.'}
16
+ </p>
17
+
18
+ <div className="space-y-3">
19
+ {isNotInstalled && (
20
+ <div className="rounded border border-border bg-surface-light p-3">
21
+ <h4 className="mb-1.5 text-xs font-medium">1. Install GitHub CLI</h4>
22
+ <div className="flex items-center gap-2 rounded bg-background px-3 py-2 font-mono text-xs">
23
+ <Terminal className="h-3 w-3 shrink-0 text-muted-foreground" />
24
+ <code>brew install gh</code>
25
+ </div>
26
+ <p className="mt-1.5 text-xs text-muted-foreground">
27
+ Or visit{' '}
28
+ <a
29
+ href="https://cli.github.com"
30
+ target="_blank"
31
+ rel="noopener noreferrer"
32
+ className="inline-flex items-center gap-0.5 text-primary hover:underline"
33
+ >
34
+ cli.github.com <ExternalLink className="h-2.5 w-2.5" />
35
+ </a>
36
+ </p>
37
+ </div>
38
+ )}
39
+
40
+ <div className="rounded border border-border bg-surface-light p-3">
41
+ <h4 className="mb-1.5 text-xs font-medium">
42
+ {isNotInstalled ? '2' : '1'}. Authenticate
43
+ </h4>
44
+ <div className="flex items-center gap-2 rounded bg-background px-3 py-2 font-mono text-xs">
45
+ <Terminal className="h-3 w-3 shrink-0 text-muted-foreground" />
46
+ <code>gh auth login</code>
47
+ </div>
48
+ </div>
49
+ </div>
50
+ </div>
51
+ );
52
+ }