@pellux/goodvibes-agent 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (398) hide show
  1. package/.goodvibes/GOODVIBES.md +35 -0
  2. package/.goodvibes/agents/reviewer.md +48 -0
  3. package/.goodvibes/skills/add-provider/SKILL.md +199 -0
  4. package/CHANGELOG.md +25 -0
  5. package/README.md +74 -0
  6. package/bin/goodvibes-agent.ts +2 -0
  7. package/docs/README.md +23 -0
  8. package/docs/deployment-and-services.md +57 -0
  9. package/docs/getting-started.md +53 -0
  10. package/docs/release-and-publishing.md +46 -0
  11. package/package.json +134 -0
  12. package/scripts/check-bun.sh +20 -0
  13. package/src/audio/player.ts +156 -0
  14. package/src/audio/spoken-turn-controller.ts +203 -0
  15. package/src/audio/spoken-turn-model-routing.ts +117 -0
  16. package/src/audio/spoken-turn-wiring.ts +44 -0
  17. package/src/audio/text-chunker.ts +110 -0
  18. package/src/cli/bundle-command.ts +227 -0
  19. package/src/cli/completion.ts +90 -0
  20. package/src/cli/config-overrides.ts +159 -0
  21. package/src/cli/endpoints.ts +63 -0
  22. package/src/cli/entrypoint.ts +172 -0
  23. package/src/cli/help.ts +299 -0
  24. package/src/cli/index.ts +11 -0
  25. package/src/cli/management-commands.ts +426 -0
  26. package/src/cli/management.ts +744 -0
  27. package/src/cli/network-posture.ts +46 -0
  28. package/src/cli/package-verification.ts +123 -0
  29. package/src/cli/parser.ts +369 -0
  30. package/src/cli/provider-auth-routes.ts +22 -0
  31. package/src/cli/provider-classification.ts +107 -0
  32. package/src/cli/redaction.ts +105 -0
  33. package/src/cli/service-command.ts +26 -0
  34. package/src/cli/service-posture.ts +482 -0
  35. package/src/cli/status.ts +383 -0
  36. package/src/cli/surface-command.ts +247 -0
  37. package/src/cli/tui-startup.ts +32 -0
  38. package/src/cli/types.ts +69 -0
  39. package/src/cli-flags.ts +21 -0
  40. package/src/config/goodvibes-home-audit.ts +465 -0
  41. package/src/config/index.ts +57 -0
  42. package/src/config/provider-model.ts +23 -0
  43. package/src/config/secret-config.ts +119 -0
  44. package/src/config/secrets.ts +71 -0
  45. package/src/config/surface.ts +1 -0
  46. package/src/core/composer-state.ts +61 -0
  47. package/src/core/conversation-rendering.ts +359 -0
  48. package/src/core/conversation.ts +551 -0
  49. package/src/core/history.ts +45 -0
  50. package/src/core/orchestrator.ts +7 -0
  51. package/src/core/system-message-router.ts +171 -0
  52. package/src/daemon/cli.ts +55 -0
  53. package/src/daemon/safe-serve.ts +61 -0
  54. package/src/input/agent-workspace.ts +428 -0
  55. package/src/input/autocomplete.ts +96 -0
  56. package/src/input/bookmark-modal.ts +115 -0
  57. package/src/input/command-args-hint.ts +36 -0
  58. package/src/input/command-registry.ts +329 -0
  59. package/src/input/commands/agent-externalized-tui.ts +73 -0
  60. package/src/input/commands/agent-workspace-runtime.ts +17 -0
  61. package/src/input/commands/branch-runtime.ts +72 -0
  62. package/src/input/commands/cloudflare-runtime.ts +370 -0
  63. package/src/input/commands/config.ts +18 -0
  64. package/src/input/commands/control-room-runtime.ts +255 -0
  65. package/src/input/commands/conversation-runtime.ts +207 -0
  66. package/src/input/commands/discovery-runtime.ts +52 -0
  67. package/src/input/commands/eval.ts +204 -0
  68. package/src/input/commands/experience-runtime.ts +278 -0
  69. package/src/input/commands/guidance-runtime.ts +106 -0
  70. package/src/input/commands/health-runtime.ts +434 -0
  71. package/src/input/commands/hooks-runtime.ts +148 -0
  72. package/src/input/commands/incident-runtime.ts +95 -0
  73. package/src/input/commands/integration-runtime.ts +394 -0
  74. package/src/input/commands/intelligence-runtime.ts +223 -0
  75. package/src/input/commands/knowledge.ts +531 -0
  76. package/src/input/commands/local-auth-runtime.ts +105 -0
  77. package/src/input/commands/local-provider-runtime.ts +170 -0
  78. package/src/input/commands/local-runtime.ts +392 -0
  79. package/src/input/commands/local-setup-review.ts +199 -0
  80. package/src/input/commands/local-setup-transfer.ts +135 -0
  81. package/src/input/commands/local-setup.ts +282 -0
  82. package/src/input/commands/managed-runtime.ts +209 -0
  83. package/src/input/commands/marketplace-runtime.ts +290 -0
  84. package/src/input/commands/mcp-runtime.ts +432 -0
  85. package/src/input/commands/memory-product-runtime.ts +111 -0
  86. package/src/input/commands/memory.ts +151 -0
  87. package/src/input/commands/notify-runtime.ts +83 -0
  88. package/src/input/commands/onboarding-runtime.ts +14 -0
  89. package/src/input/commands/operator-panel-runtime.ts +146 -0
  90. package/src/input/commands/operator-runtime.ts +392 -0
  91. package/src/input/commands/planning-runtime.ts +205 -0
  92. package/src/input/commands/platform-access-runtime.ts +422 -0
  93. package/src/input/commands/platform-services-runtime.ts +246 -0
  94. package/src/input/commands/policy-dispatch.ts +339 -0
  95. package/src/input/commands/policy.ts +17 -0
  96. package/src/input/commands/product-runtime.ts +351 -0
  97. package/src/input/commands/profile-sync-runtime.ts +99 -0
  98. package/src/input/commands/provider-accounts-runtime.ts +113 -0
  99. package/src/input/commands/provider.ts +363 -0
  100. package/src/input/commands/qrcode-runtime.ts +20 -0
  101. package/src/input/commands/quit-shared.ts +162 -0
  102. package/src/input/commands/recall-bundle.ts +132 -0
  103. package/src/input/commands/recall-capture.ts +152 -0
  104. package/src/input/commands/recall-query.ts +229 -0
  105. package/src/input/commands/recall-review.ts +98 -0
  106. package/src/input/commands/recall-shared.ts +22 -0
  107. package/src/input/commands/remote-runtime-pool.ts +106 -0
  108. package/src/input/commands/remote-runtime-setup.ts +199 -0
  109. package/src/input/commands/remote-runtime.ts +431 -0
  110. package/src/input/commands/replay-runtime.ts +18 -0
  111. package/src/input/commands/runtime-services.ts +291 -0
  112. package/src/input/commands/schedule-runtime.ts +91 -0
  113. package/src/input/commands/services-runtime.ts +209 -0
  114. package/src/input/commands/session-content.ts +408 -0
  115. package/src/input/commands/session-workflow.ts +464 -0
  116. package/src/input/commands/session.ts +375 -0
  117. package/src/input/commands/settings-sync-runtime.ts +174 -0
  118. package/src/input/commands/share-runtime.ts +119 -0
  119. package/src/input/commands/shell-core.ts +307 -0
  120. package/src/input/commands/skills-runtime.ts +221 -0
  121. package/src/input/commands/subscription-runtime.ts +434 -0
  122. package/src/input/commands/tasks-runtime.ts +230 -0
  123. package/src/input/commands/teamwork-runtime.ts +339 -0
  124. package/src/input/commands/teleport-runtime.ts +57 -0
  125. package/src/input/commands/tts-runtime.ts +29 -0
  126. package/src/input/commands/work-plan-runtime.ts +169 -0
  127. package/src/input/commands.ts +131 -0
  128. package/src/input/feed-context-factory.ts +254 -0
  129. package/src/input/file-picker.ts +192 -0
  130. package/src/input/handler-command-route.ts +180 -0
  131. package/src/input/handler-content-actions.ts +497 -0
  132. package/src/input/handler-feed-routes.ts +648 -0
  133. package/src/input/handler-feed.ts +452 -0
  134. package/src/input/handler-interactions.ts +281 -0
  135. package/src/input/handler-modal-routes.ts +418 -0
  136. package/src/input/handler-modal-stack.ts +263 -0
  137. package/src/input/handler-modal-token-routes.ts +329 -0
  138. package/src/input/handler-onboarding-cloudflare.ts +391 -0
  139. package/src/input/handler-onboarding.ts +620 -0
  140. package/src/input/handler-picker-routes.ts +472 -0
  141. package/src/input/handler-prompt-buffer.ts +320 -0
  142. package/src/input/handler-shortcuts.ts +213 -0
  143. package/src/input/handler-ui-state.ts +372 -0
  144. package/src/input/handler.ts +729 -0
  145. package/src/input/input-history.ts +297 -0
  146. package/src/input/keybindings.ts +292 -0
  147. package/src/input/mcp-workspace.ts +554 -0
  148. package/src/input/model-picker-provider-filter.ts +28 -0
  149. package/src/input/model-picker-types.ts +137 -0
  150. package/src/input/model-picker.ts +797 -0
  151. package/src/input/onboarding/handler-onboarding-routes.ts +125 -0
  152. package/src/input/onboarding/onboarding-runtime-status.ts +87 -0
  153. package/src/input/onboarding/onboarding-wizard-apply.ts +277 -0
  154. package/src/input/onboarding/onboarding-wizard-cloudflare-step.ts +494 -0
  155. package/src/input/onboarding/onboarding-wizard-cloudflare.ts +204 -0
  156. package/src/input/onboarding/onboarding-wizard-constants.ts +158 -0
  157. package/src/input/onboarding/onboarding-wizard-external-surface-extra-specs.ts +130 -0
  158. package/src/input/onboarding/onboarding-wizard-external-surfaces.ts +762 -0
  159. package/src/input/onboarding/onboarding-wizard-helpers.ts +167 -0
  160. package/src/input/onboarding/onboarding-wizard-rules.ts +256 -0
  161. package/src/input/onboarding/onboarding-wizard-state.ts +365 -0
  162. package/src/input/onboarding/onboarding-wizard-steps.ts +798 -0
  163. package/src/input/onboarding/onboarding-wizard-types.ts +195 -0
  164. package/src/input/onboarding/onboarding-wizard.ts +711 -0
  165. package/src/input/panel-integration-actions.ts +78 -0
  166. package/src/input/profile-picker-modal.ts +222 -0
  167. package/src/input/search.ts +100 -0
  168. package/src/input/selection-modal.ts +163 -0
  169. package/src/input/selection.ts +135 -0
  170. package/src/input/session-picker-modal.ts +136 -0
  171. package/src/input/settings-modal-behavior.ts +37 -0
  172. package/src/input/settings-modal-secrets.ts +41 -0
  173. package/src/input/settings-modal-subscriptions.ts +95 -0
  174. package/src/input/settings-modal-types.ts +91 -0
  175. package/src/input/settings-modal.ts +793 -0
  176. package/src/input/submission-intent.ts +17 -0
  177. package/src/input/submission-router.ts +59 -0
  178. package/src/input/tts-settings-actions.ts +100 -0
  179. package/src/main.ts +792 -0
  180. package/src/mcp/runtime-reload.ts +81 -0
  181. package/src/panels/agent-inspector-panel.ts +521 -0
  182. package/src/panels/agent-inspector-shared.ts +94 -0
  183. package/src/panels/agent-logs-panel.ts +559 -0
  184. package/src/panels/agent-logs-shared.ts +129 -0
  185. package/src/panels/approval-panel.ts +150 -0
  186. package/src/panels/automation-control-panel.ts +212 -0
  187. package/src/panels/base-panel.ts +254 -0
  188. package/src/panels/builtin/agent.ts +117 -0
  189. package/src/panels/builtin/development.ts +31 -0
  190. package/src/panels/builtin/knowledge.ts +26 -0
  191. package/src/panels/builtin/operations.ts +349 -0
  192. package/src/panels/builtin/session.ts +129 -0
  193. package/src/panels/builtin/shared.ts +274 -0
  194. package/src/panels/builtin-panels.ts +23 -0
  195. package/src/panels/cockpit-panel.ts +183 -0
  196. package/src/panels/communication-panel.ts +153 -0
  197. package/src/panels/confirm-state.ts +61 -0
  198. package/src/panels/context-visualizer-panel.ts +204 -0
  199. package/src/panels/control-plane-panel.ts +211 -0
  200. package/src/panels/cost-tracker-panel.ts +444 -0
  201. package/src/panels/debug-panel.ts +432 -0
  202. package/src/panels/diff-panel.ts +520 -0
  203. package/src/panels/docs-panel.ts +283 -0
  204. package/src/panels/eval-panel.ts +399 -0
  205. package/src/panels/file-explorer-panel.ts +584 -0
  206. package/src/panels/file-preview-panel.ts +434 -0
  207. package/src/panels/forensics-panel.ts +364 -0
  208. package/src/panels/git-panel.ts +638 -0
  209. package/src/panels/hooks-panel.ts +239 -0
  210. package/src/panels/incident-review-panel.ts +197 -0
  211. package/src/panels/index.ts +46 -0
  212. package/src/panels/intelligence-panel.ts +176 -0
  213. package/src/panels/knowledge-panel.ts +345 -0
  214. package/src/panels/local-auth-panel.ts +130 -0
  215. package/src/panels/marketplace-panel.ts +212 -0
  216. package/src/panels/memory-panel.ts +225 -0
  217. package/src/panels/ops-control-panel.ts +150 -0
  218. package/src/panels/ops-strategy-panel.ts +235 -0
  219. package/src/panels/orchestration-panel.ts +273 -0
  220. package/src/panels/panel-list-panel.ts +509 -0
  221. package/src/panels/panel-manager.ts +570 -0
  222. package/src/panels/panel-picker.ts +106 -0
  223. package/src/panels/plan-dashboard-panel.ts +274 -0
  224. package/src/panels/plugins-panel.ts +178 -0
  225. package/src/panels/policy-panel.ts +308 -0
  226. package/src/panels/polish.ts +717 -0
  227. package/src/panels/project-planning-panel.ts +711 -0
  228. package/src/panels/provider-account-snapshot.ts +259 -0
  229. package/src/panels/provider-accounts-panel.ts +218 -0
  230. package/src/panels/provider-health-domains.ts +215 -0
  231. package/src/panels/provider-health-panel.ts +727 -0
  232. package/src/panels/provider-health-tracker.ts +115 -0
  233. package/src/panels/provider-stats-panel.ts +366 -0
  234. package/src/panels/qr-panel.ts +182 -0
  235. package/src/panels/remote-panel.ts +449 -0
  236. package/src/panels/routes-panel.ts +178 -0
  237. package/src/panels/sandbox-panel.ts +283 -0
  238. package/src/panels/schedule-panel.ts +329 -0
  239. package/src/panels/scrollable-list-panel.ts +491 -0
  240. package/src/panels/search-focus.ts +32 -0
  241. package/src/panels/security-panel.ts +295 -0
  242. package/src/panels/services-panel.ts +231 -0
  243. package/src/panels/session-browser-panel.ts +400 -0
  244. package/src/panels/session-maintenance.ts +125 -0
  245. package/src/panels/settings-sync-panel.ts +120 -0
  246. package/src/panels/skills-panel.ts +431 -0
  247. package/src/panels/subscription-panel.ts +263 -0
  248. package/src/panels/symbol-outline-panel.ts +486 -0
  249. package/src/panels/system-messages-panel.ts +230 -0
  250. package/src/panels/tasks-panel.ts +399 -0
  251. package/src/panels/thinking-panel.ts +304 -0
  252. package/src/panels/token-budget-panel.ts +475 -0
  253. package/src/panels/tool-inspector-panel.ts +429 -0
  254. package/src/panels/types.ts +54 -0
  255. package/src/panels/watchers-panel.ts +193 -0
  256. package/src/panels/work-plan-panel.ts +175 -0
  257. package/src/panels/worktree-panel.ts +182 -0
  258. package/src/panels/wrfc-panel.ts +609 -0
  259. package/src/permissions/prompt.ts +165 -0
  260. package/src/planning/project-planning-coordinator.ts +543 -0
  261. package/src/plugins/loader.ts +15 -0
  262. package/src/renderer/agent-detail-modal.ts +331 -0
  263. package/src/renderer/agent-workspace.ts +238 -0
  264. package/src/renderer/ansi-sanitize.ts +76 -0
  265. package/src/renderer/autocomplete-overlay.ts +154 -0
  266. package/src/renderer/block-actions.ts +76 -0
  267. package/src/renderer/bookmark-modal.ts +101 -0
  268. package/src/renderer/bottom-bar.ts +58 -0
  269. package/src/renderer/buffer.ts +113 -0
  270. package/src/renderer/code-block.ts +373 -0
  271. package/src/renderer/compositor.ts +283 -0
  272. package/src/renderer/context-inspector.ts +219 -0
  273. package/src/renderer/conversation-layout.ts +67 -0
  274. package/src/renderer/conversation-overlays.ts +140 -0
  275. package/src/renderer/conversation-surface.ts +260 -0
  276. package/src/renderer/diff-view.ts +132 -0
  277. package/src/renderer/diff.ts +130 -0
  278. package/src/renderer/file-picker-overlay.ts +101 -0
  279. package/src/renderer/file-tree.ts +153 -0
  280. package/src/renderer/fullscreen-primitives.ts +130 -0
  281. package/src/renderer/fullscreen-workspace.ts +199 -0
  282. package/src/renderer/git-status.ts +89 -0
  283. package/src/renderer/help-overlay.ts +267 -0
  284. package/src/renderer/history-search-overlay.ts +73 -0
  285. package/src/renderer/layout-engine.ts +97 -0
  286. package/src/renderer/layout.ts +32 -0
  287. package/src/renderer/live-tail-modal.ts +156 -0
  288. package/src/renderer/markdown.ts +635 -0
  289. package/src/renderer/mcp-workspace.ts +237 -0
  290. package/src/renderer/modal-factory.ts +467 -0
  291. package/src/renderer/modal-utils.ts +24 -0
  292. package/src/renderer/model-picker-overlay.ts +473 -0
  293. package/src/renderer/model-workspace.ts +488 -0
  294. package/src/renderer/onboarding/onboarding-wizard.ts +615 -0
  295. package/src/renderer/overlay-box.ts +146 -0
  296. package/src/renderer/overlay-viewport.ts +104 -0
  297. package/src/renderer/panel-composite.ts +158 -0
  298. package/src/renderer/panel-picker-overlay.ts +202 -0
  299. package/src/renderer/panel-tab-bar.ts +69 -0
  300. package/src/renderer/panel-workspace-bar.ts +42 -0
  301. package/src/renderer/process-indicator.ts +96 -0
  302. package/src/renderer/process-modal.ts +656 -0
  303. package/src/renderer/process-summary.ts +67 -0
  304. package/src/renderer/profile-picker-modal.ts +129 -0
  305. package/src/renderer/progress.ts +98 -0
  306. package/src/renderer/qr-renderer.ts +120 -0
  307. package/src/renderer/search-overlay.ts +54 -0
  308. package/src/renderer/selection-modal-overlay.ts +214 -0
  309. package/src/renderer/semantic-diff.ts +369 -0
  310. package/src/renderer/session-picker-modal.ts +127 -0
  311. package/src/renderer/settings-modal-helpers.ts +193 -0
  312. package/src/renderer/settings-modal.ts +537 -0
  313. package/src/renderer/shell-surface.ts +88 -0
  314. package/src/renderer/status-glyphs.ts +21 -0
  315. package/src/renderer/status-token.ts +67 -0
  316. package/src/renderer/surface-layout.ts +101 -0
  317. package/src/renderer/syntax-highlighter.ts +542 -0
  318. package/src/renderer/system-message.ts +83 -0
  319. package/src/renderer/tab-strip.ts +108 -0
  320. package/src/renderer/text-layout.ts +31 -0
  321. package/src/renderer/thinking.ts +17 -0
  322. package/src/renderer/tool-call.ts +234 -0
  323. package/src/renderer/ui-factory.ts +524 -0
  324. package/src/renderer/ui-primitives.ts +96 -0
  325. package/src/runtime/bootstrap-command-context.ts +278 -0
  326. package/src/runtime/bootstrap-command-parts.ts +386 -0
  327. package/src/runtime/bootstrap-core.ts +540 -0
  328. package/src/runtime/bootstrap-hook-bridge.ts +112 -0
  329. package/src/runtime/bootstrap-shell.ts +283 -0
  330. package/src/runtime/bootstrap.ts +575 -0
  331. package/src/runtime/cloudflare-control-plane.ts +349 -0
  332. package/src/runtime/context.ts +142 -0
  333. package/src/runtime/diagnostics/panels/index.ts +24 -0
  334. package/src/runtime/diagnostics/panels/ops.ts +156 -0
  335. package/src/runtime/diagnostics/panels/panel-resources.ts +118 -0
  336. package/src/runtime/diagnostics/panels/policy.ts +177 -0
  337. package/src/runtime/index.ts +662 -0
  338. package/src/runtime/onboarding/apply.ts +642 -0
  339. package/src/runtime/onboarding/derivation.ts +534 -0
  340. package/src/runtime/onboarding/index.ts +7 -0
  341. package/src/runtime/onboarding/markers.ts +148 -0
  342. package/src/runtime/onboarding/snapshot.ts +406 -0
  343. package/src/runtime/onboarding/state.ts +141 -0
  344. package/src/runtime/onboarding/types.ts +404 -0
  345. package/src/runtime/onboarding/verify.ts +171 -0
  346. package/src/runtime/operator-token-cleanup.ts +27 -0
  347. package/src/runtime/perf/panel-contracts.ts +32 -0
  348. package/src/runtime/perf/panel-health-monitor.ts +18 -0
  349. package/src/runtime/sandbox-public-gaps.ts +358 -0
  350. package/src/runtime/services.ts +670 -0
  351. package/src/runtime/store/domains/domain-read-matrix.ts +15 -0
  352. package/src/runtime/store/domains/index.ts +222 -0
  353. package/src/runtime/store/domains/panels.ts +117 -0
  354. package/src/runtime/store/domains/ui-perf.ts +103 -0
  355. package/src/runtime/store/index.ts +305 -0
  356. package/src/runtime/store/selectors/index.ts +359 -0
  357. package/src/runtime/store/state.ts +145 -0
  358. package/src/runtime/surface-feature-flags.ts +65 -0
  359. package/src/runtime/terminal-output-guard.ts +228 -0
  360. package/src/runtime/ui/index.ts +39 -0
  361. package/src/runtime/ui/model-picker/data-provider.ts +182 -0
  362. package/src/runtime/ui/model-picker/health-enrichment.ts +228 -0
  363. package/src/runtime/ui/model-picker/index.ts +59 -0
  364. package/src/runtime/ui/model-picker/types.ts +149 -0
  365. package/src/runtime/ui/provider-health/data-provider.ts +244 -0
  366. package/src/runtime/ui/provider-health/fallback-visualizer.ts +71 -0
  367. package/src/runtime/ui/provider-health/index.ts +46 -0
  368. package/src/runtime/ui/provider-health/types.ts +146 -0
  369. package/src/runtime/ui-events.ts +1 -0
  370. package/src/runtime/ui-read-model-helpers.ts +1 -0
  371. package/src/runtime/ui-read-models-observability-maintenance.ts +1 -0
  372. package/src/runtime/ui-read-models-observability-options.ts +1 -0
  373. package/src/runtime/ui-read-models-observability-remote.ts +1 -0
  374. package/src/runtime/ui-read-models-observability-security.ts +1 -0
  375. package/src/runtime/ui-read-models-observability-system.ts +1 -0
  376. package/src/runtime/ui-read-models-observability.ts +1 -0
  377. package/src/runtime/ui-read-models.ts +61 -0
  378. package/src/runtime/ui-service-queries.ts +1 -0
  379. package/src/runtime/ui-services.ts +190 -0
  380. package/src/scripts/process-messages.ts +42 -0
  381. package/src/shell/blocking-input.ts +98 -0
  382. package/src/shell/service-settings-sync.ts +273 -0
  383. package/src/shell/ui-openers.ts +352 -0
  384. package/src/tools/index.ts +1 -0
  385. package/src/tools/wrfc-agent-guard.ts +49 -0
  386. package/src/types/grid.ts +48 -0
  387. package/src/types/sql-js.d.ts +15 -0
  388. package/src/utils/clipboard.ts +22 -0
  389. package/src/utils/splash-lines.ts +46 -0
  390. package/src/utils/terminal-width.ts +185 -0
  391. package/src/verification/live-verifier.ts +430 -0
  392. package/src/verification/verification-ledger.ts +242 -0
  393. package/src/version.ts +17 -0
  394. package/src/widget/index.ts +2 -0
  395. package/src/widget/types.ts +9 -0
  396. package/src/widget/widget.ts +8 -0
  397. package/src/work-plans/work-plan-store.ts +374 -0
  398. package/tsconfig.json +18 -0
@@ -0,0 +1,638 @@
1
+ import { BasePanel } from './base-panel.ts';
2
+ import { createEmptyLine, createStyledCell, type Line } from '../types/grid.ts';
3
+ import { GitService } from '@pellux/goodvibes-sdk/platform/git';
4
+ import { logger } from '@pellux/goodvibes-sdk/platform/utils';
5
+ import { summarizeError } from '@pellux/goodvibes-sdk/platform/utils';
6
+ import {
7
+ buildEmptyState,
8
+ buildPanelLine,
9
+ buildPanelWorkspace,
10
+ resolveScrollablePanelSection,
11
+ buildStyledPanelLine,
12
+ DEFAULT_PANEL_PALETTE,
13
+ } from './polish.ts';
14
+
15
+ // ---------------------------------------------------------------------------
16
+ // Types
17
+ // ---------------------------------------------------------------------------
18
+
19
+ interface GitFileEntry {
20
+ path: string;
21
+ staged: boolean;
22
+ }
23
+
24
+ interface CommitEntry {
25
+ hash: string;
26
+ message: string;
27
+ author: string;
28
+ date: string;
29
+ }
30
+
31
+ interface GitData {
32
+ branch: string;
33
+ ahead: number;
34
+ behind: number;
35
+ stagedFiles: GitFileEntry[];
36
+ unstagedFiles: GitFileEntry[];
37
+ recentCommits: CommitEntry[];
38
+ }
39
+
40
+ type ViewItem =
41
+ | { kind: 'header' }
42
+ | { kind: 'section'; label: string }
43
+ | { kind: 'file'; entry: GitFileEntry }
44
+ | { kind: 'commit'; entry: CommitEntry }
45
+ | { kind: 'empty'; label: string }
46
+ | { kind: 'diff-line'; text: string; diffType: 'add' | 'remove' | 'meta' | 'neutral' };
47
+
48
+ // ---------------------------------------------------------------------------
49
+ // Constants
50
+ // ---------------------------------------------------------------------------
51
+
52
+ /** Minimum number of diff lines kept visible when clamping scroll offset. */
53
+ const MIN_VISIBLE_DIFF_LINES = 5;
54
+
55
+ // ---------------------------------------------------------------------------
56
+ // Colors
57
+ // ---------------------------------------------------------------------------
58
+
59
+ const C = {
60
+ branch: '#00d7ff',
61
+ clean: '#5fd700',
62
+ dirty: '#ffaf00',
63
+ ahead: '#5fd700',
64
+ behind: '#ff5f5f',
65
+ sectionHeader: '244',
66
+ staged: '#5fd700',
67
+ unstaged: '#ff5f5f',
68
+ commit: '250',
69
+ commitHash: '238',
70
+ commitAuthor: '244',
71
+ selected: '#1c1c1c',
72
+ selectedFg: '#ffffff',
73
+ diffAdd: '#5fd700',
74
+ diffRemove: '#ff5f5f',
75
+ diffMeta: '#5f87ff',
76
+ diffNeutral: '250',
77
+ } as const;
78
+
79
+ // ---------------------------------------------------------------------------
80
+ // GitPanel
81
+ // ---------------------------------------------------------------------------
82
+
83
+ export class GitPanel extends BasePanel {
84
+ private readonly workingDirectory: string;
85
+ private data: GitData = {
86
+ branch: '...',
87
+ ahead: 0,
88
+ behind: 0,
89
+ stagedFiles: [],
90
+ unstagedFiles: [],
91
+ recentCommits: [],
92
+ };
93
+
94
+ /** Flattened list of navigable rows (for arrow-key movement). */
95
+ private items: ViewItem[] = [];
96
+
97
+ /** Selected row index within `items`. */
98
+ private selectedIndex = 0;
99
+
100
+ /** When truthy, shows the diff for the selected file. */
101
+ private expandedDiff: string[] | null = null;
102
+
103
+ /** Scroll offset for both main view and diff view. */
104
+ private scrollOffset = 0;
105
+
106
+ private refreshTimerId: ReturnType<typeof setInterval> | null = null;
107
+ private loading = true;
108
+ private error: string | null = null;
109
+
110
+ constructor(workingDirectory: string) {
111
+ super('git', 'Git', 'G', 'development');
112
+ this.workingDirectory = workingDirectory;
113
+ }
114
+
115
+ // ---------------------------------------------------------------------------
116
+ // Lifecycle
117
+ // ---------------------------------------------------------------------------
118
+
119
+ override onActivate(): void {
120
+ super.onActivate();
121
+ void this.refresh();
122
+ this.refreshTimerId = this.registerTimer(setInterval(() => {
123
+ void this.refresh();
124
+ }, 5_000));
125
+ }
126
+
127
+ override onDeactivate(): void {
128
+ if (this.refreshTimerId !== null) {
129
+ this.clearTimer(this.refreshTimerId);
130
+ this.refreshTimerId = null;
131
+ }
132
+ }
133
+
134
+ override onDestroy(): void {
135
+ this.onDeactivate();
136
+ super.onDestroy();
137
+ }
138
+
139
+ // ---------------------------------------------------------------------------
140
+ // Data fetching
141
+ // ---------------------------------------------------------------------------
142
+
143
+ private async refresh(isRetry = false): Promise<void> {
144
+ try {
145
+ const git = new GitService(this.workingDirectory);
146
+ const [statusResult, branchResult, logEntries] = await Promise.all([
147
+ git.status(),
148
+ git.branch(),
149
+ git.log(10),
150
+ ]);
151
+
152
+ const stagedFiles: GitFileEntry[] = [
153
+ ...statusResult.staged.map((p) => ({ path: p, staged: true })),
154
+ ...statusResult.created.map((p) => ({ path: p, staged: true })),
155
+ ];
156
+
157
+ const unstagedFiles: GitFileEntry[] = [
158
+ ...statusResult.modified.map((p) => ({ path: p, staged: false })),
159
+ ...statusResult.deleted.map((p) => ({ path: p, staged: false })),
160
+ ...statusResult.not_added.map((p) => ({ path: p, staged: false })),
161
+ ...statusResult.conflicted.map((p) => ({ path: p, staged: false })),
162
+ ];
163
+
164
+ this.data = {
165
+ branch: branchResult.current || 'HEAD',
166
+ ahead: statusResult.ahead ?? 0,
167
+ behind: statusResult.behind ?? 0,
168
+ stagedFiles,
169
+ unstagedFiles,
170
+ recentCommits: logEntries.map((e) => ({
171
+ hash: e.hash.slice(0, 7),
172
+ message: e.message,
173
+ author: e.author,
174
+ date: e.date,
175
+ })),
176
+ };
177
+
178
+ this.loading = false;
179
+ this.error = null;
180
+ this.rebuildItems();
181
+ // Do not clear expandedDiff during auto-refresh — only clear on explicit user action
182
+ this.markDirty();
183
+ } catch (err) {
184
+ const msg = summarizeError(err);
185
+ // If the failure is because this directory isn't a git repo, auto-initialise
186
+ // and retry once so the panel becomes functional immediately.
187
+ if (/not a git\b/i.test(msg)) {
188
+ const cwd = this.workingDirectory;
189
+ const initResult = GitService.initRepo(cwd);
190
+ if (initResult.success) {
191
+ logger.debug('GitPanel: auto-initialised git repo', { cwd });
192
+ if (!isRetry) {
193
+ // Retry refresh now that the repo exists (once only)
194
+ void this.refresh(true);
195
+ return;
196
+ }
197
+ this.error = 'Not a git repository. Auto-init succeeded but refresh failed.';
198
+ } else {
199
+ this.error = `Not a git repository. Auto-init failed: ${initResult.error ?? 'unknown error'}`;
200
+ }
201
+ } else {
202
+ this.error = msg;
203
+ }
204
+ this.loading = false;
205
+ logger.debug('GitPanel: refresh failed', { error: this.error });
206
+ this.markDirty();
207
+ }
208
+ }
209
+
210
+ /** Rebuild the flat navigable item list from current data. */
211
+ private rebuildItems(): void {
212
+ const items: ViewItem[] = [];
213
+
214
+ // Branch / status header row
215
+ items.push({ kind: 'header' });
216
+
217
+ // Staged files
218
+ items.push({ kind: 'section', label: `Staged (${this.data.stagedFiles.length})` });
219
+ if (this.data.stagedFiles.length === 0) {
220
+ items.push({ kind: 'empty', label: ' (no staged files)' });
221
+ } else {
222
+ for (const entry of this.data.stagedFiles) {
223
+ items.push({ kind: 'file', entry });
224
+ }
225
+ }
226
+
227
+ // Unstaged files
228
+ items.push({ kind: 'section', label: `Unstaged (${this.data.unstagedFiles.length})` });
229
+ if (this.data.unstagedFiles.length === 0) {
230
+ items.push({ kind: 'empty', label: ' (no unstaged files)' });
231
+ } else {
232
+ for (const entry of this.data.unstagedFiles) {
233
+ items.push({ kind: 'file', entry });
234
+ }
235
+ }
236
+
237
+ // Recent commits
238
+ items.push({ kind: 'section', label: `Recent Commits (${this.data.recentCommits.length})` });
239
+ if (this.data.recentCommits.length === 0) {
240
+ items.push({ kind: 'empty', label: ' (no commits)' });
241
+ } else {
242
+ for (const entry of this.data.recentCommits) {
243
+ items.push({ kind: 'commit', entry });
244
+ }
245
+ }
246
+
247
+ this.items = items;
248
+
249
+ // Keep selection in bounds
250
+ if (this.selectedIndex >= this.items.length) {
251
+ this.selectedIndex = Math.max(0, this.items.length - 1);
252
+ }
253
+ }
254
+
255
+ // ---------------------------------------------------------------------------
256
+ // Input handling
257
+ // ---------------------------------------------------------------------------
258
+
259
+ handleInput(key: string): boolean {
260
+ if (this.expandedDiff !== null) {
261
+ return this.handleDiffInput(key);
262
+ }
263
+ return this.handleListInput(key);
264
+ }
265
+
266
+ private handleListInput(key: string): boolean {
267
+ switch (key) {
268
+ case 'up':
269
+ case 'k': {
270
+ if (this.selectedIndex > 0) {
271
+ this.selectedIndex--;
272
+ this.markDirty();
273
+ }
274
+ return true;
275
+ }
276
+ case 'down':
277
+ case 'j': {
278
+ if (this.selectedIndex < this.items.length - 1) {
279
+ this.selectedIndex++;
280
+ this.markDirty();
281
+ }
282
+ return true;
283
+ }
284
+ case 'return': {
285
+ void this.openDiff();
286
+ return true;
287
+ }
288
+ case 'r': {
289
+ void this.refresh();
290
+ return true;
291
+ }
292
+ default:
293
+ return false;
294
+ }
295
+ }
296
+
297
+ private handleDiffInput(key: string): boolean {
298
+ switch (key) {
299
+ case 'up':
300
+ case 'k': {
301
+ if (this.scrollOffset > 0) {
302
+ this.scrollOffset--;
303
+ this.markDirty();
304
+ }
305
+ return true;
306
+ }
307
+ case 'down':
308
+ case 'j': {
309
+ const diffLen = this.expandedDiff?.length ?? 0;
310
+ this.scrollOffset = Math.min(this.scrollOffset + 1, Math.max(0, diffLen - MIN_VISIBLE_DIFF_LINES));
311
+ this.markDirty();
312
+ return true;
313
+ }
314
+ case 'escape':
315
+ case 'q': {
316
+ this.expandedDiff = null;
317
+ this.scrollOffset = 0;
318
+ this.markDirty();
319
+ return true;
320
+ }
321
+ default:
322
+ return false;
323
+ }
324
+ }
325
+
326
+ private async openDiff(): Promise<void> {
327
+ const item = this.items[this.selectedIndex];
328
+ if (!item || item.kind !== 'file') return;
329
+
330
+ // I3: withLoading guarantees spinner is cleared even if diffFile throws
331
+ try {
332
+ const raw = await this.withLoading('Loading diff…', async () => {
333
+ const git = new GitService(this.workingDirectory);
334
+ return git.diffFile(item.entry.path, item.entry.staged);
335
+ });
336
+ this.expandedDiff = raw ? raw.split('\n') : ['(no diff available)'];
337
+ this.scrollOffset = 0;
338
+ this.markDirty();
339
+ } catch (err) {
340
+ this.expandedDiff = [`Error: ${summarizeError(err)}`];
341
+ this.scrollOffset = 0;
342
+ this.markDirty();
343
+ }
344
+ }
345
+
346
+ // ---------------------------------------------------------------------------
347
+ // Rendering
348
+ // ---------------------------------------------------------------------------
349
+
350
+ override render(width: number, height: number): Line[] {
351
+ if (this.loading) {
352
+ return this.renderMessage(width, height, 'Loading git status...', C.branch);
353
+ }
354
+ if (this.error) {
355
+ return this.renderMessage(width, height, `Git error: ${this.error}`, C.unstaged);
356
+ }
357
+ // I3: spinner during openDiff() async fetch
358
+ if (this.loadingState === 'loading') {
359
+ return this.renderMessage(width, height, 'Loading diff...', C.branch);
360
+ }
361
+ if (this.expandedDiff !== null) {
362
+ return this.renderDiff(width, height);
363
+ }
364
+ return this.renderList(width, height);
365
+ }
366
+
367
+ // -- Helpers -----------------------------------------------------------------
368
+
369
+ private renderMessage(width: number, height: number, msg: string, fg: string): Line[] {
370
+ const lines: Line[] = [buildStyledPanelLine(width, [{ text: msg, fg }])];
371
+ while (lines.length < height) lines.push(createEmptyLine(width));
372
+ return lines;
373
+ }
374
+
375
+ /** Paint a single text string into a new Line at x=startX. */
376
+ private paintText(
377
+ line: Line,
378
+ text: string,
379
+ startX: number,
380
+ width: number,
381
+ fg: string,
382
+ opts: { bold?: boolean; dim?: boolean } = {},
383
+ ): number {
384
+ let x = startX;
385
+ for (const ch of text) {
386
+ if (x >= width) break;
387
+ line[x++] = createStyledCell(ch, { fg, bold: opts.bold, dim: opts.dim });
388
+ }
389
+ return x;
390
+ }
391
+
392
+ private renderBranchLine(width: number): Line {
393
+ const line = createEmptyLine(width);
394
+ let x = 0;
395
+
396
+ const branchIcon = ' git: ';
397
+ x = this.paintText(line, branchIcon, x, width, C.sectionHeader);
398
+ x = this.paintText(line, this.data.branch, x, width, C.branch, { bold: true });
399
+
400
+ if (this.data.ahead > 0) {
401
+ x = this.paintText(line, ` +${this.data.ahead}`, x, width, C.ahead);
402
+ }
403
+ if (this.data.behind > 0) {
404
+ x = this.paintText(line, ` -${this.data.behind}`, x, width, C.behind);
405
+ }
406
+
407
+ const isDirty = this.data.stagedFiles.length > 0 || this.data.unstagedFiles.length > 0;
408
+ const statusText = isDirty ? ' * dirty' : ' y clean';
409
+ const statusFg = isDirty ? C.dirty : C.clean;
410
+ this.paintText(line, statusText, x, width, statusFg);
411
+
412
+ return line;
413
+ }
414
+
415
+ private renderSectionHeader(label: string, width: number): Line {
416
+ return buildStyledPanelLine(width, [{ text: `-- ${label} `, fg: C.sectionHeader, dim: true }]);
417
+ }
418
+
419
+ private renderFileRow(entry: GitFileEntry, selected: boolean, width: number): Line {
420
+ const line = createEmptyLine(width);
421
+ const fg = entry.staged ? C.staged : C.unstaged;
422
+ const prefix = ' ';
423
+ const label = `${prefix}${entry.path}`;
424
+
425
+ if (selected) {
426
+ // Fill background highlight
427
+ for (let i = 0; i < width; i++) {
428
+ line[i] = createStyledCell(' ', { bg: C.selected, fg: C.selectedFg });
429
+ }
430
+ let x = 0;
431
+ for (const ch of label) {
432
+ if (x >= width) break;
433
+ line[x++] = createStyledCell(ch, { fg, bg: C.selected, bold: true });
434
+ }
435
+ } else {
436
+ this.paintText(line, label, 0, width, fg);
437
+ }
438
+ return line;
439
+ }
440
+
441
+ private renderCommitRow(entry: CommitEntry, selected: boolean, width: number): Line {
442
+ const line = createEmptyLine(width);
443
+ const hashPart = ` ${entry.hash} `;
444
+ const msgPart = entry.message.length > 60 ? `${entry.message.slice(0, 57)}...` : entry.message;
445
+
446
+ if (selected) {
447
+ for (let i = 0; i < width; i++) {
448
+ line[i] = createStyledCell(' ', { bg: C.selected, fg: C.selectedFg });
449
+ }
450
+ let x = 0;
451
+ for (const ch of hashPart) {
452
+ if (x >= width) break;
453
+ line[x++] = createStyledCell(ch, { fg: C.commitHash, bg: C.selected });
454
+ }
455
+ for (const ch of msgPart) {
456
+ if (x >= width) break;
457
+ line[x++] = createStyledCell(ch, { fg: C.selectedFg, bg: C.selected });
458
+ }
459
+ } else {
460
+ let x = this.paintText(line, hashPart, 0, width, C.commitHash);
461
+ this.paintText(line, msgPart, x, width, C.commit);
462
+ }
463
+ return line;
464
+ }
465
+
466
+ private renderList(width: number, height: number): Line[] {
467
+ const rows: Line[] = [];
468
+ for (let i = 0; i < this.items.length; i++) {
469
+ const item = this.items[i];
470
+ const selected = i === this.selectedIndex;
471
+ if (!item) continue;
472
+
473
+ switch (item.kind) {
474
+ case 'header':
475
+ rows.push(this.renderBranchLine(width));
476
+ rows.push(createEmptyLine(width)); // spacer
477
+ break;
478
+ case 'section':
479
+ rows.push(this.renderSectionHeader(item.label, width));
480
+ break;
481
+ case 'file':
482
+ rows.push(this.renderFileRow(item.entry, selected, width));
483
+ break;
484
+ case 'commit':
485
+ rows.push(this.renderCommitRow(item.entry, selected, width));
486
+ break;
487
+ case 'empty': {
488
+ rows.push(buildStyledPanelLine(width, [{ text: item.label, fg: C.sectionHeader, dim: true }]));
489
+ break;
490
+ }
491
+ }
492
+ }
493
+
494
+ const selectedRowIndex = this.getRowIndexForItem(this.selectedIndex);
495
+ if (selectedRowIndex >= 0) {
496
+ const isDirty = this.data.stagedFiles.length > 0 || this.data.unstagedFiles.length > 0;
497
+ const selectedItem = this.items[this.selectedIndex];
498
+ const selectedLines: Line[] = [];
499
+ if (selectedItem?.kind === 'file') {
500
+ selectedLines.push(buildPanelLine(width, [
501
+ [' File ', DEFAULT_PANEL_PALETTE.label],
502
+ [selectedItem.entry.path, DEFAULT_PANEL_PALETTE.value],
503
+ [' State ', DEFAULT_PANEL_PALETTE.label],
504
+ [selectedItem.entry.staged ? 'staged' : 'unstaged', selectedItem.entry.staged ? DEFAULT_PANEL_PALETTE.good : DEFAULT_PANEL_PALETTE.warn],
505
+ ]));
506
+ } else if (selectedItem?.kind === 'commit') {
507
+ selectedLines.push(buildPanelLine(width, [
508
+ [' Commit ', DEFAULT_PANEL_PALETTE.label],
509
+ [selectedItem.entry.hash, DEFAULT_PANEL_PALETTE.info],
510
+ [' Author ', DEFAULT_PANEL_PALETTE.label],
511
+ [selectedItem.entry.author, DEFAULT_PANEL_PALETTE.value],
512
+ ]));
513
+ selectedLines.push(buildPanelLine(width, [
514
+ [' Message ', DEFAULT_PANEL_PALETTE.label],
515
+ [selectedItem.entry.message, DEFAULT_PANEL_PALETTE.value],
516
+ ]));
517
+ }
518
+
519
+ const summarySection = {
520
+ title: 'Summary',
521
+ lines: [
522
+ buildPanelLine(width, [
523
+ [' Branch ', DEFAULT_PANEL_PALETTE.label],
524
+ [this.data.branch, DEFAULT_PANEL_PALETTE.info],
525
+ [' Ahead ', DEFAULT_PANEL_PALETTE.label],
526
+ [String(this.data.ahead), this.data.ahead > 0 ? DEFAULT_PANEL_PALETTE.good : DEFAULT_PANEL_PALETTE.dim],
527
+ [' Behind ', DEFAULT_PANEL_PALETTE.label],
528
+ [String(this.data.behind), this.data.behind > 0 ? DEFAULT_PANEL_PALETTE.bad : DEFAULT_PANEL_PALETTE.dim],
529
+ [' Status ', DEFAULT_PANEL_PALETTE.label],
530
+ [isDirty ? 'dirty' : 'clean', isDirty ? DEFAULT_PANEL_PALETTE.warn : DEFAULT_PANEL_PALETTE.good],
531
+ ]),
532
+ ],
533
+ } as const;
534
+ const selectedSection = { title: 'Selected', lines: selectedLines } as const;
535
+ const workspaceSection = resolveScrollablePanelSection(width, height, {
536
+ intro: 'Review branch status, staged and unstaged files, and recent commits. Open a file row to inspect its diff.',
537
+ footerLines: [
538
+ buildPanelLine(width, [[' Up/Down', DEFAULT_PANEL_PALETTE.info], [' navigate', DEFAULT_PANEL_PALETTE.dim], [' Enter', DEFAULT_PANEL_PALETTE.info], [' diff', DEFAULT_PANEL_PALETTE.dim], [' r', DEFAULT_PANEL_PALETTE.info], [' refresh', DEFAULT_PANEL_PALETTE.dim]]),
539
+ ],
540
+ palette: DEFAULT_PANEL_PALETTE,
541
+ beforeSections: [summarySection],
542
+ section: {
543
+ title: 'Workspace',
544
+ scrollableLines: rows,
545
+ selectedIndex: selectedRowIndex,
546
+ scrollOffset: this.scrollOffset,
547
+ minRows: 8,
548
+ },
549
+ afterSections: [selectedSection],
550
+ });
551
+ this.scrollOffset = workspaceSection.scrollOffset;
552
+
553
+ return buildPanelWorkspace(width, height, {
554
+ title: ' Git',
555
+ intro: 'Review branch status, staged and unstaged files, and recent commits. Open a file row to inspect its diff.',
556
+ sections: [
557
+ summarySection,
558
+ workspaceSection.section.lines.length > 0 ? workspaceSection.section : { title: 'Workspace', lines: buildEmptyState(width, ' No git rows', 'This repository has no files or commits to display yet.', [], DEFAULT_PANEL_PALETTE) },
559
+ selectedSection,
560
+ ],
561
+ footerLines: [
562
+ buildPanelLine(width, [[' Up/Down', DEFAULT_PANEL_PALETTE.info], [' navigate', DEFAULT_PANEL_PALETTE.dim], [' Enter', DEFAULT_PANEL_PALETTE.info], [' diff', DEFAULT_PANEL_PALETTE.dim], [' r', DEFAULT_PANEL_PALETTE.info], [' refresh', DEFAULT_PANEL_PALETTE.dim]]),
563
+ ],
564
+ palette: DEFAULT_PANEL_PALETTE,
565
+ });
566
+ }
567
+ return buildPanelWorkspace(width, height, {
568
+ title: ' Git',
569
+ intro: 'Review branch status, staged and unstaged files, and recent commits. Open a file row to inspect its diff.',
570
+ sections: [
571
+ {
572
+ lines: buildEmptyState(width, ' No git rows', 'This repository has no files or commits to display yet.', [], DEFAULT_PANEL_PALETTE),
573
+ },
574
+ ],
575
+ palette: DEFAULT_PANEL_PALETTE,
576
+ });
577
+ }
578
+
579
+ /**
580
+ * Map an item index in `this.items` to the row index in the rendered row list.
581
+ * Header items expand to 2 rows (branch + spacer).
582
+ */
583
+ private getRowIndexForItem(itemIndex: number): number {
584
+ let row = 0;
585
+ for (let i = 0; i < itemIndex && i < this.items.length; i++) {
586
+ const it = this.items[i];
587
+ if (it?.kind === 'header') {
588
+ row += 2;
589
+ } else {
590
+ row += 1;
591
+ }
592
+ }
593
+ return row;
594
+ }
595
+
596
+ private renderDiff(width: number, height: number): Line[] {
597
+ const item = this.items[this.selectedIndex];
598
+ const title =
599
+ item?.kind === 'file' ? `Diff: ${item.entry.path}` : 'Diff';
600
+ const diffLines = this.expandedDiff ?? [];
601
+ const renderedLines = diffLines.map((rawLine) => {
602
+ const dLine = createEmptyLine(width);
603
+ let fg: string;
604
+ if (rawLine.startsWith('+') && !rawLine.startsWith('+++')) {
605
+ fg = C.diffAdd;
606
+ } else if (rawLine.startsWith('-') && !rawLine.startsWith('---')) {
607
+ fg = C.diffRemove;
608
+ } else if (rawLine.startsWith('@@') || rawLine.startsWith('diff') || rawLine.startsWith('index')) {
609
+ fg = C.diffMeta;
610
+ } else {
611
+ fg = C.diffNeutral;
612
+ }
613
+ this.paintText(dLine, rawLine, 0, width, fg);
614
+ return dLine;
615
+ });
616
+ const footerLines = [
617
+ buildPanelLine(width, [[' Up/Down', DEFAULT_PANEL_PALETTE.info], [' scroll', DEFAULT_PANEL_PALETTE.dim], [' Esc/q', DEFAULT_PANEL_PALETTE.info], [' close', DEFAULT_PANEL_PALETTE.dim]]),
618
+ ];
619
+ const diffSection = resolveScrollablePanelSection(width, height, {
620
+ palette: DEFAULT_PANEL_PALETTE,
621
+ footerLines,
622
+ section: {
623
+ title: 'Patch',
624
+ scrollableLines: renderedLines,
625
+ scrollOffset: this.scrollOffset,
626
+ minRows: 1,
627
+ },
628
+ });
629
+ this.scrollOffset = diffSection.scrollOffset;
630
+
631
+ return buildPanelWorkspace(width, height, {
632
+ title: ` ${title}`,
633
+ sections: [diffSection.section],
634
+ footerLines,
635
+ palette: DEFAULT_PANEL_PALETTE,
636
+ });
637
+ }
638
+ }