orbital-command 0.1.4 → 0.3.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 (380) hide show
  1. package/bin/orbital.js +676 -53
  2. package/dist/assets/PrimitivesConfig-CrmQXYh4.js +32 -0
  3. package/dist/assets/QualityGates-BbasOsF3.js +21 -0
  4. package/dist/assets/SessionTimeline-CGeJsVvy.js +1 -0
  5. package/dist/assets/Settings-oiM496mc.js +12 -0
  6. package/dist/assets/SourceControl-B1fP2nJL.js +41 -0
  7. package/dist/assets/WorkflowVisualizer-CWLYf-f0.js +74 -0
  8. package/dist/assets/arrow-down-CPy85_J6.js +6 -0
  9. package/dist/assets/charts-DbDg0Psc.js +68 -0
  10. package/dist/assets/circle-x-Cwz6ZQDV.js +6 -0
  11. package/dist/assets/file-text-C46Xr65c.js +6 -0
  12. package/dist/assets/formatDistanceToNow-BMqsSP44.js +1 -0
  13. package/dist/assets/globe-Cn2yNZUD.js +6 -0
  14. package/dist/assets/index-Aj4sV8Al.css +1 -0
  15. package/dist/assets/index-Bc9dK3MW.js +354 -0
  16. package/dist/assets/key-OPaNTWJ5.js +6 -0
  17. package/dist/assets/minus-GMsbpKym.js +6 -0
  18. package/dist/assets/shield-DwAFkDYI.js +6 -0
  19. package/dist/assets/ui-BmsSg9jU.js +53 -0
  20. package/dist/assets/useWorkflowEditor-BJkTX_NR.js +16 -0
  21. package/dist/assets/{vendor-Dzv9lrRc.js → vendor-Bqt8AJn2.js} +1 -1
  22. package/dist/assets/zap-DfbUoOty.js +11 -0
  23. package/dist/favicon.svg +1 -0
  24. package/dist/index.html +6 -5
  25. package/dist/server/server/__tests__/data-routes.test.js +124 -0
  26. package/dist/server/server/__tests__/helpers/db.js +17 -0
  27. package/dist/server/server/__tests__/helpers/mock-emitter.js +8 -0
  28. package/dist/server/server/__tests__/scope-routes.test.js +137 -0
  29. package/dist/server/server/__tests__/sprint-routes.test.js +102 -0
  30. package/dist/server/server/__tests__/workflow-routes.test.js +107 -0
  31. package/dist/server/server/config-migrator.js +138 -0
  32. package/dist/server/server/config.js +17 -2
  33. package/dist/server/server/database.js +27 -12
  34. package/dist/server/server/global-config.js +143 -0
  35. package/dist/server/server/index.js +882 -252
  36. package/dist/server/server/init.js +579 -194
  37. package/dist/server/server/launch.js +29 -0
  38. package/dist/server/server/manifest-types.js +8 -0
  39. package/dist/server/server/manifest.js +454 -0
  40. package/dist/server/server/migrate-legacy.js +229 -0
  41. package/dist/server/server/parsers/event-parser.test.js +117 -0
  42. package/dist/server/server/parsers/scope-parser.js +74 -28
  43. package/dist/server/server/parsers/scope-parser.test.js +230 -0
  44. package/dist/server/server/project-context.js +255 -0
  45. package/dist/server/server/project-emitter.js +41 -0
  46. package/dist/server/server/project-manager.js +297 -0
  47. package/dist/server/server/routes/config-routes.js +1 -3
  48. package/dist/server/server/routes/data-routes.js +22 -110
  49. package/dist/server/server/routes/dispatch-routes.js +15 -9
  50. package/dist/server/server/routes/git-routes.js +74 -0
  51. package/dist/server/server/routes/manifest-routes.js +319 -0
  52. package/dist/server/server/routes/scope-routes.js +37 -23
  53. package/dist/server/server/routes/sync-routes.js +134 -0
  54. package/dist/server/server/routes/version-routes.js +1 -15
  55. package/dist/server/server/routes/workflow-routes.js +9 -3
  56. package/dist/server/server/schema.js +2 -0
  57. package/dist/server/server/services/batch-orchestrator.js +26 -16
  58. package/dist/server/server/services/claude-session-service.js +17 -14
  59. package/dist/server/server/services/deploy-service.test.js +119 -0
  60. package/dist/server/server/services/event-service.js +64 -1
  61. package/dist/server/server/services/event-service.test.js +191 -0
  62. package/dist/server/server/services/gate-service.test.js +105 -0
  63. package/dist/server/server/services/git-service.js +108 -4
  64. package/dist/server/server/services/github-service.js +110 -2
  65. package/dist/server/server/services/readiness-service.test.js +190 -0
  66. package/dist/server/server/services/scope-cache.js +5 -1
  67. package/dist/server/server/services/scope-cache.test.js +142 -0
  68. package/dist/server/server/services/scope-service.js +217 -126
  69. package/dist/server/server/services/scope-service.test.js +137 -0
  70. package/dist/server/server/services/sprint-orchestrator.js +7 -6
  71. package/dist/server/server/services/sprint-service.js +21 -1
  72. package/dist/server/server/services/sprint-service.test.js +238 -0
  73. package/dist/server/server/services/sync-service.js +434 -0
  74. package/dist/server/server/services/sync-types.js +2 -0
  75. package/dist/server/server/services/telemetry-service.js +143 -0
  76. package/dist/server/server/services/workflow-service.js +26 -5
  77. package/dist/server/server/services/workflow-service.test.js +159 -0
  78. package/dist/server/server/settings-sync.js +284 -0
  79. package/dist/server/server/update-planner.js +279 -0
  80. package/dist/server/server/utils/cc-hooks-parser.js +3 -0
  81. package/dist/server/server/utils/cc-hooks-parser.test.js +86 -0
  82. package/dist/server/server/utils/dispatch-utils.js +77 -20
  83. package/dist/server/server/utils/dispatch-utils.test.js +182 -0
  84. package/dist/server/server/utils/logger.js +37 -3
  85. package/dist/server/server/utils/package-info.js +30 -0
  86. package/dist/server/server/utils/route-helpers.js +10 -0
  87. package/dist/server/server/utils/terminal-launcher.js +79 -25
  88. package/dist/server/server/utils/worktree-manager.js +13 -4
  89. package/dist/server/server/validator.js +230 -0
  90. package/dist/server/server/watchers/global-watcher.js +63 -0
  91. package/dist/server/server/watchers/scope-watcher.js +27 -12
  92. package/dist/server/server/wizard/config-editor.js +237 -0
  93. package/dist/server/server/wizard/detect.js +96 -0
  94. package/dist/server/server/wizard/doctor.js +115 -0
  95. package/dist/server/server/wizard/index.js +155 -0
  96. package/dist/server/server/wizard/phases/confirm.js +39 -0
  97. package/dist/server/server/wizard/phases/project-setup.js +90 -0
  98. package/dist/server/server/wizard/phases/setup-wizard.js +66 -0
  99. package/dist/server/server/wizard/phases/welcome.js +35 -0
  100. package/dist/server/server/wizard/phases/workflow-setup.js +22 -0
  101. package/dist/server/server/wizard/types.js +29 -0
  102. package/dist/server/server/wizard/ui.js +74 -0
  103. package/dist/server/shared/__fixtures__/workflow-configs.js +75 -0
  104. package/dist/server/shared/default-workflow.json +65 -0
  105. package/dist/server/shared/onboarding-tour.test.js +81 -0
  106. package/dist/server/shared/project-colors.js +24 -0
  107. package/dist/server/shared/workflow-config.test.js +84 -0
  108. package/dist/server/shared/workflow-engine.test.js +302 -0
  109. package/dist/server/shared/workflow-normalizer.js +101 -0
  110. package/dist/server/shared/workflow-normalizer.test.js +100 -0
  111. package/dist/server/src/components/onboarding/tour-steps.js +84 -0
  112. package/package.json +20 -15
  113. package/schemas/orbital.config.schema.json +16 -1
  114. package/scripts/postinstall.js +55 -7
  115. package/server/__tests__/data-routes.test.ts +149 -0
  116. package/server/__tests__/helpers/db.ts +19 -0
  117. package/server/__tests__/helpers/mock-emitter.ts +10 -0
  118. package/server/__tests__/scope-routes.test.ts +157 -0
  119. package/server/__tests__/sprint-routes.test.ts +118 -0
  120. package/server/__tests__/workflow-routes.test.ts +120 -0
  121. package/server/config-migrator.ts +163 -0
  122. package/server/config.ts +26 -2
  123. package/server/database.ts +35 -18
  124. package/server/global-config.ts +200 -0
  125. package/server/index.ts +975 -287
  126. package/server/init.ts +625 -182
  127. package/server/launch.ts +32 -0
  128. package/server/manifest-types.ts +145 -0
  129. package/server/manifest.ts +494 -0
  130. package/server/migrate-legacy.ts +290 -0
  131. package/server/parsers/event-parser.test.ts +135 -0
  132. package/server/parsers/scope-parser.test.ts +270 -0
  133. package/server/parsers/scope-parser.ts +79 -31
  134. package/server/project-context.ts +309 -0
  135. package/server/project-emitter.ts +50 -0
  136. package/server/project-manager.ts +369 -0
  137. package/server/routes/config-routes.ts +3 -5
  138. package/server/routes/data-routes.ts +28 -141
  139. package/server/routes/dispatch-routes.ts +19 -11
  140. package/server/routes/git-routes.ts +77 -0
  141. package/server/routes/manifest-routes.ts +388 -0
  142. package/server/routes/scope-routes.ts +29 -25
  143. package/server/routes/sync-routes.ts +175 -0
  144. package/server/routes/version-routes.ts +1 -16
  145. package/server/routes/workflow-routes.ts +9 -3
  146. package/server/schema.ts +2 -0
  147. package/server/services/batch-orchestrator.ts +24 -16
  148. package/server/services/claude-session-service.ts +16 -14
  149. package/server/services/deploy-service.test.ts +145 -0
  150. package/server/services/deploy-service.ts +2 -2
  151. package/server/services/event-service.test.ts +242 -0
  152. package/server/services/event-service.ts +92 -3
  153. package/server/services/gate-service.test.ts +131 -0
  154. package/server/services/gate-service.ts +2 -2
  155. package/server/services/git-service.ts +137 -4
  156. package/server/services/github-service.ts +120 -2
  157. package/server/services/readiness-service.test.ts +217 -0
  158. package/server/services/scope-cache.test.ts +167 -0
  159. package/server/services/scope-cache.ts +4 -1
  160. package/server/services/scope-service.test.ts +169 -0
  161. package/server/services/scope-service.ts +220 -126
  162. package/server/services/sprint-orchestrator.ts +7 -7
  163. package/server/services/sprint-service.test.ts +271 -0
  164. package/server/services/sprint-service.ts +27 -3
  165. package/server/services/sync-service.ts +482 -0
  166. package/server/services/sync-types.ts +77 -0
  167. package/server/services/telemetry-service.ts +195 -0
  168. package/server/services/workflow-service.test.ts +190 -0
  169. package/server/services/workflow-service.ts +29 -9
  170. package/server/settings-sync.ts +359 -0
  171. package/server/update-planner.ts +346 -0
  172. package/server/utils/cc-hooks-parser.test.ts +96 -0
  173. package/server/utils/cc-hooks-parser.ts +4 -0
  174. package/server/utils/dispatch-utils.test.ts +245 -0
  175. package/server/utils/dispatch-utils.ts +97 -27
  176. package/server/utils/logger.ts +40 -3
  177. package/server/utils/package-info.ts +32 -0
  178. package/server/utils/route-helpers.ts +12 -0
  179. package/server/utils/terminal-launcher.ts +85 -25
  180. package/server/utils/worktree-manager.ts +9 -4
  181. package/server/validator.ts +270 -0
  182. package/server/watchers/global-watcher.ts +77 -0
  183. package/server/watchers/scope-watcher.ts +21 -9
  184. package/server/wizard/config-editor.ts +248 -0
  185. package/server/wizard/detect.ts +104 -0
  186. package/server/wizard/doctor.ts +114 -0
  187. package/server/wizard/index.ts +187 -0
  188. package/server/wizard/phases/confirm.ts +45 -0
  189. package/server/wizard/phases/project-setup.ts +106 -0
  190. package/server/wizard/phases/setup-wizard.ts +78 -0
  191. package/server/wizard/phases/welcome.ts +43 -0
  192. package/server/wizard/phases/workflow-setup.ts +28 -0
  193. package/server/wizard/types.ts +56 -0
  194. package/server/wizard/ui.ts +93 -0
  195. package/shared/__fixtures__/workflow-configs.ts +80 -0
  196. package/shared/default-workflow.json +65 -0
  197. package/shared/onboarding-tour.test.ts +94 -0
  198. package/shared/project-colors.ts +24 -0
  199. package/shared/workflow-config.test.ts +111 -0
  200. package/shared/workflow-config.ts +7 -0
  201. package/shared/workflow-engine.test.ts +388 -0
  202. package/shared/workflow-normalizer.test.ts +119 -0
  203. package/shared/workflow-normalizer.ts +118 -0
  204. package/templates/hooks/end-session.sh +3 -1
  205. package/templates/hooks/orbital-emit.sh +2 -2
  206. package/templates/hooks/orbital-report-deploy.sh +4 -4
  207. package/templates/hooks/orbital-report-gates.sh +4 -4
  208. package/templates/hooks/orbital-scope-update.sh +1 -1
  209. package/templates/hooks/scope-create-cleanup.sh +2 -2
  210. package/templates/hooks/scope-create-gate.sh +0 -1
  211. package/templates/hooks/scope-helpers.sh +18 -0
  212. package/templates/hooks/scope-prepare.sh +66 -11
  213. package/templates/migrations/renames.json +1 -0
  214. package/templates/orbital.config.json +7 -2
  215. package/templates/settings-hooks.json +1 -1
  216. package/templates/skills/git-commit/SKILL.md +9 -4
  217. package/templates/skills/git-dev/SKILL.md +8 -3
  218. package/templates/skills/git-main/SKILL.md +8 -2
  219. package/templates/skills/git-production/SKILL.md +6 -2
  220. package/templates/skills/git-staging/SKILL.md +8 -3
  221. package/templates/skills/scope-create/SKILL.md +17 -3
  222. package/templates/skills/scope-fix-review/SKILL.md +6 -3
  223. package/templates/skills/scope-implement/SKILL.md +4 -1
  224. package/templates/skills/scope-post-review/SKILL.md +63 -5
  225. package/templates/skills/scope-pre-review/SKILL.md +5 -2
  226. package/templates/skills/scope-verify/SKILL.md +5 -3
  227. package/templates/skills/test-code-review/SKILL.md +41 -33
  228. package/templates/skills/test-scaffold/SKILL.md +222 -0
  229. package/dist/assets/WorkflowVisualizer-BZ21PIIF.js +0 -84
  230. package/dist/assets/charts-D__PA1zp.js +0 -72
  231. package/dist/assets/index-D1G6i0nS.css +0 -1
  232. package/dist/assets/index-DpItvKpf.js +0 -419
  233. package/dist/assets/ui-BvF022GT.js +0 -53
  234. package/index.html +0 -15
  235. package/postcss.config.js +0 -6
  236. package/src/App.tsx +0 -33
  237. package/src/components/AgentBadge.tsx +0 -40
  238. package/src/components/BatchPreflightModal.tsx +0 -115
  239. package/src/components/CardDisplayToggle.tsx +0 -74
  240. package/src/components/ColumnHeaderActions.tsx +0 -55
  241. package/src/components/ColumnMenu.tsx +0 -99
  242. package/src/components/DeployHistory.tsx +0 -141
  243. package/src/components/DispatchModal.tsx +0 -164
  244. package/src/components/DispatchPopover.tsx +0 -139
  245. package/src/components/DragOverlay.tsx +0 -25
  246. package/src/components/DriftSidebar.tsx +0 -140
  247. package/src/components/EnvironmentStrip.tsx +0 -88
  248. package/src/components/ErrorBoundary.tsx +0 -62
  249. package/src/components/FilterChip.tsx +0 -105
  250. package/src/components/GateIndicator.tsx +0 -33
  251. package/src/components/IdeaDetailModal.tsx +0 -190
  252. package/src/components/IdeaFormDialog.tsx +0 -113
  253. package/src/components/KanbanColumn.tsx +0 -201
  254. package/src/components/MarkdownRenderer.tsx +0 -114
  255. package/src/components/NeonGrid.tsx +0 -128
  256. package/src/components/PromotionQueue.tsx +0 -89
  257. package/src/components/ScopeCard.tsx +0 -234
  258. package/src/components/ScopeDetailModal.tsx +0 -255
  259. package/src/components/ScopeFilterBar.tsx +0 -152
  260. package/src/components/SearchInput.tsx +0 -102
  261. package/src/components/SessionPanel.tsx +0 -335
  262. package/src/components/SprintContainer.tsx +0 -303
  263. package/src/components/SprintDependencyDialog.tsx +0 -78
  264. package/src/components/SprintPreflightModal.tsx +0 -138
  265. package/src/components/StatusBar.tsx +0 -168
  266. package/src/components/SwimCell.tsx +0 -67
  267. package/src/components/SwimLaneRow.tsx +0 -94
  268. package/src/components/SwimlaneBoardView.tsx +0 -108
  269. package/src/components/VersionBadge.tsx +0 -139
  270. package/src/components/ViewModeSelector.tsx +0 -114
  271. package/src/components/config/AgentChip.tsx +0 -53
  272. package/src/components/config/AgentCreateDialog.tsx +0 -321
  273. package/src/components/config/AgentEditor.tsx +0 -175
  274. package/src/components/config/DirectoryTree.tsx +0 -582
  275. package/src/components/config/FileEditor.tsx +0 -550
  276. package/src/components/config/HookChip.tsx +0 -50
  277. package/src/components/config/StageCard.tsx +0 -198
  278. package/src/components/config/TransitionZone.tsx +0 -173
  279. package/src/components/config/UnifiedWorkflowPipeline.tsx +0 -216
  280. package/src/components/config/WorkflowPipeline.tsx +0 -161
  281. package/src/components/source-control/BranchList.tsx +0 -93
  282. package/src/components/source-control/BranchPanel.tsx +0 -105
  283. package/src/components/source-control/CommitLog.tsx +0 -100
  284. package/src/components/source-control/CommitRow.tsx +0 -47
  285. package/src/components/source-control/GitHubPanel.tsx +0 -110
  286. package/src/components/source-control/GitHubSetupGuide.tsx +0 -52
  287. package/src/components/source-control/GitOverviewBar.tsx +0 -101
  288. package/src/components/source-control/PullRequestList.tsx +0 -69
  289. package/src/components/source-control/WorktreeList.tsx +0 -80
  290. package/src/components/ui/badge.tsx +0 -41
  291. package/src/components/ui/button.tsx +0 -55
  292. package/src/components/ui/card.tsx +0 -78
  293. package/src/components/ui/dialog.tsx +0 -94
  294. package/src/components/ui/popover.tsx +0 -33
  295. package/src/components/ui/scroll-area.tsx +0 -54
  296. package/src/components/ui/separator.tsx +0 -28
  297. package/src/components/ui/tabs.tsx +0 -52
  298. package/src/components/ui/toggle-switch.tsx +0 -35
  299. package/src/components/ui/tooltip.tsx +0 -27
  300. package/src/components/workflow/AddEdgeDialog.tsx +0 -217
  301. package/src/components/workflow/AddListDialog.tsx +0 -201
  302. package/src/components/workflow/ChecklistEditor.tsx +0 -239
  303. package/src/components/workflow/CommandPrefixManager.tsx +0 -118
  304. package/src/components/workflow/ConfigSettingsPanel.tsx +0 -189
  305. package/src/components/workflow/DirectionSelector.tsx +0 -133
  306. package/src/components/workflow/DispatchConfigPanel.tsx +0 -180
  307. package/src/components/workflow/EdgeDetailPanel.tsx +0 -236
  308. package/src/components/workflow/EdgePropertyEditor.tsx +0 -251
  309. package/src/components/workflow/EditToolbar.tsx +0 -138
  310. package/src/components/workflow/HookDetailPanel.tsx +0 -250
  311. package/src/components/workflow/HookExecutionLog.tsx +0 -24
  312. package/src/components/workflow/HookSourceModal.tsx +0 -129
  313. package/src/components/workflow/HooksDashboard.tsx +0 -363
  314. package/src/components/workflow/ListPropertyEditor.tsx +0 -251
  315. package/src/components/workflow/MigrationPreviewDialog.tsx +0 -237
  316. package/src/components/workflow/MovementRulesPanel.tsx +0 -188
  317. package/src/components/workflow/NodeDetailPanel.tsx +0 -245
  318. package/src/components/workflow/PresetSelector.tsx +0 -414
  319. package/src/components/workflow/SkillCommandBuilder.tsx +0 -174
  320. package/src/components/workflow/WorkflowEdgeComponent.tsx +0 -145
  321. package/src/components/workflow/WorkflowNode.tsx +0 -147
  322. package/src/components/workflow/graphLayout.ts +0 -186
  323. package/src/components/workflow/mergeHooks.ts +0 -85
  324. package/src/components/workflow/useEditHistory.ts +0 -88
  325. package/src/components/workflow/useWorkflowEditor.ts +0 -262
  326. package/src/components/workflow/validateConfig.ts +0 -70
  327. package/src/hooks/useActiveDispatches.ts +0 -198
  328. package/src/hooks/useBoardSettings.ts +0 -170
  329. package/src/hooks/useCardDisplay.ts +0 -57
  330. package/src/hooks/useCcHooks.ts +0 -24
  331. package/src/hooks/useConfigTree.ts +0 -51
  332. package/src/hooks/useEnforcementRules.ts +0 -46
  333. package/src/hooks/useEvents.ts +0 -59
  334. package/src/hooks/useFileEditor.ts +0 -165
  335. package/src/hooks/useGates.ts +0 -57
  336. package/src/hooks/useIdeaActions.ts +0 -53
  337. package/src/hooks/useKanbanDnd.ts +0 -410
  338. package/src/hooks/useOrbitalConfig.ts +0 -54
  339. package/src/hooks/usePipeline.ts +0 -47
  340. package/src/hooks/usePipelineData.ts +0 -338
  341. package/src/hooks/useReconnect.ts +0 -25
  342. package/src/hooks/useScopeFilters.ts +0 -125
  343. package/src/hooks/useScopeSessions.ts +0 -44
  344. package/src/hooks/useScopes.ts +0 -67
  345. package/src/hooks/useSearch.ts +0 -67
  346. package/src/hooks/useSettings.tsx +0 -187
  347. package/src/hooks/useSocket.ts +0 -25
  348. package/src/hooks/useSourceControl.ts +0 -105
  349. package/src/hooks/useSprintPreflight.ts +0 -55
  350. package/src/hooks/useSprints.ts +0 -154
  351. package/src/hooks/useStatusBarHighlight.ts +0 -18
  352. package/src/hooks/useSwimlaneBoardSettings.ts +0 -104
  353. package/src/hooks/useTheme.ts +0 -9
  354. package/src/hooks/useTransitionReadiness.ts +0 -53
  355. package/src/hooks/useVersion.ts +0 -155
  356. package/src/hooks/useViolations.ts +0 -65
  357. package/src/hooks/useWorkflow.tsx +0 -125
  358. package/src/hooks/useZoomModifier.ts +0 -19
  359. package/src/index.css +0 -797
  360. package/src/layouts/DashboardLayout.tsx +0 -113
  361. package/src/lib/collisionDetection.ts +0 -20
  362. package/src/lib/scope-fields.ts +0 -61
  363. package/src/lib/swimlane.ts +0 -146
  364. package/src/lib/utils.ts +0 -15
  365. package/src/main.tsx +0 -19
  366. package/src/socket.ts +0 -11
  367. package/src/types/index.ts +0 -497
  368. package/src/views/AgentFeed.tsx +0 -339
  369. package/src/views/DeployPipeline.tsx +0 -59
  370. package/src/views/EnforcementView.tsx +0 -378
  371. package/src/views/PrimitivesConfig.tsx +0 -500
  372. package/src/views/QualityGates.tsx +0 -1012
  373. package/src/views/ScopeBoard.tsx +0 -454
  374. package/src/views/SessionTimeline.tsx +0 -516
  375. package/src/views/Settings.tsx +0 -183
  376. package/src/views/SourceControl.tsx +0 -95
  377. package/src/views/WorkflowVisualizer.tsx +0 -382
  378. package/tailwind.config.js +0 -161
  379. package/tsconfig.json +0 -25
  380. package/vite.config.ts +0 -49
@@ -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
- }