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,454 @@
1
+ import { useState, useMemo, useEffect } from 'react';
2
+ import {
3
+ DndContext,
4
+ PointerSensor,
5
+ KeyboardSensor,
6
+ useSensor,
7
+ useSensors,
8
+ } from '@dnd-kit/core';
9
+ import { LayoutDashboard, X } from 'lucide-react';
10
+ import { useScopes } from '@/hooks/useScopes';
11
+ import { useCardDisplay } from '@/hooks/useCardDisplay';
12
+ import { ActiveDispatchContext, useActiveDispatchProvider } from '@/hooks/useActiveDispatches';
13
+ import { useScopeFilters } from '@/hooks/useScopeFilters';
14
+ import { useBoardSettings, sortScopes } from '@/hooks/useBoardSettings';
15
+ import { useSwimlaneBoardSettings } from '@/hooks/useSwimlaneBoardSettings';
16
+ import { useZoomModifier } from '@/hooks/useZoomModifier';
17
+ import { useKanbanDnd } from '@/hooks/useKanbanDnd';
18
+ import { useSprints } from '@/hooks/useSprints';
19
+ import { useSprintPreflight } from '@/hooks/useSprintPreflight';
20
+ import { useIdeaActions } from '@/hooks/useIdeaActions';
21
+ import { useSearch } from '@/hooks/useSearch';
22
+ import { useStatusBarHighlight } from '@/hooks/useStatusBarHighlight';
23
+ import { useWorkflow } from '@/hooks/useWorkflow';
24
+ import { KanbanColumn } from '@/components/KanbanColumn';
25
+ import { CardDisplayToggle } from '@/components/CardDisplayToggle';
26
+ import { ViewModeSelector } from '@/components/ViewModeSelector';
27
+ import { SwimlaneBoardView } from '@/components/SwimlaneBoardView';
28
+ import { DragOverlay } from '@/components/DragOverlay';
29
+ import { DispatchPopover } from '@/components/DispatchPopover';
30
+ import { DispatchModal } from '@/components/DispatchModal';
31
+ import { ScopeDetailModal } from '@/components/ScopeDetailModal';
32
+ import { IdeaFormDialog } from '@/components/IdeaFormDialog';
33
+ import { IdeaDetailModal } from '@/components/IdeaDetailModal';
34
+ import { ScopeFilterBar } from '@/components/ScopeFilterBar';
35
+ import { SprintPreflightModal } from '@/components/SprintPreflightModal';
36
+ import { BatchPreflightModal } from '@/components/BatchPreflightModal';
37
+ import { SprintDependencyDialog } from '@/components/SprintDependencyDialog';
38
+ import { ColumnHeaderActions } from '@/components/ColumnHeaderActions';
39
+ import { Badge } from '@/components/ui/badge';
40
+ import { sprintAwareCollision } from '@/lib/collisionDetection';
41
+ import { computeSwimLanes } from '@/lib/swimlane';
42
+ import type { Scope } from '@/types';
43
+
44
+ export function ScopeBoard() {
45
+ const { scopes, loading } = useScopes();
46
+ const { engine } = useWorkflow();
47
+ const activeDispatchCtx = useActiveDispatchProvider();
48
+ const { sortField, sortDirection, setSort, collapsed, toggleCollapse } = useBoardSettings();
49
+ const { viewMode, setViewMode, groupField, setGroupField, collapsedLanes, toggleLaneCollapse } = useSwimlaneBoardSettings();
50
+ const { display: cardDisplay, toggle: toggleCardDisplay, hiddenCount } = useCardDisplay();
51
+ const [selectedScopeId, setSelectedScopeId] = useState<number | null>(null);
52
+ const selectedScope = useMemo(() => scopes.find((s) => s.id === selectedScopeId) ?? null, [scopes, selectedScopeId]);
53
+ const [selectedIdea, setSelectedIdea] = useState<Scope | null>(null);
54
+ const [pendingBatchDispatch, setPendingBatchDispatch] = useState<number | null>(null);
55
+
56
+ // Dynamic board columns from engine
57
+ const boardColumns = useMemo(() => engine.getBoardColumns(), [engine]);
58
+
59
+ const {
60
+ sprints,
61
+ createSprint,
62
+ renameSprint,
63
+ deleteSprint,
64
+ addScopes: addScopesToSprint,
65
+ removeScopes: removeScopesFromSprint,
66
+ dispatchSprint,
67
+ getGraph,
68
+ } = useSprints();
69
+
70
+ const [editingSprintId, setEditingSprintId] = useState<number | null>(null);
71
+
72
+ const {
73
+ filters,
74
+ toggleFilter,
75
+ clearField,
76
+ clearAll,
77
+ hasActiveFilters,
78
+ filteredScopes,
79
+ optionsWithCounts,
80
+ } = useScopeFilters(scopes);
81
+
82
+ const search = useSearch(filteredScopes);
83
+ const { highlightedScopeId, clearHighlight } = useStatusBarHighlight();
84
+
85
+ // Merge search dimming with statusbar highlight dimming
86
+ const mergedDimmedIds = useMemo(() => {
87
+ if (highlightedScopeId == null) return search.dimmedIds;
88
+ const dimmed = new Set<number>();
89
+ for (const scope of search.displayScopes) {
90
+ if (scope.id !== highlightedScopeId) dimmed.add(scope.id);
91
+ }
92
+ return dimmed;
93
+ }, [highlightedScopeId, search.dimmedIds, search.displayScopes]);
94
+
95
+ // Click anywhere to clear highlight
96
+ useEffect(() => {
97
+ if (highlightedScopeId == null) return;
98
+ const timer = setTimeout(() => {
99
+ document.addEventListener('click', clearHighlight, { once: true });
100
+ }, 100);
101
+ return () => {
102
+ clearTimeout(timer);
103
+ document.removeEventListener('click', clearHighlight);
104
+ };
105
+ }, [highlightedScopeId, clearHighlight]);
106
+
107
+ const {
108
+ state: dndState,
109
+ onDragStart,
110
+ onDragOver,
111
+ onDragEnd,
112
+ confirmTransition,
113
+ cancelTransition,
114
+ dismissError,
115
+ openModalFromPopover,
116
+ openIdeaForm,
117
+ closeIdeaForm,
118
+ submitIdea,
119
+ dismissSprintDispatch,
120
+ dismissUnmetDeps,
121
+ resolveUnmetDeps,
122
+ showUnmetDeps,
123
+ } = useKanbanDnd({
124
+ scopes: search.displayScopes,
125
+ sprints,
126
+ onAddToSprint: addScopesToSprint,
127
+ onRemoveFromSprint: removeScopesFromSprint,
128
+ });
129
+
130
+ // 8px activation constraint so clicks pass through to detail modal
131
+ const sensors = useSensors(
132
+ useSensor(PointerSensor, { activationConstraint: { distance: 8 } }),
133
+ useSensor(KeyboardSensor),
134
+ );
135
+ const modifiers = useZoomModifier();
136
+
137
+ // Build scope lookup from full set so sprint containers always resolve
138
+ const scopeLookup = useMemo(() => {
139
+ const map = new Map<number, Scope>();
140
+ for (const scope of scopes) map.set(scope.id, scope);
141
+ return map;
142
+ }, [scopes]);
143
+
144
+ const scopesByStatus = useMemo(() => {
145
+ const groups: Record<string, Scope[]> = {};
146
+ for (const col of boardColumns) groups[col.id] = [];
147
+
148
+ const entryPointId = engine.getEntryPoint().id;
149
+ for (const scope of search.displayScopes) {
150
+ if (groups[scope.status]) {
151
+ groups[scope.status].push(scope);
152
+ } else {
153
+ groups[entryPointId]?.push(scope);
154
+ }
155
+ }
156
+
157
+ // Apply sort within each column
158
+ for (const key of Object.keys(groups)) {
159
+ groups[key] = sortScopes(groups[key], sortField, sortDirection);
160
+ }
161
+
162
+ return groups;
163
+ }, [search.displayScopes, sortField, sortDirection, boardColumns, engine]);
164
+
165
+ // Sprints/batches by column using target_column (W-6)
166
+ const sprintsByColumn = useMemo(() => {
167
+ const map: Record<string, typeof sprints> = {};
168
+ for (const group of sprints) {
169
+ // Hide completed batches — failed batches stay visible for attention
170
+ if (group.group_type === 'batch' && group.status === 'completed') continue;
171
+ // Active sprints render in implementing; everything else uses target_column
172
+ const col = group.group_type === 'sprint' && group.status !== 'assembling'
173
+ ? 'implementing'
174
+ : group.target_column;
175
+ (map[col] ??= []).push(group);
176
+ }
177
+ return map;
178
+ }, [sprints]);
179
+
180
+ // Global set of scope IDs in active sprint/batch groups across ALL columns.
181
+ // Used for cross-column deduplication so a scope never renders as both
182
+ // a loose card in one column and inside a group container in another.
183
+ const globalSprintScopeIds = useMemo(() => {
184
+ const ids = new Set<number>();
185
+ for (const group of sprints) {
186
+ if (group.group_type === 'batch' && group.status === 'completed') continue;
187
+ for (const scopeId of group.scope_ids) {
188
+ ids.add(scopeId);
189
+ }
190
+ }
191
+ return ids;
192
+ }, [sprints]);
193
+
194
+ // Swimlane computation (only when in swimlane mode)
195
+ const swimLanes = useMemo(() => {
196
+ if (viewMode !== 'swimlane') return [];
197
+ return computeSwimLanes(search.displayScopes, groupField, sortField, sortDirection);
198
+ }, [viewMode, search.displayScopes, groupField, sortField, sortDirection]);
199
+
200
+ // Compute valid drop targets for the currently dragged item
201
+ const validTargets = useMemo(() => {
202
+ if (dndState.activeScope) {
203
+ return new Set(engine.getValidTargets(dndState.activeScope.status));
204
+ }
205
+ if (dndState.activeSprint) {
206
+ return new Set<string>(['implementing']);
207
+ }
208
+ return new Set<string>();
209
+ }, [dndState.activeScope, dndState.activeSprint, engine]);
210
+
211
+ // ─── Sprint Preflight ───────────────────────────────────
212
+ const preflight = useSprintPreflight(
213
+ dndState.pendingSprintDispatch,
214
+ getGraph,
215
+ dispatchSprint,
216
+ dismissSprintDispatch,
217
+ );
218
+
219
+ // ─── Idea Actions ─────────────────────────────────────────
220
+ const { surpriseLoading, handleSurprise, handleApproveGhost, handleRejectGhost } =
221
+ useIdeaActions(closeIdeaForm, setSelectedIdea);
222
+
223
+ if (loading) {
224
+ return (
225
+ <div className="flex h-64 items-center justify-center">
226
+ <div className="h-8 w-8 animate-spin rounded-full border-2 border-primary border-t-transparent" />
227
+ </div>
228
+ );
229
+ }
230
+
231
+ return (
232
+ <ActiveDispatchContext.Provider value={activeDispatchCtx}>
233
+ <DndContext
234
+ sensors={sensors}
235
+ modifiers={modifiers}
236
+ collisionDetection={sprintAwareCollision}
237
+ onDragStart={onDragStart}
238
+ onDragOver={onDragOver}
239
+ onDragEnd={onDragEnd}
240
+ >
241
+ <div className="flex flex-1 min-h-0 flex-col">
242
+ {/* Header */}
243
+ <div className="mb-4 flex items-center justify-between">
244
+ <div className="flex items-center gap-3">
245
+ <LayoutDashboard className="h-4 w-4 text-primary" />
246
+ <h1 className="text-xl font-light">Kanban</h1>
247
+ <Badge variant="secondary" className="ml-2">
248
+ {search.hasSearch
249
+ ? `${search.matchCount} / ${scopes.length} scopes`
250
+ : hasActiveFilters
251
+ ? `${filteredScopes.length} / ${scopes.length} scopes`
252
+ : `${scopes.length} scopes`}
253
+ </Badge>
254
+ </div>
255
+ <div className="flex items-center gap-2">
256
+ <ViewModeSelector
257
+ viewMode={viewMode}
258
+ groupField={groupField}
259
+ onViewModeChange={setViewMode}
260
+ onGroupFieldChange={setGroupField}
261
+ />
262
+ <CardDisplayToggle display={cardDisplay} onToggle={toggleCardDisplay} hiddenCount={hiddenCount} />
263
+ </div>
264
+ </div>
265
+
266
+ {/* Error toast */}
267
+ {dndState.error && (
268
+ <div className="mb-3 flex items-center gap-2 rounded border border-red-500/30 bg-red-500/10 px-3 py-2 text-xs text-red-400">
269
+ <span className="flex-1">{dndState.error}</span>
270
+ <button onClick={dismissError} className="shrink-0 hover:text-red-200 transition-colors">
271
+ <X className="h-3.5 w-3.5" />
272
+ </button>
273
+ </div>
274
+ )}
275
+
276
+ {/* Filters + Search */}
277
+ <ScopeFilterBar
278
+ filters={filters}
279
+ optionsWithCounts={optionsWithCounts}
280
+ onToggle={toggleFilter}
281
+ onClearField={clearField}
282
+ onClearAll={clearAll}
283
+ hasActiveFilters={hasActiveFilters}
284
+ searchQuery={search.query}
285
+ searchMode={search.mode}
286
+ searchIsStale={search.isStale}
287
+ onSearchChange={search.setQuery}
288
+ onSearchModeChange={search.setMode}
289
+ />
290
+
291
+ {/* Board — Kanban or Swimlane */}
292
+ {viewMode === 'swimlane' ? (
293
+ <SwimlaneBoardView
294
+ lanes={swimLanes}
295
+ columns={boardColumns}
296
+ collapsedColumns={collapsed}
297
+ collapsedLanes={collapsedLanes}
298
+ onToggleLane={toggleLaneCollapse}
299
+ onToggleCollapse={toggleCollapse}
300
+ onScopeClick={(scope) => scope.status === engine.getEntryPoint().id ? setSelectedIdea(scope) : setSelectedScopeId(scope.id)}
301
+ cardDisplay={cardDisplay}
302
+ dimmedIds={mergedDimmedIds}
303
+ isDragActive={!!(dndState.activeScope || dndState.activeSprint)}
304
+ validTargets={validTargets}
305
+ sprints={sprints}
306
+ />
307
+ ) : (
308
+ <div className="min-h-0 flex-1 overflow-x-auto overflow-y-hidden">
309
+ <div className="flex h-full w-max gap-2 pb-4">
310
+ {boardColumns.map((col) => (
311
+ <KanbanColumn
312
+ key={col.id}
313
+ id={col.id}
314
+ label={col.label}
315
+ color={col.color}
316
+ scopes={scopesByStatus[col.id] ?? []}
317
+ sprints={sprintsByColumn[col.id]}
318
+ scopeLookup={scopeLookup}
319
+ globalSprintScopeIds={globalSprintScopeIds}
320
+ onScopeClick={(scope) => scope.status === engine.getEntryPoint().id ? setSelectedIdea(scope) : setSelectedScopeId(scope.id)}
321
+ onDeleteSprint={deleteSprint}
322
+ onDispatchSprint={(id) => setPendingBatchDispatch(id)}
323
+ onRenameSprint={(id, name) => renameSprint(id, name)}
324
+ editingSprintId={editingSprintId}
325
+ onSprintEditingDone={() => setEditingSprintId(null)}
326
+ onAddAllToSprint={async (sprintId, scopeIds) => {
327
+ const result = await addScopesToSprint(sprintId, scopeIds);
328
+ if (result && result.unmet_dependencies.length > 0) {
329
+ showUnmetDeps(sprintId, result.unmet_dependencies);
330
+ }
331
+ }}
332
+ isDragActive={!!(dndState.activeScope || dndState.activeSprint)}
333
+ isValidDrop={validTargets.has(col.id)}
334
+ sortField={sortField}
335
+ sortDirection={sortDirection}
336
+ onSetSort={setSort}
337
+ collapsed={collapsed.has(col.id)}
338
+ onToggleCollapse={() => toggleCollapse(col.id)}
339
+ cardDisplay={cardDisplay}
340
+ dimmedIds={mergedDimmedIds}
341
+ headerExtra={
342
+ <ColumnHeaderActions
343
+ columnId={col.id}
344
+ dispatching={dndState.dispatching}
345
+ onOpenIdeaForm={openIdeaForm}
346
+ onCreateGroup={async (name, options) => {
347
+ const sprint = await createSprint(name, options);
348
+ if (sprint) setEditingSprintId(sprint.id);
349
+ return sprint;
350
+ }}
351
+ />
352
+ }
353
+ />
354
+ ))}
355
+ </div>
356
+ </div>
357
+ )}
358
+
359
+ {/* Drag overlay — floating preview */}
360
+ <DragOverlay
361
+ activeScope={dndState.activeScope}
362
+ activeSprint={dndState.activeSprint}
363
+ cardDisplay={cardDisplay}
364
+ />
365
+
366
+ {/* Quick confirm dialog */}
367
+ <DispatchPopover
368
+ open={dndState.showPopover}
369
+ scope={dndState.pending?.scope ?? null}
370
+ transition={dndState.pending?.transition ?? null}
371
+ hasActiveSession={dndState.pending?.hasActiveSession ?? false}
372
+ onConfirm={confirmTransition}
373
+ onCancel={cancelTransition}
374
+ onViewDetails={openModalFromPopover}
375
+ />
376
+
377
+ {/* Full confirm modal */}
378
+ <DispatchModal
379
+ open={dndState.showModal}
380
+ scope={dndState.pending?.scope ?? null}
381
+ transition={dndState.pending?.transition ?? null}
382
+ hasActiveSession={dndState.pending?.hasActiveSession ?? false}
383
+ onConfirm={confirmTransition}
384
+ onCancel={cancelTransition}
385
+ />
386
+
387
+ {/* Scope detail modal */}
388
+ <ScopeDetailModal
389
+ scope={selectedScope}
390
+ open={!!selectedScope}
391
+ onClose={() => setSelectedScopeId(null)}
392
+ />
393
+
394
+ {/* Idea form dialog */}
395
+ <IdeaFormDialog
396
+ open={dndState.showIdeaForm}
397
+ loading={dndState.dispatching}
398
+ onSubmit={submitIdea}
399
+ onCancel={closeIdeaForm}
400
+ onSurprise={handleSurprise}
401
+ surpriseLoading={surpriseLoading}
402
+ />
403
+
404
+ {/* Idea detail modal */}
405
+ <IdeaDetailModal
406
+ scope={selectedIdea}
407
+ open={!!selectedIdea}
408
+ onClose={() => setSelectedIdea(null)}
409
+ onDelete={async (id) => {
410
+ try {
411
+ const res = await fetch(`/api/orbital/ideas/${id}`, { method: 'DELETE' });
412
+ if (!res.ok) throw new Error(`HTTP ${res.status}`);
413
+ setSelectedIdea(null);
414
+ } catch {
415
+ setSelectedIdea(null);
416
+ }
417
+ }}
418
+ onApprove={handleApproveGhost}
419
+ onReject={handleRejectGhost}
420
+ />
421
+
422
+ {/* Sprint preflight modal */}
423
+ <SprintPreflightModal
424
+ open={preflight.showPreflight}
425
+ sprint={preflight.pendingSprint}
426
+ graph={preflight.graph}
427
+ loading={preflight.loading}
428
+ onConfirm={preflight.onConfirm}
429
+ onCancel={preflight.onCancel}
430
+ />
431
+
432
+ {/* Batch preflight modal */}
433
+ <BatchPreflightModal
434
+ open={pendingBatchDispatch != null}
435
+ batch={sprints.find((s) => s.id === pendingBatchDispatch) ?? null}
436
+ onConfirm={() => {
437
+ if (pendingBatchDispatch != null) dispatchSprint(pendingBatchDispatch);
438
+ setPendingBatchDispatch(null);
439
+ }}
440
+ onCancel={() => setPendingBatchDispatch(null)}
441
+ />
442
+
443
+ {/* Unmet dependency dialog */}
444
+ <SprintDependencyDialog
445
+ open={dndState.pendingUnmetDeps != null}
446
+ unmetDeps={dndState.pendingUnmetDeps ?? []}
447
+ onAddAll={resolveUnmetDeps}
448
+ onCancel={dismissUnmetDeps}
449
+ />
450
+ </div>
451
+ </DndContext>
452
+ </ActiveDispatchContext.Provider>
453
+ );
454
+ }