orbital-command 0.2.0 → 1.0.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 (431) hide show
  1. package/README.md +67 -42
  2. package/bin/commands/config.js +19 -0
  3. package/bin/commands/events.js +40 -0
  4. package/bin/commands/launch.js +126 -0
  5. package/bin/commands/manifest.js +283 -0
  6. package/bin/commands/registry.js +104 -0
  7. package/bin/commands/update.js +24 -0
  8. package/bin/lib/helpers.js +229 -0
  9. package/bin/orbital.js +147 -319
  10. package/dist/assets/Landing-CfQdHR0N.js +11 -0
  11. package/dist/assets/PrimitivesConfig-DThSipFy.js +32 -0
  12. package/dist/assets/QualityGates-B4kxM5UU.js +26 -0
  13. package/dist/assets/SessionTimeline-Bz1iZnmg.js +1 -0
  14. package/dist/assets/Settings-DLcZwbCT.js +12 -0
  15. package/dist/assets/SourceControl-BMNIz7Lt.js +36 -0
  16. package/dist/assets/WorkflowVisualizer-CxuSBOYu.js +69 -0
  17. package/dist/assets/arrow-down-DVPp6_qp.js +6 -0
  18. package/dist/assets/bot-NFaJBDn_.js +6 -0
  19. package/dist/assets/charts-LGLb8hyU.js +68 -0
  20. package/dist/assets/circle-x-IsFCkBZu.js +6 -0
  21. package/dist/assets/file-text-J1cebZXF.js +6 -0
  22. package/dist/assets/globe-WzeyHsUc.js +6 -0
  23. package/dist/assets/index-BdJ57EhC.css +1 -0
  24. package/dist/assets/index-o4ScMAuR.js +349 -0
  25. package/dist/assets/key-CKR8JJSj.js +6 -0
  26. package/dist/assets/minus-CHBsJyjp.js +6 -0
  27. package/dist/assets/radio-xqZaR-Uk.js +6 -0
  28. package/dist/assets/rocket-D_xvvNG6.js +6 -0
  29. package/dist/assets/shield-TdB1yv_a.js +6 -0
  30. package/dist/assets/ui-BmsSg9jU.js +53 -0
  31. package/dist/assets/useSocketListener-0L5yiN5i.js +1 -0
  32. package/dist/assets/useWorkflowEditor-CqeRWVQX.js +11 -0
  33. package/dist/assets/{vendor-Dzv9lrRc.js → vendor-Bqt8AJn2.js} +1 -1
  34. package/dist/assets/workflow-constants-Rw-GmgHZ.js +6 -0
  35. package/dist/assets/zap-C9wqYMpl.js +6 -0
  36. package/dist/favicon.svg +1 -0
  37. package/dist/index.html +6 -5
  38. package/dist/server/server/__tests__/data-routes.test.js +126 -0
  39. package/dist/server/server/__tests__/helpers/db.js +17 -0
  40. package/dist/server/server/__tests__/helpers/mock-emitter.js +8 -0
  41. package/dist/server/server/__tests__/scope-routes.test.js +138 -0
  42. package/dist/server/server/__tests__/sprint-routes.test.js +102 -0
  43. package/dist/server/server/__tests__/workflow-routes.test.js +107 -0
  44. package/dist/server/server/config-migrator.js +135 -0
  45. package/dist/server/server/config.js +51 -7
  46. package/dist/server/server/database.js +21 -28
  47. package/dist/server/server/global-config.js +143 -0
  48. package/dist/server/server/index.js +118 -276
  49. package/dist/server/server/init.js +243 -225
  50. package/dist/server/server/launch.js +29 -0
  51. package/dist/server/server/manifest-types.js +8 -0
  52. package/dist/server/server/manifest.js +454 -0
  53. package/dist/server/server/migrate-legacy.js +229 -0
  54. package/dist/server/server/parsers/event-parser.js +4 -1
  55. package/dist/server/server/parsers/event-parser.test.js +117 -0
  56. package/dist/server/server/parsers/scope-parser.js +74 -28
  57. package/dist/server/server/parsers/scope-parser.test.js +230 -0
  58. package/dist/server/server/project-context.js +265 -0
  59. package/dist/server/server/project-emitter.js +41 -0
  60. package/dist/server/server/project-manager.js +297 -0
  61. package/dist/server/server/routes/aggregate-routes.js +871 -0
  62. package/dist/server/server/routes/config-routes.js +41 -90
  63. package/dist/server/server/routes/data-routes.js +25 -123
  64. package/dist/server/server/routes/dispatch-routes.js +37 -15
  65. package/dist/server/server/routes/git-routes.js +74 -0
  66. package/dist/server/server/routes/manifest-routes.js +319 -0
  67. package/dist/server/server/routes/scope-routes.js +45 -28
  68. package/dist/server/server/routes/sync-routes.js +134 -0
  69. package/dist/server/server/routes/version-routes.js +1 -15
  70. package/dist/server/server/routes/workflow-routes.js +9 -3
  71. package/dist/server/server/schema.js +3 -0
  72. package/dist/server/server/services/batch-orchestrator.js +41 -17
  73. package/dist/server/server/services/claude-session-service.js +17 -14
  74. package/dist/server/server/services/config-service.js +10 -1
  75. package/dist/server/server/services/deploy-service.test.js +119 -0
  76. package/dist/server/server/services/event-service.js +64 -1
  77. package/dist/server/server/services/event-service.test.js +191 -0
  78. package/dist/server/server/services/gate-service.test.js +105 -0
  79. package/dist/server/server/services/git-service.js +108 -4
  80. package/dist/server/server/services/github-service.js +110 -2
  81. package/dist/server/server/services/readiness-service.test.js +190 -0
  82. package/dist/server/server/services/scope-cache.js +5 -1
  83. package/dist/server/server/services/scope-cache.test.js +142 -0
  84. package/dist/server/server/services/scope-service.js +222 -131
  85. package/dist/server/server/services/scope-service.test.js +137 -0
  86. package/dist/server/server/services/sprint-orchestrator.js +29 -15
  87. package/dist/server/server/services/sprint-service.js +23 -3
  88. package/dist/server/server/services/sprint-service.test.js +238 -0
  89. package/dist/server/server/services/sync-service.js +434 -0
  90. package/dist/server/server/services/sync-types.js +2 -0
  91. package/dist/server/server/services/workflow-service.js +26 -5
  92. package/dist/server/server/services/workflow-service.test.js +159 -0
  93. package/dist/server/server/settings-sync.js +284 -0
  94. package/dist/server/server/uninstall.js +195 -0
  95. package/dist/server/server/update-planner.js +279 -0
  96. package/dist/server/server/update.js +212 -0
  97. package/dist/server/server/utils/cc-hooks-parser.js +3 -0
  98. package/dist/server/server/utils/cc-hooks-parser.test.js +86 -0
  99. package/dist/server/server/utils/dispatch-utils.js +83 -24
  100. package/dist/server/server/utils/dispatch-utils.test.js +182 -0
  101. package/dist/server/server/utils/flag-builder.js +54 -0
  102. package/dist/server/server/utils/json-fields.js +14 -0
  103. package/dist/server/server/utils/json-fields.test.js +73 -0
  104. package/dist/server/server/utils/logger.js +37 -3
  105. package/dist/server/server/utils/package-info.js +30 -0
  106. package/dist/server/server/utils/route-helpers.js +47 -0
  107. package/dist/server/server/utils/route-helpers.test.js +115 -0
  108. package/dist/server/server/utils/terminal-launcher.js +79 -25
  109. package/dist/server/server/utils/worktree-manager.js +13 -4
  110. package/dist/server/server/validator.js +230 -0
  111. package/dist/server/server/watchers/event-watcher.js +28 -13
  112. package/dist/server/server/watchers/global-watcher.js +63 -0
  113. package/dist/server/server/watchers/scope-watcher.js +27 -12
  114. package/dist/server/server/wizard/config-editor.js +237 -0
  115. package/dist/server/server/wizard/detect.js +96 -0
  116. package/dist/server/server/wizard/doctor.js +115 -0
  117. package/dist/server/server/wizard/index.js +340 -0
  118. package/dist/server/server/wizard/phases/confirm.js +39 -0
  119. package/dist/server/server/wizard/phases/project-setup.js +90 -0
  120. package/dist/server/server/wizard/phases/setup-wizard.js +66 -0
  121. package/dist/server/server/wizard/phases/welcome.js +32 -0
  122. package/dist/server/server/wizard/phases/workflow-setup.js +22 -0
  123. package/dist/server/server/wizard/types.js +29 -0
  124. package/dist/server/server/wizard/ui.js +73 -0
  125. package/dist/server/shared/__fixtures__/workflow-configs.js +75 -0
  126. package/dist/server/shared/api-types.js +80 -1
  127. package/dist/server/shared/default-workflow.json +65 -0
  128. package/dist/server/shared/onboarding-tour.test.js +81 -0
  129. package/dist/server/shared/project-colors.js +24 -0
  130. package/dist/server/shared/workflow-config.test.js +84 -0
  131. package/dist/server/shared/workflow-engine.js +1 -1
  132. package/dist/server/shared/workflow-engine.test.js +302 -0
  133. package/dist/server/shared/workflow-normalizer.js +101 -0
  134. package/dist/server/shared/workflow-normalizer.test.js +100 -0
  135. package/dist/server/src/components/onboarding/tour-steps.js +84 -0
  136. package/package.json +34 -29
  137. package/schemas/orbital.config.schema.json +2 -5
  138. package/scripts/postinstall.js +18 -6
  139. package/scripts/release.sh +53 -0
  140. package/server/__tests__/data-routes.test.ts +151 -0
  141. package/server/__tests__/helpers/db.ts +19 -0
  142. package/server/__tests__/helpers/mock-emitter.ts +10 -0
  143. package/server/__tests__/scope-routes.test.ts +158 -0
  144. package/server/__tests__/sprint-routes.test.ts +118 -0
  145. package/server/__tests__/workflow-routes.test.ts +120 -0
  146. package/server/config-migrator.ts +160 -0
  147. package/server/config.ts +64 -12
  148. package/server/database.ts +22 -31
  149. package/server/global-config.ts +204 -0
  150. package/server/index.ts +139 -316
  151. package/server/init.ts +266 -234
  152. package/server/launch.ts +32 -0
  153. package/server/manifest-types.ts +145 -0
  154. package/server/manifest.ts +494 -0
  155. package/server/migrate-legacy.ts +290 -0
  156. package/server/parsers/event-parser.test.ts +135 -0
  157. package/server/parsers/event-parser.ts +4 -1
  158. package/server/parsers/scope-parser.test.ts +270 -0
  159. package/server/parsers/scope-parser.ts +79 -31
  160. package/server/project-context.ts +325 -0
  161. package/server/project-emitter.ts +50 -0
  162. package/server/project-manager.ts +368 -0
  163. package/server/routes/aggregate-routes.ts +968 -0
  164. package/server/routes/config-routes.ts +43 -85
  165. package/server/routes/data-routes.ts +34 -156
  166. package/server/routes/dispatch-routes.ts +46 -17
  167. package/server/routes/git-routes.ts +77 -0
  168. package/server/routes/manifest-routes.ts +388 -0
  169. package/server/routes/scope-routes.ts +39 -30
  170. package/server/routes/sync-routes.ts +175 -0
  171. package/server/routes/version-routes.ts +1 -16
  172. package/server/routes/workflow-routes.ts +9 -3
  173. package/server/schema.ts +3 -0
  174. package/server/services/batch-orchestrator.ts +41 -17
  175. package/server/services/claude-session-service.ts +16 -14
  176. package/server/services/config-service.ts +10 -1
  177. package/server/services/deploy-service.test.ts +145 -0
  178. package/server/services/deploy-service.ts +2 -2
  179. package/server/services/event-service.test.ts +242 -0
  180. package/server/services/event-service.ts +92 -3
  181. package/server/services/gate-service.test.ts +131 -0
  182. package/server/services/gate-service.ts +2 -2
  183. package/server/services/git-service.ts +137 -4
  184. package/server/services/github-service.ts +120 -2
  185. package/server/services/readiness-service.test.ts +217 -0
  186. package/server/services/scope-cache.test.ts +167 -0
  187. package/server/services/scope-cache.ts +4 -1
  188. package/server/services/scope-service.test.ts +169 -0
  189. package/server/services/scope-service.ts +224 -130
  190. package/server/services/sprint-orchestrator.ts +30 -15
  191. package/server/services/sprint-service.test.ts +271 -0
  192. package/server/services/sprint-service.ts +29 -5
  193. package/server/services/sync-service.ts +482 -0
  194. package/server/services/sync-types.ts +77 -0
  195. package/server/services/workflow-service.test.ts +190 -0
  196. package/server/services/workflow-service.ts +29 -9
  197. package/server/settings-sync.ts +359 -0
  198. package/server/uninstall.ts +214 -0
  199. package/server/update-planner.ts +346 -0
  200. package/server/update.ts +263 -0
  201. package/server/utils/cc-hooks-parser.test.ts +96 -0
  202. package/server/utils/cc-hooks-parser.ts +4 -0
  203. package/server/utils/dispatch-utils.test.ts +245 -0
  204. package/server/utils/dispatch-utils.ts +102 -30
  205. package/server/utils/flag-builder.ts +56 -0
  206. package/server/utils/json-fields.test.ts +83 -0
  207. package/server/utils/json-fields.ts +14 -0
  208. package/server/utils/logger.ts +40 -3
  209. package/server/utils/package-info.ts +32 -0
  210. package/server/utils/route-helpers.test.ts +144 -0
  211. package/server/utils/route-helpers.ts +50 -0
  212. package/server/utils/terminal-launcher.ts +85 -25
  213. package/server/utils/worktree-manager.ts +9 -4
  214. package/server/validator.ts +270 -0
  215. package/server/watchers/event-watcher.ts +24 -12
  216. package/server/watchers/global-watcher.ts +77 -0
  217. package/server/watchers/scope-watcher.ts +21 -9
  218. package/server/wizard/config-editor.ts +248 -0
  219. package/server/wizard/detect.ts +104 -0
  220. package/server/wizard/doctor.ts +114 -0
  221. package/server/wizard/index.ts +438 -0
  222. package/server/wizard/phases/confirm.ts +45 -0
  223. package/server/wizard/phases/project-setup.ts +106 -0
  224. package/server/wizard/phases/setup-wizard.ts +78 -0
  225. package/server/wizard/phases/welcome.ts +39 -0
  226. package/server/wizard/phases/workflow-setup.ts +28 -0
  227. package/server/wizard/types.ts +56 -0
  228. package/server/wizard/ui.ts +92 -0
  229. package/shared/__fixtures__/workflow-configs.ts +80 -0
  230. package/shared/api-types.ts +106 -0
  231. package/shared/onboarding-tour.test.ts +94 -0
  232. package/shared/project-colors.ts +24 -0
  233. package/shared/workflow-config.test.ts +111 -0
  234. package/shared/workflow-config.ts +7 -0
  235. package/shared/workflow-engine.test.ts +388 -0
  236. package/shared/workflow-engine.ts +1 -1
  237. package/shared/workflow-normalizer.test.ts +119 -0
  238. package/shared/workflow-normalizer.ts +118 -0
  239. package/templates/agents/QUICK-REFERENCE.md +1 -0
  240. package/templates/agents/README.md +1 -0
  241. package/templates/agents/SKILL-TRIGGERS.md +11 -0
  242. package/templates/agents/green-team/deep-dive.md +361 -0
  243. package/templates/hooks/end-session.sh +4 -1
  244. package/templates/hooks/init-session.sh +1 -0
  245. package/templates/hooks/orbital-emit.sh +2 -2
  246. package/templates/hooks/orbital-report-deploy.sh +4 -4
  247. package/templates/hooks/orbital-report-gates.sh +4 -4
  248. package/templates/hooks/orbital-scope-update.sh +1 -1
  249. package/templates/hooks/scope-commit-logger.sh +2 -2
  250. package/templates/hooks/scope-create-cleanup.sh +2 -2
  251. package/templates/hooks/scope-create-gate.sh +2 -5
  252. package/templates/hooks/scope-gate.sh +4 -6
  253. package/templates/hooks/scope-helpers.sh +28 -1
  254. package/templates/hooks/scope-lifecycle-gate.sh +14 -5
  255. package/templates/hooks/scope-prepare.sh +67 -12
  256. package/templates/hooks/scope-transition.sh +14 -6
  257. package/templates/hooks/time-tracker.sh +2 -5
  258. package/templates/migrations/renames.json +1 -0
  259. package/templates/orbital.config.json +8 -6
  260. package/{shared/default-workflow.json → templates/presets/default.json} +65 -0
  261. package/templates/presets/development.json +4 -4
  262. package/templates/presets/gitflow.json +7 -0
  263. package/templates/prompts/README.md +23 -0
  264. package/templates/prompts/deep-dive-audit.md +94 -0
  265. package/templates/quick/rules.md +56 -5
  266. package/templates/settings-hooks.json +1 -1
  267. package/templates/skills/git-commit/SKILL.md +27 -7
  268. package/templates/skills/git-dev/SKILL.md +13 -4
  269. package/templates/skills/git-main/SKILL.md +13 -3
  270. package/templates/skills/git-production/SKILL.md +9 -2
  271. package/templates/skills/git-staging/SKILL.md +11 -3
  272. package/templates/skills/scope-create/SKILL.md +17 -3
  273. package/templates/skills/scope-fix-review/SKILL.md +14 -7
  274. package/templates/skills/scope-implement/SKILL.md +15 -4
  275. package/templates/skills/scope-post-review/SKILL.md +77 -7
  276. package/templates/skills/scope-pre-review/SKILL.md +11 -4
  277. package/templates/skills/scope-verify/SKILL.md +5 -3
  278. package/templates/skills/test-code-review/SKILL.md +41 -33
  279. package/templates/skills/test-scaffold/SKILL.md +222 -0
  280. package/dist/assets/WorkflowVisualizer-BZ21PIIF.js +0 -84
  281. package/dist/assets/charts-D__PA1zp.js +0 -72
  282. package/dist/assets/index-D1G6i0nS.css +0 -1
  283. package/dist/assets/index-DpItvKpf.js +0 -419
  284. package/dist/assets/ui-BvF022GT.js +0 -53
  285. package/index.html +0 -15
  286. package/postcss.config.js +0 -6
  287. package/src/App.tsx +0 -33
  288. package/src/components/AgentBadge.tsx +0 -40
  289. package/src/components/BatchPreflightModal.tsx +0 -115
  290. package/src/components/CardDisplayToggle.tsx +0 -74
  291. package/src/components/ColumnHeaderActions.tsx +0 -55
  292. package/src/components/ColumnMenu.tsx +0 -99
  293. package/src/components/DeployHistory.tsx +0 -141
  294. package/src/components/DispatchModal.tsx +0 -164
  295. package/src/components/DispatchPopover.tsx +0 -139
  296. package/src/components/DragOverlay.tsx +0 -25
  297. package/src/components/DriftSidebar.tsx +0 -140
  298. package/src/components/EnvironmentStrip.tsx +0 -88
  299. package/src/components/ErrorBoundary.tsx +0 -62
  300. package/src/components/FilterChip.tsx +0 -105
  301. package/src/components/GateIndicator.tsx +0 -33
  302. package/src/components/IdeaDetailModal.tsx +0 -190
  303. package/src/components/IdeaFormDialog.tsx +0 -113
  304. package/src/components/KanbanColumn.tsx +0 -201
  305. package/src/components/MarkdownRenderer.tsx +0 -114
  306. package/src/components/NeonGrid.tsx +0 -128
  307. package/src/components/PromotionQueue.tsx +0 -89
  308. package/src/components/ScopeCard.tsx +0 -234
  309. package/src/components/ScopeDetailModal.tsx +0 -255
  310. package/src/components/ScopeFilterBar.tsx +0 -152
  311. package/src/components/SearchInput.tsx +0 -102
  312. package/src/components/SessionPanel.tsx +0 -335
  313. package/src/components/SprintContainer.tsx +0 -303
  314. package/src/components/SprintDependencyDialog.tsx +0 -78
  315. package/src/components/SprintPreflightModal.tsx +0 -138
  316. package/src/components/StatusBar.tsx +0 -168
  317. package/src/components/SwimCell.tsx +0 -67
  318. package/src/components/SwimLaneRow.tsx +0 -94
  319. package/src/components/SwimlaneBoardView.tsx +0 -108
  320. package/src/components/VersionBadge.tsx +0 -139
  321. package/src/components/ViewModeSelector.tsx +0 -114
  322. package/src/components/config/AgentChip.tsx +0 -53
  323. package/src/components/config/AgentCreateDialog.tsx +0 -321
  324. package/src/components/config/AgentEditor.tsx +0 -175
  325. package/src/components/config/DirectoryTree.tsx +0 -582
  326. package/src/components/config/FileEditor.tsx +0 -550
  327. package/src/components/config/HookChip.tsx +0 -50
  328. package/src/components/config/StageCard.tsx +0 -198
  329. package/src/components/config/TransitionZone.tsx +0 -173
  330. package/src/components/config/UnifiedWorkflowPipeline.tsx +0 -216
  331. package/src/components/config/WorkflowPipeline.tsx +0 -161
  332. package/src/components/source-control/BranchList.tsx +0 -93
  333. package/src/components/source-control/BranchPanel.tsx +0 -105
  334. package/src/components/source-control/CommitLog.tsx +0 -100
  335. package/src/components/source-control/CommitRow.tsx +0 -47
  336. package/src/components/source-control/GitHubPanel.tsx +0 -110
  337. package/src/components/source-control/GitHubSetupGuide.tsx +0 -52
  338. package/src/components/source-control/GitOverviewBar.tsx +0 -101
  339. package/src/components/source-control/PullRequestList.tsx +0 -69
  340. package/src/components/source-control/WorktreeList.tsx +0 -80
  341. package/src/components/ui/badge.tsx +0 -41
  342. package/src/components/ui/button.tsx +0 -55
  343. package/src/components/ui/card.tsx +0 -78
  344. package/src/components/ui/dialog.tsx +0 -94
  345. package/src/components/ui/popover.tsx +0 -33
  346. package/src/components/ui/scroll-area.tsx +0 -54
  347. package/src/components/ui/separator.tsx +0 -28
  348. package/src/components/ui/tabs.tsx +0 -52
  349. package/src/components/ui/toggle-switch.tsx +0 -35
  350. package/src/components/ui/tooltip.tsx +0 -27
  351. package/src/components/workflow/AddEdgeDialog.tsx +0 -217
  352. package/src/components/workflow/AddListDialog.tsx +0 -201
  353. package/src/components/workflow/ChecklistEditor.tsx +0 -239
  354. package/src/components/workflow/CommandPrefixManager.tsx +0 -118
  355. package/src/components/workflow/ConfigSettingsPanel.tsx +0 -189
  356. package/src/components/workflow/DirectionSelector.tsx +0 -133
  357. package/src/components/workflow/DispatchConfigPanel.tsx +0 -180
  358. package/src/components/workflow/EdgeDetailPanel.tsx +0 -236
  359. package/src/components/workflow/EdgePropertyEditor.tsx +0 -251
  360. package/src/components/workflow/EditToolbar.tsx +0 -138
  361. package/src/components/workflow/HookDetailPanel.tsx +0 -250
  362. package/src/components/workflow/HookExecutionLog.tsx +0 -24
  363. package/src/components/workflow/HookSourceModal.tsx +0 -129
  364. package/src/components/workflow/HooksDashboard.tsx +0 -363
  365. package/src/components/workflow/ListPropertyEditor.tsx +0 -251
  366. package/src/components/workflow/MigrationPreviewDialog.tsx +0 -237
  367. package/src/components/workflow/MovementRulesPanel.tsx +0 -188
  368. package/src/components/workflow/NodeDetailPanel.tsx +0 -245
  369. package/src/components/workflow/PresetSelector.tsx +0 -414
  370. package/src/components/workflow/SkillCommandBuilder.tsx +0 -174
  371. package/src/components/workflow/WorkflowEdgeComponent.tsx +0 -145
  372. package/src/components/workflow/WorkflowNode.tsx +0 -147
  373. package/src/components/workflow/graphLayout.ts +0 -186
  374. package/src/components/workflow/mergeHooks.ts +0 -85
  375. package/src/components/workflow/useEditHistory.ts +0 -88
  376. package/src/components/workflow/useWorkflowEditor.ts +0 -262
  377. package/src/components/workflow/validateConfig.ts +0 -70
  378. package/src/hooks/useActiveDispatches.ts +0 -198
  379. package/src/hooks/useBoardSettings.ts +0 -170
  380. package/src/hooks/useCardDisplay.ts +0 -57
  381. package/src/hooks/useCcHooks.ts +0 -24
  382. package/src/hooks/useConfigTree.ts +0 -51
  383. package/src/hooks/useEnforcementRules.ts +0 -46
  384. package/src/hooks/useEvents.ts +0 -59
  385. package/src/hooks/useFileEditor.ts +0 -165
  386. package/src/hooks/useGates.ts +0 -57
  387. package/src/hooks/useIdeaActions.ts +0 -53
  388. package/src/hooks/useKanbanDnd.ts +0 -410
  389. package/src/hooks/useOrbitalConfig.ts +0 -54
  390. package/src/hooks/usePipeline.ts +0 -47
  391. package/src/hooks/usePipelineData.ts +0 -338
  392. package/src/hooks/useReconnect.ts +0 -25
  393. package/src/hooks/useScopeFilters.ts +0 -125
  394. package/src/hooks/useScopeSessions.ts +0 -44
  395. package/src/hooks/useScopes.ts +0 -67
  396. package/src/hooks/useSearch.ts +0 -67
  397. package/src/hooks/useSettings.tsx +0 -187
  398. package/src/hooks/useSocket.ts +0 -25
  399. package/src/hooks/useSourceControl.ts +0 -105
  400. package/src/hooks/useSprintPreflight.ts +0 -55
  401. package/src/hooks/useSprints.ts +0 -154
  402. package/src/hooks/useStatusBarHighlight.ts +0 -18
  403. package/src/hooks/useSwimlaneBoardSettings.ts +0 -104
  404. package/src/hooks/useTheme.ts +0 -9
  405. package/src/hooks/useTransitionReadiness.ts +0 -53
  406. package/src/hooks/useVersion.ts +0 -155
  407. package/src/hooks/useViolations.ts +0 -65
  408. package/src/hooks/useWorkflow.tsx +0 -125
  409. package/src/hooks/useZoomModifier.ts +0 -19
  410. package/src/index.css +0 -797
  411. package/src/layouts/DashboardLayout.tsx +0 -113
  412. package/src/lib/collisionDetection.ts +0 -20
  413. package/src/lib/scope-fields.ts +0 -61
  414. package/src/lib/swimlane.ts +0 -146
  415. package/src/lib/utils.ts +0 -15
  416. package/src/main.tsx +0 -19
  417. package/src/socket.ts +0 -11
  418. package/src/types/index.ts +0 -497
  419. package/src/views/AgentFeed.tsx +0 -339
  420. package/src/views/DeployPipeline.tsx +0 -59
  421. package/src/views/EnforcementView.tsx +0 -378
  422. package/src/views/PrimitivesConfig.tsx +0 -500
  423. package/src/views/QualityGates.tsx +0 -1012
  424. package/src/views/ScopeBoard.tsx +0 -454
  425. package/src/views/SessionTimeline.tsx +0 -516
  426. package/src/views/Settings.tsx +0 -183
  427. package/src/views/SourceControl.tsx +0 -95
  428. package/src/views/WorkflowVisualizer.tsx +0 -382
  429. package/tailwind.config.js +0 -161
  430. package/tsconfig.json +0 -25
  431. package/vite.config.ts +0 -38
@@ -1,262 +0,0 @@
1
- import { useState, useCallback, useMemo } from 'react';
2
- import type { WorkflowConfig, WorkflowList, WorkflowEdge } from '../../../shared/workflow-config';
3
- import { useEditHistory } from './useEditHistory';
4
- import { validateConfig } from './validateConfig';
5
- import type { ConfigValidationResult } from './validateConfig';
6
-
7
- // ─── Types ──────────────────────────────────────────────
8
-
9
- interface MigrationPlan {
10
- valid: boolean;
11
- validationErrors: string[];
12
- removedLists: string[];
13
- addedLists: string[];
14
- dirsToCreate: string[];
15
- dirsToRemove: string[];
16
- orphanedScopes: Array<{ listId: string; scopeFiles: string[] }>;
17
- lostEdges: Array<{ from: string; to: string }>;
18
- suggestedMappings: Record<string, string>;
19
- impactSummary: string;
20
- }
21
-
22
- interface WorkflowEditorState {
23
- editMode: boolean;
24
- editConfig: WorkflowConfig;
25
- canUndo: boolean;
26
- canRedo: boolean;
27
- changeCount: number;
28
- validation: ConfigValidationResult;
29
- saving: boolean;
30
- previewPlan: MigrationPlan | null;
31
- previewLoading: boolean;
32
- previewError: string | null;
33
- showPreview: boolean;
34
- showAddList: boolean;
35
- showAddEdge: boolean;
36
- showConfigSettings: boolean;
37
- }
38
-
39
- interface WorkflowEditorActions {
40
- enterEditMode: () => void;
41
- exitEditMode: () => void;
42
- undo: () => void;
43
- redo: () => void;
44
- updateList: (original: WorkflowList, updated: WorkflowList) => void;
45
- deleteList: (listId: string) => void;
46
- addList: (list: WorkflowList) => void;
47
- updateEdge: (original: WorkflowEdge, updated: WorkflowEdge) => void;
48
- deleteEdge: (from: string, to: string) => void;
49
- addEdge: (edge: WorkflowEdge) => void;
50
- updateConfig: (config: WorkflowConfig) => void;
51
- save: () => Promise<void>;
52
- discard: () => void;
53
- preview: () => Promise<void>;
54
- applyMigration: (orphanMappings: Record<string, string>) => Promise<void>;
55
- setShowPreview: (show: boolean) => void;
56
- setShowAddList: (show: boolean) => void;
57
- setShowAddEdge: (show: boolean) => void;
58
- setShowConfigSettings: (show: boolean) => void;
59
- }
60
-
61
- export type WorkflowEditor = WorkflowEditorState & WorkflowEditorActions;
62
-
63
- // ─── Hook ───────────────────────────────────────────────
64
-
65
- export function useWorkflowEditor(activeConfig: WorkflowConfig): WorkflowEditor {
66
- const [editMode, setEditMode] = useState(false);
67
- const [saving, setSaving] = useState(false);
68
- const [previewPlan, setPreviewPlan] = useState<MigrationPlan | null>(null);
69
- const [previewLoading, setPreviewLoading] = useState(false);
70
- const [previewError, setPreviewError] = useState<string | null>(null);
71
- const [showPreview, setShowPreview] = useState(false);
72
- const [showAddList, setShowAddList] = useState(false);
73
- const [showAddEdge, setShowAddEdge] = useState(false);
74
- const [showConfigSettings, setShowConfigSettings] = useState(false);
75
-
76
- const history = useEditHistory(activeConfig);
77
-
78
- const validation = useMemo(
79
- () => validateConfig(history.present),
80
- [history.present],
81
- );
82
-
83
- // ─── Mode Toggle ────────────────────────────────────
84
-
85
- const enterEditMode = useCallback(() => {
86
- history.reset(activeConfig);
87
- setEditMode(true);
88
- }, [activeConfig, history]);
89
-
90
- const exitEditMode = useCallback(() => {
91
- setEditMode(false);
92
- setShowPreview(false);
93
- setShowAddList(false);
94
- setShowAddEdge(false);
95
- setShowConfigSettings(false);
96
- }, []);
97
-
98
- const discard = useCallback(() => {
99
- history.reset(activeConfig);
100
- exitEditMode();
101
- }, [activeConfig, history, exitEditMode]);
102
-
103
- // ─── List Operations ────────────────────────────────
104
-
105
- const addList = useCallback((list: WorkflowList) => {
106
- const config = structuredClone(history.present);
107
- config.lists.push(list);
108
- history.pushState(config);
109
- }, [history]);
110
-
111
- const updateList = useCallback((original: WorkflowList, updated: WorkflowList) => {
112
- const config = structuredClone(history.present);
113
- const idx = config.lists.findIndex((l) => l.id === original.id);
114
- if (idx === -1) return;
115
- // If ID changed, update all edge references
116
- if (original.id !== updated.id) {
117
- for (const edge of config.edges) {
118
- if (edge.from === original.id) edge.from = updated.id;
119
- if (edge.to === original.id) edge.to = updated.id;
120
- }
121
- }
122
- config.lists[idx] = updated;
123
- history.pushState(config);
124
- }, [history]);
125
-
126
- const deleteList = useCallback((listId: string) => {
127
- const config = structuredClone(history.present);
128
- config.lists = config.lists.filter((l) => l.id !== listId);
129
- config.edges = config.edges.filter((e) => e.from !== listId && e.to !== listId);
130
- history.pushState(config);
131
- }, [history]);
132
-
133
- // ─── Edge Operations ────────────────────────────────
134
-
135
- const addEdge = useCallback((edge: WorkflowEdge) => {
136
- const config = structuredClone(history.present);
137
- config.edges.push(edge);
138
- history.pushState(config);
139
- }, [history]);
140
-
141
- const updateConfig = useCallback((updated: WorkflowConfig) => {
142
- history.pushState(structuredClone(updated));
143
- }, [history]);
144
-
145
- const updateEdge = useCallback((original: WorkflowEdge, updated: WorkflowEdge) => {
146
- const config = structuredClone(history.present);
147
- const key = `${original.from}:${original.to}`;
148
- const idx = config.edges.findIndex((e) => `${e.from}:${e.to}` === key);
149
- if (idx === -1) return;
150
- config.edges[idx] = updated;
151
- history.pushState(config);
152
- }, [history]);
153
-
154
- const deleteEdge = useCallback((from: string, to: string) => {
155
- const config = structuredClone(history.present);
156
- config.edges = config.edges.filter((e) => !(e.from === from && e.to === to));
157
- history.pushState(config);
158
- }, [history]);
159
-
160
- // ─── Save ───────────────────────────────────────────
161
-
162
- const save = useCallback(async () => {
163
- if (!validation.valid || saving) return;
164
- setSaving(true);
165
- try {
166
- const res = await fetch('/api/orbital/workflow', {
167
- method: 'PUT',
168
- headers: { 'Content-Type': 'application/json' },
169
- body: JSON.stringify(history.present),
170
- });
171
- const json: { success: boolean; error?: string } = await res.json();
172
- if (!json.success) throw new Error(json.error ?? 'Save failed');
173
- exitEditMode();
174
- } catch (err) {
175
- setPreviewError(err instanceof Error ? err.message : 'Save failed');
176
- } finally {
177
- setSaving(false);
178
- }
179
- }, [validation.valid, saving, history.present, exitEditMode]);
180
-
181
- // ─── Preview ────────────────────────────────────────
182
-
183
- const preview = useCallback(async () => {
184
- setPreviewLoading(true);
185
- setPreviewError(null);
186
- setPreviewPlan(null);
187
- setShowPreview(true);
188
- try {
189
- const res = await fetch('/api/orbital/workflow/preview', {
190
- method: 'POST',
191
- headers: { 'Content-Type': 'application/json' },
192
- body: JSON.stringify(history.present),
193
- });
194
- const json: { success: boolean; data?: MigrationPlan; error?: string } = await res.json();
195
- if (!json.success) throw new Error(json.error ?? 'Preview failed');
196
- setPreviewPlan(json.data ?? null);
197
- } catch (err) {
198
- setPreviewError(err instanceof Error ? err.message : 'Preview failed');
199
- } finally {
200
- setPreviewLoading(false);
201
- }
202
- }, [history.present]);
203
-
204
- // ─── Apply Migration ────────────────────────────────
205
-
206
- const applyMigration = useCallback(async (orphanMappings: Record<string, string>) => {
207
- setSaving(true);
208
- try {
209
- const res = await fetch('/api/orbital/workflow/apply', {
210
- method: 'POST',
211
- headers: { 'Content-Type': 'application/json' },
212
- body: JSON.stringify({ config: history.present, orphanMappings }),
213
- });
214
- const json: { success: boolean; error?: string } = await res.json();
215
- if (!json.success) throw new Error(json.error ?? 'Migration failed');
216
- setShowPreview(false);
217
- exitEditMode();
218
- } catch (err) {
219
- setPreviewError(err instanceof Error ? err.message : 'Migration failed');
220
- } finally {
221
- setSaving(false);
222
- }
223
- }, [history.present, exitEditMode]);
224
-
225
- return {
226
- // State
227
- editMode,
228
- editConfig: history.present,
229
- canUndo: history.canUndo,
230
- canRedo: history.canRedo,
231
- changeCount: history.changeCount,
232
- validation,
233
- saving,
234
- previewPlan,
235
- previewLoading,
236
- previewError,
237
- showPreview,
238
- showAddList,
239
- showAddEdge,
240
- showConfigSettings,
241
- // Actions
242
- enterEditMode,
243
- exitEditMode,
244
- undo: history.undo,
245
- redo: history.redo,
246
- updateList,
247
- deleteList,
248
- addList,
249
- updateEdge,
250
- deleteEdge,
251
- addEdge,
252
- updateConfig,
253
- save,
254
- discard,
255
- preview,
256
- applyMigration,
257
- setShowPreview,
258
- setShowAddList,
259
- setShowAddEdge,
260
- setShowConfigSettings,
261
- };
262
- }
@@ -1,70 +0,0 @@
1
- import type { WorkflowConfig } from '../../../shared/workflow-config';
2
- import { isWorkflowConfig } from '../../../shared/workflow-config';
3
-
4
- // ─── Types ──────────────────────────────────────────────
5
-
6
- export interface ConfigValidationResult {
7
- valid: boolean;
8
- errors: string[];
9
- }
10
-
11
- // ─── Validation ─────────────────────────────────────────
12
- // Mirrors server-side validation in workflow-service.ts
13
-
14
- export function validateConfig(config: WorkflowConfig): ConfigValidationResult {
15
- const errors: string[] = [];
16
-
17
- if (!isWorkflowConfig(config)) {
18
- errors.push('Invalid config shape: must have version=1, name, lists[], edges[]');
19
- return { valid: false, errors };
20
- }
21
-
22
- if (config.branchingMode !== undefined && config.branchingMode !== 'trunk' && config.branchingMode !== 'worktree') {
23
- errors.push(`Invalid branchingMode: "${config.branchingMode}" (must be "trunk" or "worktree")`);
24
- }
25
-
26
- // Unique list IDs
27
- const listIds = new Set<string>();
28
- for (const list of config.lists) {
29
- if (listIds.has(list.id)) errors.push(`Duplicate list ID: "${list.id}"`);
30
- listIds.add(list.id);
31
- }
32
-
33
- // Valid edge references + no duplicates
34
- const edgeKeys = new Set<string>();
35
- for (const edge of config.edges) {
36
- if (!listIds.has(edge.from)) errors.push(`Edge references unknown list: from="${edge.from}"`);
37
- if (!listIds.has(edge.to)) errors.push(`Edge references unknown list: to="${edge.to}"`);
38
- if (edge.from === edge.to) errors.push(`Self-referencing edge: "${edge.from}" → "${edge.to}"`);
39
- const key = `${edge.from}:${edge.to}`;
40
- if (edgeKeys.has(key)) errors.push(`Duplicate edge: ${key}`);
41
- edgeKeys.add(key);
42
- }
43
-
44
- // Exactly 1 entry point
45
- const entryPoints = config.lists.filter((l) => l.isEntryPoint);
46
- if (entryPoints.length === 0) errors.push('No entry point defined (isEntryPoint=true)');
47
- if (entryPoints.length > 1) errors.push(`Multiple entry points: ${entryPoints.map((l) => l.id).join(', ')}`);
48
-
49
- // Graph connectivity — all non-terminal lists reachable from entry point
50
- if (entryPoints.length === 1 && errors.length === 0) {
51
- const terminal = new Set(config.terminalStatuses ?? []);
52
- const reachable = new Set<string>();
53
- const queue = [entryPoints[0].id];
54
- while (queue.length > 0) {
55
- const current = queue.shift()!;
56
- if (reachable.has(current)) continue;
57
- reachable.add(current);
58
- for (const edge of config.edges) {
59
- if (edge.from === current && !reachable.has(edge.to)) queue.push(edge.to);
60
- }
61
- }
62
- for (const list of config.lists) {
63
- if (!terminal.has(list.id) && !reachable.has(list.id)) {
64
- errors.push(`List "${list.id}" is not reachable from entry point`);
65
- }
66
- }
67
- }
68
-
69
- return { valid: errors.length === 0, errors };
70
- }
@@ -1,198 +0,0 @@
1
- import { createContext, useContext, useState, useEffect, useCallback, useRef, useMemo } from 'react';
2
- import { socket } from '../socket';
3
- import type { OrbitalEvent, Scope, DispatchResolvedPayload } from '@/types';
4
- import { useWorkflow } from './useWorkflow';
5
-
6
- export interface AbandonedInfo {
7
- from_status: string | null;
8
- abandoned_at: string;
9
- }
10
-
11
- interface ActiveDispatchContextValue {
12
- activeScopes: Set<number>;
13
- abandonedScopes: Map<number, AbandonedInfo>;
14
- recoverScope: (scopeId: number, fromStatus: string) => Promise<void>;
15
- dismissAbandoned: (scopeId: number) => Promise<void>;
16
- }
17
-
18
- const DEFAULT_VALUE: ActiveDispatchContextValue = {
19
- activeScopes: new Set(),
20
- abandonedScopes: new Map(),
21
- recoverScope: async () => {},
22
- dismissAbandoned: async () => {},
23
- };
24
-
25
- export const ActiveDispatchContext = createContext<ActiveDispatchContextValue>(DEFAULT_VALUE);
26
-
27
- /** Provider hook — call once at ScopeBoard level.
28
- * Fetches initial set from REST, then maintains via socket events. */
29
- export function useActiveDispatchProvider(): ActiveDispatchContextValue {
30
- const { engine } = useWorkflow();
31
- const terminalStatuses = useMemo(
32
- () => new Set(engine.getConfig().terminalStatuses ?? []),
33
- [engine],
34
- );
35
- const [activeScopes, setActiveScopes] = useState<Set<number>>(new Set());
36
- const [abandonedScopes, setAbandonedScopes] = useState<Map<number, AbandonedInfo>>(new Map());
37
- const mountedRef = useRef(true);
38
-
39
- const removeFromAbandoned = useCallback((scopeId: number) => {
40
- setAbandonedScopes((prev) => {
41
- if (!prev.has(scopeId)) return prev;
42
- const next = new Map(prev);
43
- next.delete(scopeId);
44
- return next;
45
- });
46
- }, []);
47
-
48
- const fetchActiveScopes = useCallback(async () => {
49
- try {
50
- const res = await fetch('/api/orbital/dispatch/active-scopes');
51
- if (!res.ok) {
52
- console.warn('[Orbital] Failed to fetch active scopes:', res.status, res.statusText);
53
- return;
54
- }
55
- const data = await res.json() as {
56
- scope_ids: number[];
57
- abandoned_scopes?: Array<{ scope_id: number; from_status: string | null; abandoned_at: string }>;
58
- };
59
- if (!mountedRef.current) return;
60
- setActiveScopes(new Set(data.scope_ids));
61
-
62
- if (data.abandoned_scopes) {
63
- const map = new Map<number, AbandonedInfo>();
64
- for (const item of data.abandoned_scopes) {
65
- map.set(item.scope_id, { from_status: item.from_status, abandoned_at: item.abandoned_at });
66
- }
67
- setAbandonedScopes(map);
68
- }
69
- } catch {
70
- // Silently ignore — will retry on next reconnect
71
- }
72
- }, []);
73
-
74
- const recoverScope = useCallback(async (scopeId: number, fromStatus: string) => {
75
- try {
76
- const res = await fetch(`/api/orbital/dispatch/recover/${scopeId}`, {
77
- method: 'POST',
78
- headers: { 'Content-Type': 'application/json' },
79
- body: JSON.stringify({ from_status: fromStatus }),
80
- });
81
- if (!res.ok) {
82
- const body = await res.json().catch(() => ({ error: res.statusText }));
83
- console.error('[Orbital] Failed to recover scope:', body.error);
84
- return;
85
- }
86
- removeFromAbandoned(scopeId);
87
- } catch (err) {
88
- console.error('[Orbital] Failed to recover scope:', err);
89
- }
90
- }, [removeFromAbandoned]);
91
-
92
- const dismissAbandoned = useCallback(async (scopeId: number) => {
93
- try {
94
- const res = await fetch(`/api/orbital/dispatch/dismiss-abandoned/${scopeId}`, {
95
- method: 'POST',
96
- headers: { 'Content-Type': 'application/json' },
97
- });
98
- if (!res.ok) {
99
- const body = await res.json().catch(() => ({ error: res.statusText }));
100
- console.error('[Orbital] Failed to dismiss abandoned scope:', body.error);
101
- return;
102
- }
103
- removeFromAbandoned(scopeId);
104
- } catch (err) {
105
- console.error('[Orbital] Failed to dismiss abandoned scope:', err);
106
- }
107
- }, [removeFromAbandoned]);
108
-
109
- useEffect(() => {
110
- mountedRef.current = true;
111
- fetchActiveScopes();
112
- return () => { mountedRef.current = false; };
113
- }, [fetchActiveScopes]);
114
-
115
- useEffect(() => {
116
- function onNewEvent(event: OrbitalEvent) {
117
- if (event.type !== 'DISPATCH' || event.data.resolved != null) return;
118
-
119
- // Collect scope IDs: single dispatch uses event.scope_id, batch uses data.scope_ids
120
- const ids: number[] = [];
121
- if (event.scope_id != null) ids.push(event.scope_id);
122
- if (Array.isArray(event.data.scope_ids)) {
123
- for (const id of event.data.scope_ids as number[]) {
124
- if (!ids.includes(id)) ids.push(id);
125
- }
126
- }
127
- if (ids.length === 0) return;
128
-
129
- setActiveScopes((prev) => {
130
- const toAdd = ids.filter(id => !prev.has(id));
131
- if (toAdd.length === 0) return prev;
132
- const next = new Set(prev);
133
- for (const id of toAdd) next.add(id);
134
- return next;
135
- });
136
- for (const id of ids) removeFromAbandoned(id);
137
- }
138
-
139
- function onDispatchResolved(payload: DispatchResolvedPayload) {
140
- // Collect all scope IDs: single dispatch + batch scope_ids
141
- const ids: number[] = [];
142
- if (payload.scope_id != null) ids.push(payload.scope_id);
143
- if (Array.isArray(payload.scope_ids)) ids.push(...payload.scope_ids);
144
- if (ids.length === 0) return;
145
-
146
- setActiveScopes((prev) => {
147
- const toRemove = ids.filter(id => prev.has(id));
148
- if (toRemove.length === 0) return prev;
149
- const next = new Set(prev);
150
- for (const id of toRemove) next.delete(id);
151
- return next;
152
- });
153
-
154
- if (payload.outcome === 'abandoned') {
155
- fetchActiveScopes();
156
- } else {
157
- for (const id of ids) removeFromAbandoned(id);
158
- }
159
- }
160
-
161
- function onScopeUpdated(scope: Scope) {
162
- if (terminalStatuses.has(scope.status)) {
163
- const scopeId = scope.id;
164
- setActiveScopes((prev) => {
165
- if (!prev.has(scopeId)) return prev;
166
- const next = new Set(prev);
167
- next.delete(scopeId);
168
- return next;
169
- });
170
- // Terminal state clears abandoned
171
- removeFromAbandoned(scopeId);
172
- }
173
- }
174
-
175
- function onReconnect() {
176
- fetchActiveScopes();
177
- }
178
-
179
- socket.on('event:new', onNewEvent);
180
- socket.on('dispatch:resolved', onDispatchResolved);
181
- socket.on('scope:updated', onScopeUpdated);
182
- socket.on('connect', onReconnect);
183
-
184
- return () => {
185
- socket.off('event:new', onNewEvent);
186
- socket.off('dispatch:resolved', onDispatchResolved);
187
- socket.off('scope:updated', onScopeUpdated);
188
- socket.off('connect', onReconnect);
189
- };
190
- }, [fetchActiveScopes, removeFromAbandoned, terminalStatuses]);
191
-
192
- return { activeScopes, abandonedScopes, recoverScope, dismissAbandoned };
193
- }
194
-
195
- /** Consumer hook — use in ScopeCard to check dispatch state */
196
- export function useActiveDispatches(): ActiveDispatchContextValue {
197
- return useContext(ActiveDispatchContext);
198
- }
@@ -1,170 +0,0 @@
1
- import { useState, useCallback, useEffect } from 'react';
2
- import type { Scope } from '@/types';
3
- import { EFFORT_BUCKETS } from '@/types';
4
-
5
- // ─── Types ─────────────────────────────────────────────────
6
- export type SortField = 'id' | 'priority' | 'effort' | 'updated_at' | 'created_at' | 'title';
7
- export type SortDirection = 'asc' | 'desc';
8
-
9
- export interface BoardSettings {
10
- sortField: SortField;
11
- sortDirection: SortDirection;
12
- collapsed: Set<string>;
13
- }
14
-
15
- // ─── Constants ─────────────────────────────────────────────
16
- const SORT_KEY = 'cc-board-sort';
17
- const COLLAPSE_KEY = 'cc-board-collapsed';
18
-
19
- const DEFAULT_SORT_DIRECTIONS: Record<SortField, SortDirection> = {
20
- id: 'asc',
21
- priority: 'asc',
22
- effort: 'asc',
23
- updated_at: 'desc',
24
- created_at: 'desc',
25
- title: 'asc',
26
- };
27
-
28
- const PRIORITY_ORDER: Record<string, number> = {
29
- critical: 0,
30
- high: 1,
31
- medium: 2,
32
- low: 3,
33
- };
34
-
35
- function effortRank(raw: string | null): number {
36
- if (!raw) return Infinity;
37
- const idx = EFFORT_BUCKETS.indexOf(raw as typeof EFFORT_BUCKETS[number]);
38
- return idx >= 0 ? idx : Infinity;
39
- }
40
-
41
- // ─── localStorage helpers ──────────────────────────────────
42
- function readSort(): { field: SortField; direction: SortDirection } {
43
- try {
44
- const raw = localStorage.getItem(SORT_KEY);
45
- if (raw) {
46
- const parsed = JSON.parse(raw) as { field: string; direction: string };
47
- if (parsed.field in DEFAULT_SORT_DIRECTIONS) {
48
- return {
49
- field: parsed.field as SortField,
50
- direction: parsed.direction === 'desc' ? 'desc' : 'asc',
51
- };
52
- }
53
- }
54
- } catch { /* use defaults */ }
55
- return { field: 'id', direction: 'asc' };
56
- }
57
-
58
- function readCollapsed(): Set<string> {
59
- try {
60
- const raw = localStorage.getItem(COLLAPSE_KEY);
61
- if (raw) {
62
- const arr = JSON.parse(raw) as string[];
63
- if (Array.isArray(arr)) return new Set(arr);
64
- }
65
- } catch { /* use defaults */ }
66
- return new Set();
67
- }
68
-
69
- function persistSort(field: SortField, direction: SortDirection) {
70
- try { localStorage.setItem(SORT_KEY, JSON.stringify({ field, direction })); } catch { /* noop */ }
71
- }
72
-
73
- function persistCollapsed(collapsed: Set<string>) {
74
- try { localStorage.setItem(COLLAPSE_KEY, JSON.stringify([...collapsed])); } catch { /* noop */ }
75
- }
76
-
77
- // ─── Sort comparator ───────────────────────────────────────
78
- export function sortScopes(scopes: Scope[], field: SortField, direction: SortDirection): Scope[] {
79
- const sorted = [...scopes].sort((a, b) => {
80
- const cmp = compareByField(a, b, field);
81
- return direction === 'desc' ? -cmp : cmp;
82
- });
83
- return sorted;
84
- }
85
-
86
- function compareByField(a: Scope, b: Scope, field: SortField): number {
87
- switch (field) {
88
- case 'id':
89
- return a.id - b.id;
90
-
91
- case 'priority': {
92
- const pa = a.priority ? (PRIORITY_ORDER[a.priority] ?? Infinity) : Infinity;
93
- const pb = b.priority ? (PRIORITY_ORDER[b.priority] ?? Infinity) : Infinity;
94
- return pa - pb;
95
- }
96
-
97
- case 'effort':
98
- return effortRank(a.effort_estimate) - effortRank(b.effort_estimate);
99
-
100
- case 'updated_at': {
101
- const ua = a.updated_at ? new Date(a.updated_at).getTime() : 0;
102
- const ub = b.updated_at ? new Date(b.updated_at).getTime() : 0;
103
- return ua - ub;
104
- }
105
-
106
- case 'created_at': {
107
- const ca = a.created_at ? new Date(a.created_at).getTime() : 0;
108
- const cb = b.created_at ? new Date(b.created_at).getTime() : 0;
109
- return ca - cb;
110
- }
111
-
112
- case 'title':
113
- return a.title.localeCompare(b.title);
114
-
115
- default:
116
- return 0;
117
- }
118
- }
119
-
120
- // ─── Hook ──────────────────────────────────────────────────
121
- export function useBoardSettings() {
122
- const [sortField, setSortField] = useState<SortField>(() => readSort().field);
123
- const [sortDirection, setSortDirection] = useState<SortDirection>(() => readSort().direction);
124
- const [collapsed, setCollapsed] = useState<Set<string>>(readCollapsed);
125
-
126
- // Cross-tab sync
127
- useEffect(() => {
128
- function onStorage(e: StorageEvent) {
129
- if (e.key === SORT_KEY) {
130
- const s = readSort();
131
- setSortField(s.field);
132
- setSortDirection(s.direction);
133
- }
134
- if (e.key === COLLAPSE_KEY) {
135
- setCollapsed(readCollapsed());
136
- }
137
- }
138
- window.addEventListener('storage', onStorage);
139
- return () => window.removeEventListener('storage', onStorage);
140
- }, []);
141
-
142
- const setSort = useCallback((field: SortField) => {
143
- setSortField((prevField) => {
144
- setSortDirection((prevDir) => {
145
- // Same field → toggle direction; different field → default direction
146
- const nextDir = prevField === field
147
- ? (prevDir === 'asc' ? 'desc' : 'asc')
148
- : DEFAULT_SORT_DIRECTIONS[field];
149
- persistSort(field, nextDir);
150
- return nextDir;
151
- });
152
- return field;
153
- });
154
- }, []);
155
-
156
- const toggleCollapse = useCallback((columnId: string) => {
157
- setCollapsed((prev) => {
158
- const next = new Set(prev);
159
- if (next.has(columnId)) {
160
- next.delete(columnId);
161
- } else {
162
- next.add(columnId);
163
- }
164
- persistCollapsed(next);
165
- return next;
166
- });
167
- }, []);
168
-
169
- return { sortField, sortDirection, setSort, collapsed, toggleCollapse } as const;
170
- }