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,234 +0,0 @@
1
- import { useDraggable } from '@dnd-kit/core';
2
- import { CSS } from '@dnd-kit/utilities';
3
- import { Lightbulb, Sparkles, AlertTriangle, Undo2, X } from 'lucide-react';
4
- import type { Scope, CardDisplayConfig } from '@/types';
5
- import { Card, CardContent } from '@/components/ui/card';
6
- import { Popover, PopoverTrigger, PopoverContent } from '@/components/ui/popover';
7
- import { cn, formatScopeId } from '@/lib/utils';
8
- import { useActiveDispatches } from '@/hooks/useActiveDispatches';
9
- import { useWorkflow } from '@/hooks/useWorkflow';
10
-
11
- interface ScopeCardProps {
12
- scope: Scope;
13
- onClick?: (scope: Scope) => void;
14
- isDragOverlay?: boolean;
15
- cardDisplay?: CardDisplayConfig;
16
- dimmed?: boolean;
17
- }
18
-
19
- const PRIORITY_COLOR: Record<string, string> = {
20
- critical: 'border-ask-red text-ask-red',
21
- high: 'border-warning-amber text-warning-amber',
22
- medium: 'border-accent-blue text-accent-blue',
23
- low: 'border-muted-foreground/30 text-muted-foreground',
24
- };
25
-
26
- const CATEGORY_COLOR: Record<string, string> = {
27
- 'feature': 'border-category-feature text-category-feature',
28
- 'bugfix': 'border-category-bugfix text-category-bugfix',
29
- 'refactor': 'border-category-refactor text-category-refactor',
30
- 'infrastructure': 'border-category-infrastructure text-category-infrastructure',
31
- 'docs': 'border-category-docs text-category-docs',
32
- };
33
-
34
- const CATEGORY_BORDER: Record<string, string> = {
35
- 'feature': 'border-l-2 border-l-category-feature scope-cat-feature',
36
- 'bugfix': 'border-l-2 border-l-category-bugfix scope-cat-bugfix',
37
- 'refactor': 'border-l-2 border-l-category-refactor scope-cat-refactor',
38
- 'infrastructure': 'border-l-2 border-l-category-infrastructure scope-cat-infrastructure',
39
- 'docs': 'border-l-2 border-l-category-docs scope-cat-docs',
40
- };
41
-
42
- const GHOST = 'inline-block rounded border px-1.5 py-0 text-[10px] uppercase bg-transparent';
43
-
44
- interface AbandonedPopoverProps {
45
- scopeId: number;
46
- info: { from_status: string | null };
47
- onRecover: (scopeId: number, fromStatus: string) => Promise<void>;
48
- onDismiss: (scopeId: number) => Promise<void>;
49
- }
50
-
51
- function AbandonedPopover({ scopeId, info, onRecover, onDismiss }: AbandonedPopoverProps) {
52
- return (
53
- <Popover>
54
- <PopoverTrigger asChild onClick={(e) => e.stopPropagation()}>
55
- <button className="flex items-center gap-0.5 text-amber-500 hover:text-amber-400 transition-colors">
56
- <AlertTriangle className="h-3 w-3" />
57
- </button>
58
- </PopoverTrigger>
59
- <PopoverContent
60
- className="w-56 p-3"
61
- side="top"
62
- align="end"
63
- onClick={(e) => e.stopPropagation()}
64
- >
65
- <p className="text-xs text-muted-foreground mb-2">
66
- Session ended without completing work.
67
- </p>
68
- <div className="flex flex-col gap-1.5">
69
- {info.from_status && (
70
- <button
71
- className="flex items-center gap-1.5 text-xs px-2 py-1.5 rounded bg-amber-500/10 text-amber-500 hover:bg-amber-500/20 transition-colors"
72
- onClick={() => onRecover(scopeId, info.from_status!)}
73
- >
74
- <Undo2 className="h-3 w-3" />
75
- Revert to {info.from_status}
76
- </button>
77
- )}
78
- <button
79
- className="flex items-center gap-1.5 text-xs px-2 py-1.5 rounded bg-muted text-muted-foreground hover:bg-muted/80 transition-colors"
80
- onClick={() => onDismiss(scopeId)}
81
- >
82
- <X className="h-3 w-3" />
83
- Keep here
84
- </button>
85
- </div>
86
- </PopoverContent>
87
- </Popover>
88
- );
89
- }
90
-
91
- function abbreviateEffort(raw: string): string {
92
- const s = raw.toLowerCase().trim();
93
- const minMatch = s.match(/^~?(\d+)(?:\s*-\s*\d+)?\s*min/);
94
- if (minMatch) return `${minMatch[1]}M`;
95
- const hrMatch = s.match(/^~?(\d+(?:\.\d+)?)(?:\s*-\s*\d+(?:\.\d+)?)?\s*hour/);
96
- if (hrMatch) return `${hrMatch[1]}H`;
97
- const parenMatch = s.match(/\((\d+(?:\.\d+)?)(?:\s*-\s*\d+(?:\.\d+)?)?\s*(hour|min)/);
98
- if (parenMatch) return `${parenMatch[1]}${parenMatch[2].startsWith('h') ? 'H' : 'M'}`;
99
- if (s.includes('large') || s.includes('multi')) return 'LG';
100
- if (s.includes('medium') || s.includes('half')) return 'MD';
101
- if (s.includes('small')) return 'SM';
102
- if (s === 'tbd') return 'TBD';
103
- return 'TBD';
104
- }
105
-
106
- export function ScopeCard({ scope, onClick, isDragOverlay, cardDisplay, dimmed }: ScopeCardProps) {
107
- const {
108
- attributes,
109
- listeners,
110
- setNodeRef,
111
- transform,
112
- isDragging,
113
- } = useDraggable({
114
- id: scope.id,
115
- disabled: isDragOverlay || dimmed,
116
- });
117
-
118
- const style = transform
119
- ? { transform: CSS.Translate.toString(transform) }
120
- : undefined;
121
-
122
- const { engine } = useWorkflow();
123
- const { activeScopes, abandonedScopes, recoverScope, dismissAbandoned } = useActiveDispatches();
124
- const entryPointId = engine.getEntryPoint().id;
125
- const isIdea = scope.status === entryPointId;
126
- const isGhost = isIdea && !!scope.is_ghost;
127
- const isDispatched = !isIdea && activeScopes.has(scope.id);
128
- const abandonedInfo = !isIdea ? abandonedScopes.get(scope.id) : undefined;
129
- const isAbandoned = !!abandonedInfo && !isDispatched;
130
-
131
- return (
132
- <Card
133
- ref={setNodeRef}
134
- style={style}
135
- className={cn(
136
- 'scope-card cursor-grab transition-[colors,opacity] duration-200 hover:bg-surface-light active:cursor-grabbing',
137
- isGhost
138
- ? 'scope-card-ghost ghost-shimmer opacity-70'
139
- : isIdea
140
- ? 'border-l-2 border-dashed border-l-warning-amber/60'
141
- : scope.category ? CATEGORY_BORDER[scope.category] : '',
142
- isDispatched && 'scope-card-dispatched',
143
- isAbandoned && 'scope-card-abandoned',
144
- isDragging && 'opacity-30',
145
- dimmed && !isDragging && 'opacity-30 cursor-default',
146
- )}
147
- onClick={() => {
148
- if (!isDragging) onClick?.(scope);
149
- }}
150
- {...attributes}
151
- {...listeners}
152
- >
153
- <CardContent className="px-2.5 py-1.5">
154
- {/* Header: ID/idea label + badges */}
155
- <div className="mb-1.5 flex items-center gap-1.5">
156
- {isGhost ? (
157
- <span className="flex items-center gap-1 text-xxs text-purple-400">
158
- <Sparkles className="h-3 w-3" />
159
- ai suggestion
160
- </span>
161
- ) : isIdea ? (
162
- <span className="flex items-center gap-1 text-xxs text-warning-amber">
163
- <Lightbulb className="h-3 w-3" />
164
- idea
165
- </span>
166
- ) : (
167
- <span className="font-mono text-xxs text-muted-foreground flex items-center gap-1">
168
- {isDispatched && <span className="h-1.5 w-1.5 rounded-full bg-pink-500 dispatch-pulse" />}
169
- {isAbandoned && <span className="h-1.5 w-1.5 rounded-full bg-amber-500" />}
170
- {formatScopeId(scope.id)}
171
- </span>
172
- )}
173
- {!isIdea && (
174
- <div className="ml-auto flex items-center gap-1">
175
- {isAbandoned && (
176
- <AbandonedPopover
177
- scopeId={scope.id}
178
- info={abandonedInfo}
179
- onRecover={recoverScope}
180
- onDismiss={dismissAbandoned}
181
- />
182
- )}
183
- {scope.effort_estimate && cardDisplay?.effort !== false && (
184
- <span className={cn(GHOST, 'effort-ghost border-muted-foreground/30 text-muted-foreground')}>
185
- {abbreviateEffort(scope.effort_estimate)}
186
- </span>
187
- )}
188
- {scope.category && cardDisplay?.category !== false && (
189
- <span className={cn(
190
- GHOST,
191
- CATEGORY_COLOR[scope.category] ?? 'border-muted-foreground/30 text-muted-foreground'
192
- )}>
193
- {scope.category}
194
- </span>
195
- )}
196
- {scope.priority && cardDisplay?.priority !== false && (
197
- <span className={cn(
198
- GHOST,
199
- PRIORITY_COLOR[scope.priority] ?? 'border-muted-foreground/30 text-muted-foreground'
200
- )}>
201
- {scope.priority}
202
- </span>
203
- )}
204
- </div>
205
- )}
206
- </div>
207
-
208
- {/* Title */}
209
- <h3 className="text-xs font-light leading-snug line-clamp-2">
210
- {scope.title}
211
- </h3>
212
-
213
- {/* Tags (ideas don't have tags) */}
214
- {!isIdea && scope.tags.length > 0 && cardDisplay?.tags !== false && (
215
- <div className="mt-2 flex flex-wrap gap-1">
216
- {scope.tags.slice(0, 3).map((tag) => (
217
- <span
218
- key={tag}
219
- className="glass-pill inline-block rounded bg-muted px-1.5 py-0.5 text-[10px] text-muted-foreground"
220
- >
221
- {tag}
222
- </span>
223
- ))}
224
- {scope.tags.length > 3 && (
225
- <span className="text-[10px] text-muted-foreground">
226
- +{scope.tags.length - 3}
227
- </span>
228
- )}
229
- </div>
230
- )}
231
- </CardContent>
232
- </Card>
233
- );
234
- }
@@ -1,255 +0,0 @@
1
- import { useState, useEffect, useCallback } from 'react';
2
- import { ExternalLink, X as XIcon, Plus } from 'lucide-react';
3
- import {
4
- Dialog, DialogContent, DialogHeader, DialogTitle, DialogDescription,
5
- } from '@/components/ui/dialog';
6
- import { Badge } from '@/components/ui/badge';
7
- import { Button } from '@/components/ui/button';
8
- import { ScrollArea } from '@/components/ui/scroll-area';
9
- import { Separator } from '@/components/ui/separator';
10
- import { MarkdownRenderer } from '@/components/MarkdownRenderer';
11
- import { SessionPanel } from '@/components/SessionPanel';
12
- import { useScopeSessions } from '@/hooks/useScopeSessions';
13
- import { useWorkflow } from '@/hooks/useWorkflow';
14
- import { formatScopeId } from '@/lib/utils';
15
- import type { Scope } from '@/types';
16
- import { PRIORITY_OPTIONS, EFFORT_BUCKETS, CATEGORY_OPTIONS } from '@/types';
17
-
18
- interface ScopeDetailModalProps {
19
- scope: Scope | null;
20
- open: boolean;
21
- onClose: () => void;
22
- }
23
-
24
- interface EditableFields {
25
- title: string;
26
- status: string;
27
- priority: string;
28
- effort_estimate: string;
29
- category: string;
30
- tags: string[];
31
- blocked_by: number[];
32
- blocks: number[];
33
- }
34
-
35
- const SELECT_CLS = 'h-6 rounded border border-border bg-muted/30 px-1.5 text-xxs text-foreground focus:outline-none focus:ring-1 focus:ring-primary/50';
36
-
37
- function fieldsFromScope(scope: Scope): EditableFields {
38
- return {
39
- title: scope.title, status: scope.status,
40
- priority: scope.priority ?? '', effort_estimate: scope.effort_estimate ?? '',
41
- category: scope.category ?? '', tags: [...scope.tags],
42
- blocked_by: [...scope.blocked_by], blocks: [...scope.blocks],
43
- };
44
- }
45
-
46
- function fieldsEqual(a: EditableFields, b: EditableFields): boolean {
47
- return a.title === b.title && a.status === b.status &&
48
- a.priority === b.priority && a.effort_estimate === b.effort_estimate &&
49
- a.category === b.category && JSON.stringify(a.tags) === JSON.stringify(b.tags) &&
50
- JSON.stringify(a.blocked_by) === JSON.stringify(b.blocked_by) &&
51
- JSON.stringify(a.blocks) === JSON.stringify(b.blocks);
52
- }
53
-
54
- function DepEditor({ label, ids, onRemove, onAdd }: {
55
- label: string; ids: number[];
56
- onRemove: (id: number) => void; onAdd: (val: string) => void;
57
- }) {
58
- const [editing, setEditing] = useState(false);
59
- const [val, setVal] = useState('');
60
- return (
61
- <span className="inline-flex items-center gap-1">
62
- {label}:
63
- {ids.map((id) => (
64
- <span key={id} className="group inline-flex items-center gap-0.5 rounded bg-muted px-1 py-0.5">
65
- {formatScopeId(id)}
66
- <button onClick={() => onRemove(id)} className="opacity-0 group-hover:opacity-100 transition-opacity">
67
- <XIcon className="h-2.5 w-2.5" />
68
- </button>
69
- </span>
70
- ))}
71
- {editing ? (
72
- <input autoFocus className="h-5 w-12 rounded bg-muted/50 px-1 text-xxs border border-primary/30 focus:outline-none"
73
- placeholder="ID" value={val}
74
- onChange={(e) => setVal(e.target.value)}
75
- onKeyDown={(e) => { if (e.key === 'Enter') { onAdd(val); setEditing(false); setVal(''); } if (e.key === 'Escape') setEditing(false); }}
76
- onBlur={() => setEditing(false)} />
77
- ) : (
78
- <button onClick={() => setEditing(true)} className="hover:text-foreground transition-colors"><Plus className="h-3 w-3" /></button>
79
- )}
80
- </span>
81
- );
82
- }
83
-
84
- export function ScopeDetailModal({ scope, open, onClose }: ScopeDetailModalProps) {
85
- const { engine } = useWorkflow();
86
- const { sessions, loading: sessionsLoading } = useScopeSessions(scope?.id ?? null);
87
- const [fields, setFields] = useState<EditableFields | null>(null);
88
- const [saved, setSaved] = useState<EditableFields | null>(null);
89
- const [saving, setSaving] = useState(false);
90
- const [error, setError] = useState<string | null>(null);
91
- const [tagInput, setTagInput] = useState('');
92
-
93
- const isDirty = fields && saved ? !fieldsEqual(fields, saved) : false;
94
-
95
- useEffect(() => {
96
- if (scope && open) {
97
- const f = fieldsFromScope(scope);
98
- setFields(f); setSaved(f); setError(null); setTagInput('');
99
- }
100
- }, [scope?.id, scope?.updated_at, open]); // eslint-disable-line react-hooks/exhaustive-deps
101
-
102
- const save = useCallback(async () => {
103
- if (!scope || !fields || !isDirty || saving) return;
104
- setSaving(true); setError(null);
105
- try {
106
- const payload: Record<string, unknown> = {};
107
- if (saved) {
108
- if (fields.title !== saved.title) payload.title = fields.title;
109
- if (fields.status !== saved.status) payload.status = fields.status;
110
- if (fields.priority !== saved.priority) payload.priority = fields.priority || null;
111
- if (fields.effort_estimate !== saved.effort_estimate) payload.effort_estimate = fields.effort_estimate || null;
112
- if (fields.category !== saved.category) payload.category = fields.category || null;
113
- if (JSON.stringify(fields.tags) !== JSON.stringify(saved.tags)) payload.tags = fields.tags;
114
- if (JSON.stringify(fields.blocked_by) !== JSON.stringify(saved.blocked_by)) payload.blocked_by = fields.blocked_by;
115
- if (JSON.stringify(fields.blocks) !== JSON.stringify(saved.blocks)) payload.blocks = fields.blocks;
116
- }
117
- const res = await fetch(`/api/orbital/scopes/${scope.id}`, {
118
- method: 'PATCH', headers: { 'Content-Type': 'application/json' },
119
- body: JSON.stringify(payload),
120
- });
121
- if (!res.ok) {
122
- const body = await res.json().catch(() => ({ error: 'Save failed' }));
123
- setError(body.error ?? `HTTP ${res.status}`); return;
124
- }
125
- setSaved({ ...fields });
126
- } catch { setError('Network error — could not save'); }
127
- finally { setSaving(false); }
128
- }, [scope, fields, saved, isDirty, saving]);
129
-
130
- function handleClose() {
131
- if (isDirty && window.confirm('Save changes before closing?')) { save().then(onClose); return; }
132
- onClose();
133
- }
134
-
135
- function update(partial: Partial<EditableFields>) {
136
- setFields((prev) => prev ? { ...prev, ...partial } : prev); setError(null);
137
- }
138
-
139
- function addTag(tag: string) {
140
- const t = tag.trim().toLowerCase();
141
- if (!t || !fields) return;
142
- if (!fields.tags.includes(t)) update({ tags: [...fields.tags, t] });
143
- setTagInput('');
144
- }
145
-
146
- function addDep(field: 'blocked_by' | 'blocks', value: string) {
147
- const num = parseInt(value, 10);
148
- if (isNaN(num) || num <= 0 || !fields) return;
149
- if (!fields[field].includes(num)) update({ [field]: [...fields[field], num] });
150
- }
151
-
152
- if (!scope || !fields) return null;
153
-
154
- const validTargets = engine.getValidTargets(scope.status);
155
- const statusOptions = [scope.status, ...validTargets.filter((t) => t !== scope.status)];
156
-
157
- return (
158
- <Dialog open={open} onOpenChange={(isOpen) => { if (!isOpen) handleClose(); }}>
159
- <DialogContent className="max-w-[min(72rem,calc(100vw_-_2rem))] h-[85vh] flex flex-col p-0 gap-0 overflow-hidden">
160
- <DialogHeader className="px-4 pt-3 pb-2">
161
- <div className="flex items-start gap-3 pr-8">
162
- <span className="font-mono text-xxs text-muted-foreground mt-1.5">{formatScopeId(scope.id)}</span>
163
- <div className="min-w-0 flex-1">
164
- <DialogTitle asChild>
165
- <input className="w-full bg-transparent text-sm font-normal text-foreground border-none focus:outline-none focus:ring-0 placeholder:text-muted-foreground leading-tight"
166
- value={fields.title} onChange={(e) => update({ title: e.target.value })} placeholder="Scope title..." />
167
- </DialogTitle>
168
- <DialogDescription asChild>
169
- <div className="mt-2 flex flex-wrap items-center gap-2">
170
- <select className={SELECT_CLS} value={fields.status} onChange={(e) => update({ status: e.target.value })}>
171
- {statusOptions.map((s) => <option key={s} value={s}>{s}</option>)}
172
- </select>
173
- <select className={SELECT_CLS} value={fields.priority} onChange={(e) => update({ priority: e.target.value })}>
174
- <option value="">priority</option>
175
- {PRIORITY_OPTIONS.map((p) => <option key={p} value={p}>{p}</option>)}
176
- </select>
177
- <select className={SELECT_CLS} value={fields.effort_estimate} onChange={(e) => update({ effort_estimate: e.target.value })}>
178
- <option value="">effort</option>
179
- {EFFORT_BUCKETS.map((e) => <option key={e} value={e}>{e}</option>)}
180
- </select>
181
- <select className={SELECT_CLS} value={fields.category} onChange={(e) => update({ category: e.target.value })}>
182
- <option value="">category</option>
183
- {CATEGORY_OPTIONS.map((c) => <option key={c} value={c}>{c}</option>)}
184
- </select>
185
- {fields.tags.map((tag) => (
186
- <span key={tag} className="group inline-flex items-center gap-0.5 glass-pill rounded bg-muted px-1.5 py-0.5 text-xxs text-muted-foreground">
187
- {tag}
188
- <button onClick={() => update({ tags: fields.tags.filter((t) => t !== tag) })} className="opacity-0 group-hover:opacity-100 transition-opacity">
189
- <XIcon className="h-2.5 w-2.5" />
190
- </button>
191
- </span>
192
- ))}
193
- <input className="h-5 w-16 rounded bg-transparent text-xxs text-muted-foreground placeholder:text-muted-foreground/50 border-none focus:outline-none"
194
- placeholder="+tag" value={tagInput} onChange={(e) => setTagInput(e.target.value)}
195
- onKeyDown={(e) => { if (e.key === 'Enter') { e.preventDefault(); addTag(tagInput); } }}
196
- onBlur={() => { if (tagInput.trim()) addTag(tagInput); }} />
197
- </div>
198
- </DialogDescription>
199
- </div>
200
- </div>
201
- <div className="mt-2 flex flex-wrap items-center gap-x-4 gap-y-1 text-xxs text-muted-foreground">
202
- <DepEditor label="Blocked by" ids={fields.blocked_by}
203
- onRemove={(id) => update({ blocked_by: fields.blocked_by.filter((d) => d !== id) })}
204
- onAdd={(v) => addDep('blocked_by', v)} />
205
- <DepEditor label="Blocks" ids={fields.blocks}
206
- onRemove={(id) => update({ blocks: fields.blocks.filter((d) => d !== id) })}
207
- onAdd={(v) => addDep('blocks', v)} />
208
- <span className="flex items-center gap-1">
209
- <ExternalLink className="h-3 w-3" />
210
- <span className="truncate max-w-[300px]">{scope.file_path}</span>
211
- </span>
212
- </div>
213
- </DialogHeader>
214
-
215
- <Separator />
216
-
217
- {error && (
218
- <div className="mx-4 mt-2 flex items-center gap-2 rounded border border-red-500/30 bg-red-500/10 px-3 py-1.5 text-xs text-red-400">
219
- <span className="flex-1">{error}</span>
220
- <button onClick={() => setError(null)} className="shrink-0 hover:text-red-200 transition-colors"><XIcon className="h-3.5 w-3.5" /></button>
221
- </div>
222
- )}
223
-
224
- <div className="flex flex-1 min-h-0">
225
- <div className="flex-[6] min-w-0 border-r bg-[#0a0a12]">
226
- <ScrollArea className="h-full">
227
- <div className="px-6 py-5">
228
- {scope.raw_content ? <MarkdownRenderer content={scope.raw_content} /> : (
229
- <p className="text-xs text-muted-foreground italic">No content available</p>
230
- )}
231
- </div>
232
- </ScrollArea>
233
- </div>
234
- <div className="flex-[4] min-w-0">
235
- <div className="flex h-full flex-col p-4">
236
- <SessionPanel sessions={sessions} loading={sessionsLoading} />
237
- </div>
238
- </div>
239
- </div>
240
-
241
- {isDirty && (
242
- <div className="mx-4 mb-2 flex items-center gap-2 rounded border border-border bg-card px-3 py-2">
243
- <Badge variant="outline">Unsaved changes</Badge>
244
- <div className="flex-1" />
245
- <Button variant="ghost" size="sm" onClick={() => { setFields(saved); setError(null); }}>Discard</Button>
246
- <Button size="sm" onClick={() => save()} disabled={saving}>
247
- {saving ? 'Saving...' : 'Save'}
248
- </Button>
249
- {error && <span className="text-xs text-destructive ml-2">{error}</span>}
250
- </div>
251
- )}
252
- </DialogContent>
253
- </Dialog>
254
- );
255
- }
@@ -1,152 +0,0 @@
1
- import { FilterChip } from './FilterChip';
2
- import { SearchInput } from './SearchInput';
3
- import { Badge } from '@/components/ui/badge';
4
- import { X } from 'lucide-react';
5
- import { cn } from '@/lib/utils';
6
- import { useTheme } from '@/hooks/useTheme';
7
- import type { FilterField, ScopeFilterState } from '@/types';
8
- import type { FilterOption } from '@/hooks/useScopeFilters';
9
- import type { SearchMode } from '@/hooks/useSearch';
10
-
11
- interface ScopeFilterBarProps {
12
- filters: ScopeFilterState;
13
- optionsWithCounts: Record<FilterField, FilterOption[]>;
14
- onToggle: (field: FilterField, value: string) => void;
15
- onClearField: (field: FilterField) => void;
16
- onClearAll: () => void;
17
- hasActiveFilters: boolean;
18
- searchQuery?: string;
19
- searchMode?: SearchMode;
20
- searchIsStale?: boolean;
21
- onSearchChange?: (query: string) => void;
22
- onSearchModeChange?: (mode: SearchMode) => void;
23
- }
24
-
25
- const CHIP_CONFIG: { field: FilterField; label: string; glowClass: string }[] = [
26
- { field: 'priority', label: 'Priority', glowClass: 'glow-amber' },
27
- { field: 'category', label: 'Category', glowClass: 'glow-blue' },
28
- { field: 'tags', label: 'Tags', glowClass: 'glow-blue' },
29
- { field: 'effort', label: 'Effort', glowClass: '' },
30
- { field: 'dependencies', label: 'Deps', glowClass: 'glow-red' },
31
- ];
32
-
33
- const FIELD_LABEL: Record<FilterField, string> = {
34
- priority: 'Priority',
35
- category: 'Category',
36
- tags: 'Tag',
37
- effort: 'Effort',
38
- dependencies: 'Dep',
39
- };
40
-
41
- const CATEGORY_COLOR: Record<string, string> = {
42
- feature: '#536dfe',
43
- bugfix: '#ff1744',
44
- refactor: '#8B5CF6',
45
- infrastructure: '#40c4ff',
46
- docs: '#6B7280',
47
- };
48
-
49
- export function ScopeFilterBar({
50
- filters,
51
- optionsWithCounts,
52
- onToggle,
53
- onClearField,
54
- onClearAll,
55
- hasActiveFilters,
56
- searchQuery = '',
57
- searchMode = 'filter',
58
- searchIsStale,
59
- onSearchChange,
60
- onSearchModeChange,
61
- }: ScopeFilterBarProps) {
62
- const { neonGlass } = useTheme();
63
-
64
- // Group active values by field
65
- const activeFields: { field: FilterField; values: { value: string; label: string }[] }[] = [];
66
- for (const { field } of CHIP_CONFIG) {
67
- if (filters[field].size === 0) continue;
68
- const values: { value: string; label: string }[] = [];
69
- for (const value of filters[field]) {
70
- const opt = optionsWithCounts[field].find((o) => o.value === value);
71
- values.push({ value, label: opt?.label ?? value });
72
- }
73
- activeFields.push({ field, values });
74
- }
75
-
76
- return (
77
- <div className="mb-4 space-y-2">
78
- {/* Chip row */}
79
- <div className="flex flex-wrap items-center gap-2">
80
- {CHIP_CONFIG.map(({ field, label, glowClass }) => (
81
- <FilterChip
82
- key={field}
83
- field={field}
84
- label={label}
85
- options={optionsWithCounts[field]}
86
- selected={filters[field]}
87
- onToggle={onToggle}
88
- glowClass={glowClass}
89
- />
90
- ))}
91
-
92
- {hasActiveFilters && (
93
- <button
94
- onClick={onClearAll}
95
- className="ml-2 text-xxs text-muted-foreground hover:text-foreground transition-colors"
96
- >
97
- Clear all
98
- </button>
99
- )}
100
-
101
- {onSearchChange && onSearchModeChange && (
102
- <SearchInput
103
- query={searchQuery}
104
- mode={searchMode}
105
- isStale={searchIsStale}
106
- onQueryChange={onSearchChange}
107
- onModeChange={onSearchModeChange}
108
- />
109
- )}
110
- </div>
111
-
112
- {/* Active filter summary — field label + individual value badges + clear-field */}
113
- {activeFields.length > 0 && (
114
- <div className="flex flex-wrap items-center gap-2">
115
- {activeFields.map(({ field, values }) => (
116
- <div key={field} className="flex items-center gap-1">
117
- <span className="text-[10px] text-muted-foreground capitalize">{FIELD_LABEL[field]}:</span>
118
- {values.map(({ value, label }) => {
119
- const isCat = field === 'category';
120
- return (
121
- <Badge
122
- key={value}
123
- variant={isCat ? 'outline' : 'secondary'}
124
- className={cn(
125
- 'gap-0.5 capitalize cursor-pointer hover:bg-secondary/60 py-0 px-1 text-[10px] font-light',
126
- neonGlass && !isCat && 'glass-pill',
127
- neonGlass && isCat && 'bg-[rgba(var(--neon-blue),0.08)]'
128
- )}
129
- style={isCat ? { borderColor: CATEGORY_COLOR[value] } : undefined}
130
- onClick={() => onToggle(field, value)}
131
- >
132
- {label}
133
- <X className="h-2 w-2 opacity-60" />
134
- </Badge>
135
- );
136
- })}
137
- {values.length > 1 && (
138
- <button
139
- onClick={() => onClearField(field)}
140
- className="rounded p-0.5 text-muted-foreground hover:bg-surface-light hover:text-foreground"
141
- aria-label={`Clear all ${FIELD_LABEL[field]} filters`}
142
- >
143
- <X className="h-2.5 w-2.5" />
144
- </button>
145
- )}
146
- </div>
147
- ))}
148
- </div>
149
- )}
150
- </div>
151
- );
152
- }