@akiojin/gwt 4.10.0 → 4.12.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 (630) hide show
  1. package/README.ja.md +4 -4
  2. package/README.md +4 -4
  3. package/bin/gwt.js +1 -1
  4. package/dist/claude.d.ts +2 -0
  5. package/dist/claude.d.ts.map +1 -1
  6. package/dist/claude.js +97 -68
  7. package/dist/claude.js.map +1 -1
  8. package/dist/cli/ui/App.solid.d.ts +29 -0
  9. package/dist/cli/ui/App.solid.d.ts.map +1 -0
  10. package/dist/cli/ui/App.solid.js +1395 -0
  11. package/dist/cli/ui/App.solid.js.map +1 -0
  12. package/dist/cli/ui/components/solid/Footer.d.ts +10 -0
  13. package/dist/cli/ui/components/solid/Footer.d.ts.map +1 -0
  14. package/dist/cli/ui/components/solid/Footer.js +10 -0
  15. package/dist/cli/ui/components/solid/Footer.js.map +1 -0
  16. package/dist/cli/ui/components/{parts → solid}/Header.d.ts +1 -6
  17. package/dist/cli/ui/components/solid/Header.d.ts.map +1 -0
  18. package/dist/cli/ui/components/solid/Header.js +13 -0
  19. package/dist/cli/ui/components/solid/Header.js.map +1 -0
  20. package/dist/cli/ui/components/solid/HelpOverlay.d.ts +8 -0
  21. package/dist/cli/ui/components/solid/HelpOverlay.d.ts.map +1 -0
  22. package/dist/cli/ui/components/solid/HelpOverlay.js +118 -0
  23. package/dist/cli/ui/components/solid/HelpOverlay.js.map +1 -0
  24. package/dist/cli/ui/components/solid/QuickStartStep.d.ts +17 -0
  25. package/dist/cli/ui/components/solid/QuickStartStep.d.ts.map +1 -0
  26. package/dist/cli/ui/components/solid/QuickStartStep.js +152 -0
  27. package/dist/cli/ui/components/solid/QuickStartStep.js.map +1 -0
  28. package/dist/cli/ui/components/solid/SelectInput.d.ts +22 -0
  29. package/dist/cli/ui/components/solid/SelectInput.d.ts.map +1 -0
  30. package/dist/cli/ui/components/solid/SelectInput.js +45 -0
  31. package/dist/cli/ui/components/solid/SelectInput.js.map +1 -0
  32. package/dist/cli/ui/components/solid/TextInput.d.ts +12 -0
  33. package/dist/cli/ui/components/solid/TextInput.d.ts.map +1 -0
  34. package/dist/cli/ui/components/solid/TextInput.js +9 -0
  35. package/dist/cli/ui/components/solid/TextInput.js.map +1 -0
  36. package/dist/cli/ui/components/solid/WizardController.d.ts +34 -0
  37. package/dist/cli/ui/components/solid/WizardController.d.ts.map +1 -0
  38. package/dist/cli/ui/components/solid/WizardController.js +223 -0
  39. package/dist/cli/ui/components/solid/WizardController.js.map +1 -0
  40. package/dist/cli/ui/components/solid/WizardPopup.d.ts +26 -0
  41. package/dist/cli/ui/components/solid/WizardPopup.d.ts.map +1 -0
  42. package/dist/cli/ui/components/solid/WizardPopup.js +68 -0
  43. package/dist/cli/ui/components/solid/WizardPopup.js.map +1 -0
  44. package/dist/cli/ui/components/solid/WizardSteps.d.ts +52 -0
  45. package/dist/cli/ui/components/solid/WizardSteps.d.ts.map +1 -0
  46. package/dist/cli/ui/components/solid/WizardSteps.js +419 -0
  47. package/dist/cli/ui/components/solid/WizardSteps.js.map +1 -0
  48. package/dist/cli/ui/core/index.d.ts +12 -0
  49. package/dist/cli/ui/core/index.d.ts.map +1 -0
  50. package/dist/cli/ui/core/index.js +15 -0
  51. package/dist/cli/ui/core/index.js.map +1 -0
  52. package/dist/cli/ui/core/keybindings.d.ts +106 -0
  53. package/dist/cli/ui/core/keybindings.d.ts.map +1 -0
  54. package/dist/cli/ui/core/keybindings.js +270 -0
  55. package/dist/cli/ui/core/keybindings.js.map +1 -0
  56. package/dist/cli/ui/core/theme.d.ts +123 -0
  57. package/dist/cli/ui/core/theme.d.ts.map +1 -0
  58. package/dist/cli/ui/core/theme.js +191 -0
  59. package/dist/cli/ui/core/theme.js.map +1 -0
  60. package/dist/cli/ui/core/types.d.ts +156 -0
  61. package/dist/cli/ui/core/types.d.ts.map +1 -0
  62. package/dist/cli/ui/core/types.js +10 -0
  63. package/dist/cli/ui/core/types.js.map +1 -0
  64. package/dist/cli/ui/hooks/solid/useScrollableList.d.ts +26 -0
  65. package/dist/cli/ui/hooks/solid/useScrollableList.d.ts.map +1 -0
  66. package/dist/cli/ui/hooks/solid/useScrollableList.js +89 -0
  67. package/dist/cli/ui/hooks/solid/useScrollableList.js.map +1 -0
  68. package/dist/cli/ui/hooks/solid/useTerminalSize.d.ts +10 -0
  69. package/dist/cli/ui/hooks/solid/useTerminalSize.d.ts.map +1 -0
  70. package/dist/cli/ui/hooks/solid/useTerminalSize.js +16 -0
  71. package/dist/cli/ui/hooks/solid/useTerminalSize.js.map +1 -0
  72. package/dist/cli/ui/index.solid.d.ts +5 -0
  73. package/dist/cli/ui/index.solid.d.ts.map +1 -0
  74. package/dist/cli/ui/index.solid.js +21 -0
  75. package/dist/cli/ui/index.solid.js.map +1 -0
  76. package/dist/cli/ui/screens/solid/BranchListScreen.d.ts +53 -0
  77. package/dist/cli/ui/screens/solid/BranchListScreen.d.ts.map +1 -0
  78. package/dist/cli/ui/screens/solid/BranchListScreen.js +829 -0
  79. package/dist/cli/ui/screens/solid/BranchListScreen.js.map +1 -0
  80. package/dist/cli/ui/screens/solid/ConfirmScreen.d.ts +11 -0
  81. package/dist/cli/ui/screens/solid/ConfirmScreen.d.ts.map +1 -0
  82. package/dist/cli/ui/screens/solid/ConfirmScreen.js +45 -0
  83. package/dist/cli/ui/screens/solid/ConfirmScreen.js.map +1 -0
  84. package/dist/cli/ui/screens/solid/EnvironmentScreen.d.ts +14 -0
  85. package/dist/cli/ui/screens/solid/EnvironmentScreen.d.ts.map +1 -0
  86. package/dist/cli/ui/screens/solid/EnvironmentScreen.js +90 -0
  87. package/dist/cli/ui/screens/solid/EnvironmentScreen.js.map +1 -0
  88. package/dist/cli/ui/screens/solid/ErrorScreen.d.ts +8 -0
  89. package/dist/cli/ui/screens/solid/ErrorScreen.d.ts.map +1 -0
  90. package/dist/cli/ui/screens/solid/ErrorScreen.js +18 -0
  91. package/dist/cli/ui/screens/solid/ErrorScreen.js.map +1 -0
  92. package/dist/cli/ui/screens/solid/InputScreen.d.ts +13 -0
  93. package/dist/cli/ui/screens/solid/InputScreen.d.ts.map +1 -0
  94. package/dist/cli/ui/screens/solid/InputScreen.js +17 -0
  95. package/dist/cli/ui/screens/solid/InputScreen.js.map +1 -0
  96. package/dist/cli/ui/screens/solid/LoadingIndicator.d.ts +9 -0
  97. package/dist/cli/ui/screens/solid/LoadingIndicator.d.ts.map +1 -0
  98. package/dist/cli/ui/screens/solid/LoadingIndicator.js +43 -0
  99. package/dist/cli/ui/screens/solid/LoadingIndicator.js.map +1 -0
  100. package/dist/cli/ui/{components/screens → screens/solid}/LogDetailScreen.d.ts +2 -2
  101. package/dist/cli/ui/screens/solid/LogDetailScreen.d.ts.map +1 -0
  102. package/dist/cli/ui/screens/solid/LogDetailScreen.js +34 -0
  103. package/dist/cli/ui/screens/solid/LogDetailScreen.js.map +1 -0
  104. package/dist/cli/ui/{components/screens/LogListScreen.d.ts → screens/solid/LogScreen.d.ts} +10 -4
  105. package/dist/cli/ui/screens/solid/LogScreen.d.ts.map +1 -0
  106. package/dist/cli/ui/screens/solid/LogScreen.js +333 -0
  107. package/dist/cli/ui/screens/solid/LogScreen.js.map +1 -0
  108. package/dist/cli/ui/screens/solid/ProfileEnvScreen.d.ts +17 -0
  109. package/dist/cli/ui/screens/solid/ProfileEnvScreen.d.ts.map +1 -0
  110. package/dist/cli/ui/screens/solid/ProfileEnvScreen.js +115 -0
  111. package/dist/cli/ui/screens/solid/ProfileEnvScreen.js.map +1 -0
  112. package/dist/cli/ui/screens/solid/ProfileScreen.d.ts +17 -0
  113. package/dist/cli/ui/screens/solid/ProfileScreen.d.ts.map +1 -0
  114. package/dist/cli/ui/screens/solid/ProfileScreen.js +50 -0
  115. package/dist/cli/ui/screens/solid/ProfileScreen.js.map +1 -0
  116. package/dist/cli/ui/screens/solid/SelectorScreen.d.ts +20 -0
  117. package/dist/cli/ui/screens/solid/SelectorScreen.d.ts.map +1 -0
  118. package/dist/cli/ui/screens/solid/SelectorScreen.js +98 -0
  119. package/dist/cli/ui/screens/solid/SelectorScreen.js.map +1 -0
  120. package/dist/cli/ui/screens/solid/WorktreeCreateScreen.d.ts +13 -0
  121. package/dist/cli/ui/screens/solid/WorktreeCreateScreen.d.ts.map +1 -0
  122. package/dist/cli/ui/screens/solid/WorktreeCreateScreen.js +59 -0
  123. package/dist/cli/ui/screens/solid/WorktreeCreateScreen.js.map +1 -0
  124. package/dist/cli/ui/stores/appStore.d.ts +143 -0
  125. package/dist/cli/ui/stores/appStore.d.ts.map +1 -0
  126. package/dist/cli/ui/stores/appStore.js +158 -0
  127. package/dist/cli/ui/stores/appStore.js.map +1 -0
  128. package/dist/cli/ui/stores/branchStore.d.ts +159 -0
  129. package/dist/cli/ui/stores/branchStore.d.ts.map +1 -0
  130. package/dist/cli/ui/stores/branchStore.js +275 -0
  131. package/dist/cli/ui/stores/branchStore.js.map +1 -0
  132. package/dist/cli/ui/stores/index.d.ts +11 -0
  133. package/dist/cli/ui/stores/index.d.ts.map +1 -0
  134. package/dist/cli/ui/stores/index.js +14 -0
  135. package/dist/cli/ui/stores/index.js.map +1 -0
  136. package/dist/cli/ui/stores/uiStore.d.ts +146 -0
  137. package/dist/cli/ui/stores/uiStore.d.ts.map +1 -0
  138. package/dist/cli/ui/stores/uiStore.js +166 -0
  139. package/dist/cli/ui/stores/uiStore.js.map +1 -0
  140. package/dist/cli/ui/types.d.ts +17 -1
  141. package/dist/cli/ui/types.d.ts.map +1 -1
  142. package/dist/cli/ui/utils/branchFormatter.d.ts +1 -0
  143. package/dist/cli/ui/utils/branchFormatter.d.ts.map +1 -1
  144. package/dist/cli/ui/utils/branchFormatter.js +36 -217
  145. package/dist/cli/ui/utils/branchFormatter.js.map +1 -1
  146. package/dist/cli/ui/utils/continueSession.d.ts +18 -0
  147. package/dist/cli/ui/utils/continueSession.d.ts.map +1 -1
  148. package/dist/cli/ui/utils/continueSession.js +90 -2
  149. package/dist/cli/ui/utils/continueSession.js.map +1 -1
  150. package/dist/cli/ui/utils/versionCache.d.ts +37 -0
  151. package/dist/cli/ui/utils/versionCache.d.ts.map +1 -0
  152. package/dist/cli/ui/utils/versionCache.js +70 -0
  153. package/dist/cli/ui/utils/versionCache.js.map +1 -0
  154. package/dist/cli/ui/utils/versionFetcher.d.ts +41 -0
  155. package/dist/cli/ui/utils/versionFetcher.d.ts.map +1 -0
  156. package/dist/cli/ui/utils/versionFetcher.js +89 -0
  157. package/dist/cli/ui/utils/versionFetcher.js.map +1 -0
  158. package/dist/client/assets/{index-ChHC-Puh.css → index-BbfV7Wuj.css} +1 -1
  159. package/dist/client/assets/index-CoAyq5x1.js +78 -0
  160. package/dist/client/index.html +2 -2
  161. package/dist/codex.d.ts +2 -0
  162. package/dist/codex.d.ts.map +1 -1
  163. package/dist/codex.js +109 -39
  164. package/dist/codex.js.map +1 -1
  165. package/dist/config/builtin-coding-agents.js +4 -4
  166. package/dist/config/builtin-coding-agents.js.map +1 -1
  167. package/dist/config/index.d.ts +2 -0
  168. package/dist/config/index.d.ts.map +1 -1
  169. package/dist/config/index.js +11 -0
  170. package/dist/config/index.js.map +1 -1
  171. package/dist/gemini.d.ts +2 -0
  172. package/dist/gemini.d.ts.map +1 -1
  173. package/dist/gemini.js +77 -39
  174. package/dist/gemini.js.map +1 -1
  175. package/dist/index.d.ts +1 -1
  176. package/dist/index.d.ts.map +1 -1
  177. package/dist/index.js +153 -103
  178. package/dist/index.js.map +1 -1
  179. package/dist/launcher.d.ts.map +1 -1
  180. package/dist/launcher.js +56 -6
  181. package/dist/launcher.js.map +1 -1
  182. package/dist/logging/agentOutput.d.ts +21 -0
  183. package/dist/logging/agentOutput.d.ts.map +1 -0
  184. package/dist/logging/agentOutput.js +164 -0
  185. package/dist/logging/agentOutput.js.map +1 -0
  186. package/dist/logging/formatter.d.ts.map +1 -1
  187. package/dist/logging/formatter.js +18 -4
  188. package/dist/logging/formatter.js.map +1 -1
  189. package/dist/logging/logger.d.ts.map +1 -1
  190. package/dist/logging/logger.js +28 -9
  191. package/dist/logging/logger.js.map +1 -1
  192. package/dist/logging/reader.d.ts +21 -0
  193. package/dist/logging/reader.d.ts.map +1 -1
  194. package/dist/logging/reader.js +79 -0
  195. package/dist/logging/reader.js.map +1 -1
  196. package/dist/opentui/highlights-eq9cgrbb.scm +604 -0
  197. package/dist/opentui/highlights-ghv9g403.scm +205 -0
  198. package/dist/opentui/highlights-hk7bwhj4.scm +284 -0
  199. package/dist/opentui/highlights-r812a2qc.scm +150 -0
  200. package/dist/opentui/highlights-x6tmsnaa.scm +115 -0
  201. package/dist/opentui/index.solid.d.ts +2 -0
  202. package/dist/opentui/index.solid.d.ts.map +1 -0
  203. package/dist/opentui/index.solid.js +53687 -0
  204. package/dist/opentui/index.solid.js.map +1 -0
  205. package/dist/opentui/injections-73j83es3.scm +27 -0
  206. package/dist/opentui/tree-sitter-javascript-nd0q4pe9.wasm +0 -0
  207. package/dist/opentui/tree-sitter-markdown-411r6y9b.wasm +0 -0
  208. package/dist/opentui/tree-sitter-markdown_inline-j5349f42.wasm +0 -0
  209. package/dist/opentui/tree-sitter-typescript-zxjzwt75.wasm +0 -0
  210. package/dist/opentui/tree-sitter-zig-e78zbjpm.wasm +0 -0
  211. package/dist/repositories/worktree.repository.d.ts +1 -0
  212. package/dist/repositories/worktree.repository.d.ts.map +1 -1
  213. package/dist/repositories/worktree.repository.js +7 -0
  214. package/dist/repositories/worktree.repository.js.map +1 -1
  215. package/dist/services/codingAgentResolver.d.ts +2 -0
  216. package/dist/services/codingAgentResolver.d.ts.map +1 -1
  217. package/dist/services/codingAgentResolver.js +30 -4
  218. package/dist/services/codingAgentResolver.js.map +1 -1
  219. package/dist/services/dependency-installer.d.ts.map +1 -1
  220. package/dist/services/dependency-installer.js +2 -7
  221. package/dist/services/dependency-installer.js.map +1 -1
  222. package/dist/types/api.d.ts +3 -0
  223. package/dist/types/api.d.ts.map +1 -1
  224. package/dist/types/coding-agent.d.ts +62 -0
  225. package/dist/types/coding-agent.d.ts.map +1 -0
  226. package/dist/types/coding-agent.js +29 -0
  227. package/dist/types/coding-agent.js.map +1 -0
  228. package/dist/types/tools.d.ts +17 -0
  229. package/dist/types/tools.d.ts.map +1 -1
  230. package/dist/utils/coding-agent-colors.d.ts +88 -0
  231. package/dist/utils/coding-agent-colors.d.ts.map +1 -0
  232. package/dist/utils/coding-agent-colors.js +137 -0
  233. package/dist/utils/coding-agent-colors.js.map +1 -0
  234. package/dist/utils/command.d.ts +1 -1
  235. package/dist/utils/command.js +1 -1
  236. package/dist/utils/error-utils.d.ts +27 -0
  237. package/dist/utils/error-utils.d.ts.map +1 -0
  238. package/dist/utils/error-utils.js +98 -0
  239. package/dist/utils/error-utils.js.map +1 -0
  240. package/dist/utils/npmRegistry.d.ts +61 -0
  241. package/dist/utils/npmRegistry.d.ts.map +1 -0
  242. package/dist/utils/npmRegistry.js +180 -0
  243. package/dist/utils/npmRegistry.js.map +1 -0
  244. package/dist/utils/prompt.d.ts +1 -1
  245. package/dist/utils/prompt.js +1 -1
  246. package/dist/utils/session/common.d.ts +8 -0
  247. package/dist/utils/session/common.d.ts.map +1 -1
  248. package/dist/utils/session/common.js +22 -0
  249. package/dist/utils/session/common.js.map +1 -1
  250. package/dist/utils/session/parsers/claude.d.ts +10 -4
  251. package/dist/utils/session/parsers/claude.d.ts.map +1 -1
  252. package/dist/utils/session/parsers/claude.js +64 -18
  253. package/dist/utils/session/parsers/claude.js.map +1 -1
  254. package/dist/utils/session/parsers/codex.d.ts.map +1 -1
  255. package/dist/utils/session/parsers/codex.js +47 -20
  256. package/dist/utils/session/parsers/codex.js.map +1 -1
  257. package/dist/utils/session/parsers/gemini.d.ts.map +1 -1
  258. package/dist/utils/session/parsers/gemini.js +43 -6
  259. package/dist/utils/session/parsers/gemini.js.map +1 -1
  260. package/dist/utils/session/parsers/opencode.d.ts.map +1 -1
  261. package/dist/utils/session/parsers/opencode.js +43 -6
  262. package/dist/utils/session/parsers/opencode.js.map +1 -1
  263. package/dist/utils/session/types.d.ts +7 -0
  264. package/dist/utils/session/types.d.ts.map +1 -1
  265. package/dist/utils/terminal.d.ts +1 -0
  266. package/dist/utils/terminal.d.ts.map +1 -1
  267. package/dist/utils/terminal.js +20 -0
  268. package/dist/utils/terminal.js.map +1 -1
  269. package/dist/utils.d.ts +9 -0
  270. package/dist/utils.d.ts.map +1 -1
  271. package/dist/utils.js +33 -2
  272. package/dist/utils.js.map +1 -1
  273. package/dist/web/client/src/components/CodingAgentLaunchModal.d.ts.map +1 -1
  274. package/dist/web/client/src/components/CodingAgentLaunchModal.js +7 -16
  275. package/dist/web/client/src/components/CodingAgentLaunchModal.js.map +1 -1
  276. package/dist/web/client/src/components/branch-detail/BranchInfoCards.d.ts.map +1 -1
  277. package/dist/web/client/src/components/branch-detail/BranchInfoCards.js +7 -2
  278. package/dist/web/client/src/components/branch-detail/BranchInfoCards.js.map +1 -1
  279. package/dist/web/client/src/components/branch-detail/SessionHistoryTable.d.ts.map +1 -1
  280. package/dist/web/client/src/components/branch-detail/SessionHistoryTable.js +2 -1
  281. package/dist/web/client/src/components/branch-detail/SessionHistoryTable.js.map +1 -1
  282. package/dist/web/client/src/components/branch-detail/ToolLauncher.d.ts +2 -2
  283. package/dist/web/client/src/components/branch-detail/ToolLauncher.d.ts.map +1 -1
  284. package/dist/web/client/src/components/branch-detail/ToolLauncher.js +5 -5
  285. package/dist/web/client/src/components/branch-detail/ToolLauncher.js.map +1 -1
  286. package/dist/web/client/src/components/ui/alert.d.ts +1 -1
  287. package/dist/web/client/src/lib/coding-agent-colors.d.ts +86 -0
  288. package/dist/web/client/src/lib/coding-agent-colors.d.ts.map +1 -0
  289. package/dist/web/client/src/lib/coding-agent-colors.js +135 -0
  290. package/dist/web/client/src/lib/coding-agent-colors.js.map +1 -0
  291. package/dist/web/client/src/pages/BranchDetailPage.js +10 -10
  292. package/dist/web/client/src/pages/BranchDetailPage.js.map +1 -1
  293. package/dist/web/server/pty/manager.d.ts +2 -0
  294. package/dist/web/server/pty/manager.d.ts.map +1 -1
  295. package/dist/web/server/pty/manager.js +104 -0
  296. package/dist/web/server/pty/manager.js.map +1 -1
  297. package/dist/web/server/routes/sessions.d.ts.map +1 -1
  298. package/dist/web/server/routes/sessions.js +5 -1
  299. package/dist/web/server/routes/sessions.js.map +1 -1
  300. package/dist/web/server/services/branches.d.ts.map +1 -1
  301. package/dist/web/server/services/branches.js +10 -8
  302. package/dist/web/server/services/branches.js.map +1 -1
  303. package/dist/web/server/services/worktrees.js +2 -2
  304. package/dist/web/server/services/worktrees.js.map +1 -1
  305. package/dist/worktree.d.ts +50 -1
  306. package/dist/worktree.d.ts.map +1 -1
  307. package/dist/worktree.js +292 -100
  308. package/dist/worktree.js.map +1 -1
  309. package/package.json +13 -14
  310. package/src/claude.ts +129 -95
  311. package/src/cli/ui/App.solid.tsx +2096 -0
  312. package/src/cli/ui/__tests__/solid/AppSolid.cleanup.test.tsx +1084 -0
  313. package/src/cli/ui/__tests__/solid/BranchListScreen.test.tsx +343 -0
  314. package/src/cli/ui/__tests__/solid/ConfirmScreen.test.tsx +77 -0
  315. package/src/cli/ui/__tests__/solid/LogScreen.test.tsx +351 -0
  316. package/src/cli/ui/__tests__/solid/components/QuickStartStep.test.tsx +308 -0
  317. package/src/cli/ui/__tests__/solid/components/WizardPopup.test.tsx +231 -0
  318. package/src/cli/ui/__tests__/solid/components/WizardSteps.test.tsx +241 -0
  319. package/src/cli/ui/__tests__/utils/branchFormatter.test.ts +79 -334
  320. package/src/cli/ui/__tests__/utils/clipboard.test.ts +3 -3
  321. package/src/cli/ui/__tests__/utils/statisticsCalculator.test.ts +1 -1
  322. package/src/cli/ui/components/solid/Footer.tsx +36 -0
  323. package/src/cli/ui/components/{parts → solid}/Header.tsx +17 -28
  324. package/src/cli/ui/components/solid/HelpOverlay.tsx +194 -0
  325. package/src/cli/ui/components/solid/QuickStartStep.tsx +209 -0
  326. package/src/cli/ui/components/{parts → solid}/ScrollableList.tsx +7 -8
  327. package/src/cli/ui/components/solid/SearchInput.tsx +42 -0
  328. package/src/cli/ui/components/solid/SelectInput.tsx +88 -0
  329. package/src/cli/ui/components/solid/Stats.tsx +92 -0
  330. package/src/cli/ui/components/solid/TextInput.tsx +49 -0
  331. package/src/cli/ui/components/solid/WizardController.tsx +393 -0
  332. package/src/cli/ui/components/solid/WizardPopup.tsx +135 -0
  333. package/src/cli/ui/components/solid/WizardSteps.tsx +736 -0
  334. package/src/cli/ui/core/index.ts +17 -0
  335. package/src/cli/ui/core/keybindings.ts +367 -0
  336. package/src/cli/ui/core/theme.ts +266 -0
  337. package/src/cli/ui/core/types.ts +235 -0
  338. package/src/cli/ui/hooks/solid/useAsyncOperation.ts +78 -0
  339. package/src/cli/ui/hooks/solid/useFilter.ts +86 -0
  340. package/src/cli/ui/hooks/solid/useGitOperations.ts +81 -0
  341. package/src/cli/ui/hooks/solid/useKeyHandler.ts +103 -0
  342. package/src/cli/ui/hooks/solid/useScrollableList.ts +149 -0
  343. package/src/cli/ui/hooks/solid/useSelection.ts +77 -0
  344. package/src/cli/ui/hooks/solid/useTerminalSize.ts +22 -0
  345. package/src/cli/ui/index.solid.ts +28 -0
  346. package/src/cli/ui/screens/solid/BranchListScreen.tsx +1153 -0
  347. package/src/cli/ui/screens/solid/ConfirmScreen.tsx +86 -0
  348. package/src/cli/ui/screens/solid/EnvironmentScreen.tsx +161 -0
  349. package/src/cli/ui/screens/solid/ErrorScreen.tsx +42 -0
  350. package/src/cli/ui/screens/solid/InputScreen.tsx +55 -0
  351. package/src/cli/ui/screens/solid/LoadingIndicator.tsx +77 -0
  352. package/src/cli/ui/screens/solid/LogDetailScreen.tsx +75 -0
  353. package/src/cli/ui/screens/solid/LogScreen.tsx +504 -0
  354. package/src/cli/ui/screens/solid/ProfileEnvScreen.tsx +196 -0
  355. package/src/cli/ui/screens/solid/ProfileScreen.tsx +98 -0
  356. package/src/cli/ui/screens/solid/SelectorScreen.tsx +181 -0
  357. package/src/cli/ui/screens/solid/SettingsScreen.tsx +52 -0
  358. package/src/cli/ui/screens/solid/WorktreeCreateScreen.tsx +136 -0
  359. package/src/cli/ui/screens/solid/WorktreeDeleteScreen.tsx +40 -0
  360. package/src/cli/ui/stores/appStore.ts +208 -0
  361. package/src/cli/ui/stores/branchStore.ts +357 -0
  362. package/src/cli/ui/stores/index.ts +31 -0
  363. package/src/cli/ui/stores/uiStore.ts +226 -0
  364. package/src/cli/ui/types.ts +21 -3
  365. package/src/cli/ui/utils/__tests__/branchFormatter.test.ts +227 -0
  366. package/src/cli/ui/utils/branchFormatter.ts +43 -222
  367. package/src/cli/ui/utils/continueSession.ts +133 -2
  368. package/src/cli/ui/utils/modelOptions.test.ts +1 -1
  369. package/src/cli/ui/utils/versionCache.ts +93 -0
  370. package/src/cli/ui/utils/versionFetcher.ts +120 -0
  371. package/src/codex.ts +138 -39
  372. package/src/config/__tests__/saveSession.test.ts +143 -0
  373. package/src/config/builtin-coding-agents.ts +4 -4
  374. package/src/config/index.ts +14 -0
  375. package/src/gemini.ts +107 -46
  376. package/src/index.test.ts +25 -19
  377. package/src/index.ts +202 -143
  378. package/src/launcher.ts +66 -6
  379. package/src/logging/agentOutput.ts +216 -0
  380. package/src/logging/formatter.ts +23 -4
  381. package/src/logging/logger.ts +34 -10
  382. package/src/logging/reader.ts +117 -0
  383. package/src/opentui/index.solid.ts +1 -0
  384. package/src/repositories/worktree.repository.ts +8 -0
  385. package/src/services/__tests__/BatchMergeService.test.ts +83 -67
  386. package/src/services/__tests__/WorktreeOrchestrator.test.ts +8 -7
  387. package/src/services/codingAgentResolver.ts +30 -4
  388. package/src/services/dependency-installer.ts +2 -9
  389. package/src/types/api.ts +3 -0
  390. package/src/types/coding-agent.ts +85 -0
  391. package/src/types/tools.ts +19 -0
  392. package/src/utils/__tests__/npmRegistry.test.ts +250 -0
  393. package/src/utils/__tests__/prompt.test.ts +4 -5
  394. package/src/utils/coding-agent-colors.ts +165 -0
  395. package/src/utils/command.ts +1 -1
  396. package/src/utils/error-utils.ts +133 -0
  397. package/src/utils/npmRegistry.ts +249 -0
  398. package/src/utils/prompt.ts +1 -1
  399. package/src/utils/session/common.ts +28 -0
  400. package/src/utils/session/parsers/claude.ts +79 -29
  401. package/src/utils/session/parsers/codex.ts +50 -18
  402. package/src/utils/session/parsers/gemini.ts +46 -5
  403. package/src/utils/session/parsers/opencode.ts +46 -5
  404. package/src/utils/session/types.ts +4 -0
  405. package/src/utils/terminal.ts +24 -0
  406. package/src/utils.test.ts +1 -1
  407. package/src/utils.ts +37 -4
  408. package/src/web/client/src/components/CodingAgentLaunchModal.tsx +12 -21
  409. package/src/web/client/src/components/branch-detail/BranchInfoCards.tsx +16 -1
  410. package/src/web/client/src/components/branch-detail/SessionHistoryTable.tsx +7 -1
  411. package/src/web/client/src/components/branch-detail/ToolLauncher.tsx +11 -6
  412. package/src/web/client/src/lib/coding-agent-colors.ts +149 -0
  413. package/src/web/client/src/pages/BranchDetailPage.tsx +11 -11
  414. package/src/web/server/pty/manager.ts +139 -0
  415. package/src/web/server/routes/sessions.ts +6 -0
  416. package/src/web/server/services/branches.ts +11 -8
  417. package/src/web/server/services/worktrees.ts +2 -2
  418. package/src/worktree.ts +386 -114
  419. package/dist/cli/ui/components/App.d.ts +0 -25
  420. package/dist/cli/ui/components/App.d.ts.map +0 -1
  421. package/dist/cli/ui/components/App.js +0 -1006
  422. package/dist/cli/ui/components/App.js.map +0 -1
  423. package/dist/cli/ui/components/common/Confirm.d.ts +0 -13
  424. package/dist/cli/ui/components/common/Confirm.d.ts.map +0 -1
  425. package/dist/cli/ui/components/common/Confirm.js +0 -20
  426. package/dist/cli/ui/components/common/Confirm.js.map +0 -1
  427. package/dist/cli/ui/components/common/ErrorBoundary.d.ts +0 -23
  428. package/dist/cli/ui/components/common/ErrorBoundary.d.ts.map +0 -1
  429. package/dist/cli/ui/components/common/ErrorBoundary.js +0 -37
  430. package/dist/cli/ui/components/common/ErrorBoundary.js.map +0 -1
  431. package/dist/cli/ui/components/common/Input.d.ts +0 -19
  432. package/dist/cli/ui/components/common/Input.d.ts.map +0 -1
  433. package/dist/cli/ui/components/common/Input.js +0 -22
  434. package/dist/cli/ui/components/common/Input.js.map +0 -1
  435. package/dist/cli/ui/components/common/LoadingIndicator.d.ts +0 -19
  436. package/dist/cli/ui/components/common/LoadingIndicator.d.ts.map +0 -1
  437. package/dist/cli/ui/components/common/LoadingIndicator.js +0 -61
  438. package/dist/cli/ui/components/common/LoadingIndicator.js.map +0 -1
  439. package/dist/cli/ui/components/common/Select.d.ts +0 -38
  440. package/dist/cli/ui/components/common/Select.d.ts.map +0 -1
  441. package/dist/cli/ui/components/common/Select.js +0 -151
  442. package/dist/cli/ui/components/common/Select.js.map +0 -1
  443. package/dist/cli/ui/components/common/SpinnerIcon.d.ts +0 -20
  444. package/dist/cli/ui/components/common/SpinnerIcon.d.ts.map +0 -1
  445. package/dist/cli/ui/components/common/SpinnerIcon.js +0 -61
  446. package/dist/cli/ui/components/common/SpinnerIcon.js.map +0 -1
  447. package/dist/cli/ui/components/parts/Footer.d.ts +0 -15
  448. package/dist/cli/ui/components/parts/Footer.d.ts.map +0 -1
  449. package/dist/cli/ui/components/parts/Footer.js +0 -20
  450. package/dist/cli/ui/components/parts/Footer.js.map +0 -1
  451. package/dist/cli/ui/components/parts/Header.d.ts.map +0 -1
  452. package/dist/cli/ui/components/parts/Header.js +0 -24
  453. package/dist/cli/ui/components/parts/Header.js.map +0 -1
  454. package/dist/cli/ui/components/parts/MergeStatusList.d.ts +0 -13
  455. package/dist/cli/ui/components/parts/MergeStatusList.d.ts.map +0 -1
  456. package/dist/cli/ui/components/parts/MergeStatusList.js +0 -52
  457. package/dist/cli/ui/components/parts/MergeStatusList.js.map +0 -1
  458. package/dist/cli/ui/components/parts/ProgressBar.d.ts +0 -13
  459. package/dist/cli/ui/components/parts/ProgressBar.d.ts.map +0 -1
  460. package/dist/cli/ui/components/parts/ProgressBar.js +0 -53
  461. package/dist/cli/ui/components/parts/ProgressBar.js.map +0 -1
  462. package/dist/cli/ui/components/parts/ScrollableList.d.ts +0 -12
  463. package/dist/cli/ui/components/parts/ScrollableList.d.ts.map +0 -1
  464. package/dist/cli/ui/components/parts/ScrollableList.js +0 -11
  465. package/dist/cli/ui/components/parts/ScrollableList.js.map +0 -1
  466. package/dist/cli/ui/components/parts/Stats.d.ts +0 -10
  467. package/dist/cli/ui/components/parts/Stats.d.ts.map +0 -1
  468. package/dist/cli/ui/components/parts/Stats.js +0 -55
  469. package/dist/cli/ui/components/parts/Stats.js.map +0 -1
  470. package/dist/cli/ui/components/screens/BatchMergeProgressScreen.d.ts +0 -17
  471. package/dist/cli/ui/components/screens/BatchMergeProgressScreen.d.ts.map +0 -1
  472. package/dist/cli/ui/components/screens/BatchMergeProgressScreen.js +0 -42
  473. package/dist/cli/ui/components/screens/BatchMergeProgressScreen.js.map +0 -1
  474. package/dist/cli/ui/components/screens/BatchMergeResultScreen.d.ts +0 -17
  475. package/dist/cli/ui/components/screens/BatchMergeResultScreen.d.ts.map +0 -1
  476. package/dist/cli/ui/components/screens/BatchMergeResultScreen.js +0 -72
  477. package/dist/cli/ui/components/screens/BatchMergeResultScreen.js.map +0 -1
  478. package/dist/cli/ui/components/screens/BranchCreatorScreen.d.ts +0 -18
  479. package/dist/cli/ui/components/screens/BranchCreatorScreen.d.ts.map +0 -1
  480. package/dist/cli/ui/components/screens/BranchCreatorScreen.js +0 -151
  481. package/dist/cli/ui/components/screens/BranchCreatorScreen.js.map +0 -1
  482. package/dist/cli/ui/components/screens/BranchListScreen.d.ts +0 -60
  483. package/dist/cli/ui/components/screens/BranchListScreen.d.ts.map +0 -1
  484. package/dist/cli/ui/components/screens/BranchListScreen.js +0 -476
  485. package/dist/cli/ui/components/screens/BranchListScreen.js.map +0 -1
  486. package/dist/cli/ui/components/screens/BranchQuickStartScreen.d.ts +0 -30
  487. package/dist/cli/ui/components/screens/BranchQuickStartScreen.d.ts.map +0 -1
  488. package/dist/cli/ui/components/screens/BranchQuickStartScreen.js +0 -148
  489. package/dist/cli/ui/components/screens/BranchQuickStartScreen.js.map +0 -1
  490. package/dist/cli/ui/components/screens/CodingAgentSelectorScreen.d.ts +0 -27
  491. package/dist/cli/ui/components/screens/CodingAgentSelectorScreen.d.ts.map +0 -1
  492. package/dist/cli/ui/components/screens/CodingAgentSelectorScreen.js +0 -93
  493. package/dist/cli/ui/components/screens/CodingAgentSelectorScreen.js.map +0 -1
  494. package/dist/cli/ui/components/screens/EnvironmentProfileScreen.d.ts +0 -19
  495. package/dist/cli/ui/components/screens/EnvironmentProfileScreen.d.ts.map +0 -1
  496. package/dist/cli/ui/components/screens/EnvironmentProfileScreen.js +0 -577
  497. package/dist/cli/ui/components/screens/EnvironmentProfileScreen.js.map +0 -1
  498. package/dist/cli/ui/components/screens/ExecutionModeSelectorScreen.d.ts +0 -45
  499. package/dist/cli/ui/components/screens/ExecutionModeSelectorScreen.d.ts.map +0 -1
  500. package/dist/cli/ui/components/screens/ExecutionModeSelectorScreen.js +0 -95
  501. package/dist/cli/ui/components/screens/ExecutionModeSelectorScreen.js.map +0 -1
  502. package/dist/cli/ui/components/screens/LogDatePickerScreen.d.ts +0 -10
  503. package/dist/cli/ui/components/screens/LogDatePickerScreen.d.ts.map +0 -1
  504. package/dist/cli/ui/components/screens/LogDatePickerScreen.js +0 -44
  505. package/dist/cli/ui/components/screens/LogDatePickerScreen.js.map +0 -1
  506. package/dist/cli/ui/components/screens/LogDetailScreen.d.ts.map +0 -1
  507. package/dist/cli/ui/components/screens/LogDetailScreen.js +0 -34
  508. package/dist/cli/ui/components/screens/LogDetailScreen.js.map +0 -1
  509. package/dist/cli/ui/components/screens/LogListScreen.d.ts.map +0 -1
  510. package/dist/cli/ui/components/screens/LogListScreen.js +0 -107
  511. package/dist/cli/ui/components/screens/LogListScreen.js.map +0 -1
  512. package/dist/cli/ui/components/screens/ModelSelectorScreen.d.ts +0 -24
  513. package/dist/cli/ui/components/screens/ModelSelectorScreen.d.ts.map +0 -1
  514. package/dist/cli/ui/components/screens/ModelSelectorScreen.js +0 -197
  515. package/dist/cli/ui/components/screens/ModelSelectorScreen.js.map +0 -1
  516. package/dist/cli/ui/components/screens/PRCleanupScreen.d.ts +0 -29
  517. package/dist/cli/ui/components/screens/PRCleanupScreen.d.ts.map +0 -1
  518. package/dist/cli/ui/components/screens/PRCleanupScreen.js +0 -92
  519. package/dist/cli/ui/components/screens/PRCleanupScreen.js.map +0 -1
  520. package/dist/cli/ui/components/screens/SessionSelectorScreen.d.ts +0 -31
  521. package/dist/cli/ui/components/screens/SessionSelectorScreen.d.ts.map +0 -1
  522. package/dist/cli/ui/components/screens/SessionSelectorScreen.js +0 -67
  523. package/dist/cli/ui/components/screens/SessionSelectorScreen.js.map +0 -1
  524. package/dist/cli/ui/hooks/useAppInput.d.ts +0 -21
  525. package/dist/cli/ui/hooks/useAppInput.d.ts.map +0 -1
  526. package/dist/cli/ui/hooks/useAppInput.js +0 -138
  527. package/dist/cli/ui/hooks/useAppInput.js.map +0 -1
  528. package/dist/cli/ui/hooks/useBatchMerge.d.ts +0 -17
  529. package/dist/cli/ui/hooks/useBatchMerge.d.ts.map +0 -1
  530. package/dist/cli/ui/hooks/useBatchMerge.js +0 -77
  531. package/dist/cli/ui/hooks/useBatchMerge.js.map +0 -1
  532. package/dist/cli/ui/hooks/useGitData.d.ts +0 -21
  533. package/dist/cli/ui/hooks/useGitData.d.ts.map +0 -1
  534. package/dist/cli/ui/hooks/useGitData.js +0 -229
  535. package/dist/cli/ui/hooks/useGitData.js.map +0 -1
  536. package/dist/cli/ui/hooks/useProfiles.d.ts +0 -41
  537. package/dist/cli/ui/hooks/useProfiles.d.ts.map +0 -1
  538. package/dist/cli/ui/hooks/useProfiles.js +0 -136
  539. package/dist/cli/ui/hooks/useProfiles.js.map +0 -1
  540. package/dist/cli/ui/hooks/useScreenState.d.ts +0 -12
  541. package/dist/cli/ui/hooks/useScreenState.d.ts.map +0 -1
  542. package/dist/cli/ui/hooks/useScreenState.js +0 -30
  543. package/dist/cli/ui/hooks/useScreenState.js.map +0 -1
  544. package/dist/cli/ui/hooks/useTerminalSize.d.ts +0 -9
  545. package/dist/cli/ui/hooks/useTerminalSize.d.ts.map +0 -1
  546. package/dist/cli/ui/hooks/useTerminalSize.js +0 -24
  547. package/dist/cli/ui/hooks/useTerminalSize.js.map +0 -1
  548. package/dist/cli/ui/hooks/useToolStatus.d.ts +0 -30
  549. package/dist/cli/ui/hooks/useToolStatus.d.ts.map +0 -1
  550. package/dist/cli/ui/hooks/useToolStatus.js +0 -49
  551. package/dist/cli/ui/hooks/useToolStatus.js.map +0 -1
  552. package/dist/cli/ui/screens/BranchActionSelectorScreen.d.ts +0 -24
  553. package/dist/cli/ui/screens/BranchActionSelectorScreen.d.ts.map +0 -1
  554. package/dist/cli/ui/screens/BranchActionSelectorScreen.js +0 -65
  555. package/dist/cli/ui/screens/BranchActionSelectorScreen.js.map +0 -1
  556. package/dist/client/assets/index-LNPtOrn3.js +0 -78
  557. package/src/cli/ui/__tests__/SKIPPED_TESTS.md +0 -119
  558. package/src/cli/ui/__tests__/acceptance/branchList.acceptance.test.tsx.skip +0 -239
  559. package/src/cli/ui/__tests__/acceptance/navigation.acceptance.test.tsx +0 -225
  560. package/src/cli/ui/__tests__/acceptance/realtimeUpdate.acceptance.test.tsx.skip +0 -219
  561. package/src/cli/ui/__tests__/components/App.protected-branch.test.tsx +0 -212
  562. package/src/cli/ui/__tests__/components/App.shortcuts.test.tsx +0 -440
  563. package/src/cli/ui/__tests__/components/App.test.tsx +0 -365
  564. package/src/cli/ui/__tests__/components/ModelSelectorScreen.initial.test.tsx +0 -91
  565. package/src/cli/ui/__tests__/components/common/Confirm.test.tsx +0 -80
  566. package/src/cli/ui/__tests__/components/common/ErrorBoundary.test.tsx +0 -104
  567. package/src/cli/ui/__tests__/components/common/Input.test.tsx +0 -100
  568. package/src/cli/ui/__tests__/components/common/LoadingIndicator.test.tsx +0 -148
  569. package/src/cli/ui/__tests__/components/common/Select.memo.test.tsx +0 -255
  570. package/src/cli/ui/__tests__/components/common/Select.test.tsx +0 -335
  571. package/src/cli/ui/__tests__/components/parts/Footer.test.tsx +0 -65
  572. package/src/cli/ui/__tests__/components/parts/Header.test.tsx +0 -55
  573. package/src/cli/ui/__tests__/components/parts/ScrollableList.test.tsx +0 -69
  574. package/src/cli/ui/__tests__/components/parts/Stats.test.tsx +0 -148
  575. package/src/cli/ui/__tests__/components/screens/BranchCreatorScreen.test.tsx +0 -253
  576. package/src/cli/ui/__tests__/components/screens/BranchListScreen.test.tsx +0 -1070
  577. package/src/cli/ui/__tests__/components/screens/BranchQuickStartScreen.test.tsx +0 -142
  578. package/src/cli/ui/__tests__/components/screens/CodingAgentSelectorScreen.test.tsx +0 -174
  579. package/src/cli/ui/__tests__/components/screens/ExecutionModeSelectorScreen.test.tsx +0 -182
  580. package/src/cli/ui/__tests__/components/screens/LogDetailScreen.test.tsx +0 -57
  581. package/src/cli/ui/__tests__/components/screens/LogListScreen.test.tsx +0 -102
  582. package/src/cli/ui/__tests__/components/screens/PRCleanupScreen.test.tsx +0 -216
  583. package/src/cli/ui/__tests__/components/screens/SessionSelectorScreen.test.tsx +0 -147
  584. package/src/cli/ui/__tests__/hooks/useGitData.nonblocking.test.tsx +0 -206
  585. package/src/cli/ui/__tests__/hooks/useGitData.test.ts +0 -197
  586. package/src/cli/ui/__tests__/hooks/useGitData.test.ts.skip +0 -228
  587. package/src/cli/ui/__tests__/hooks/useScreenState.test.ts +0 -147
  588. package/src/cli/ui/__tests__/hooks/useTerminalSize.test.ts +0 -99
  589. package/src/cli/ui/__tests__/integration/branchList.test.tsx.skip +0 -253
  590. package/src/cli/ui/__tests__/integration/edgeCases.test.tsx +0 -436
  591. package/src/cli/ui/__tests__/integration/navigation.test.tsx +0 -514
  592. package/src/cli/ui/__tests__/integration/realtimeUpdate.test.tsx +0 -509
  593. package/src/cli/ui/__tests__/integration/realtimeUpdate.test.tsx.skip +0 -216
  594. package/src/cli/ui/__tests__/performance/branchList.performance.test.tsx +0 -193
  595. package/src/cli/ui/__tests__/performance/useMemoOptimization.test.tsx +0 -234
  596. package/src/cli/ui/components/App.tsx +0 -1478
  597. package/src/cli/ui/components/common/Confirm.tsx +0 -44
  598. package/src/cli/ui/components/common/ErrorBoundary.tsx +0 -60
  599. package/src/cli/ui/components/common/Input.tsx +0 -58
  600. package/src/cli/ui/components/common/LoadingIndicator.tsx +0 -98
  601. package/src/cli/ui/components/common/Select.tsx +0 -247
  602. package/src/cli/ui/components/common/SpinnerIcon.tsx +0 -86
  603. package/src/cli/ui/components/parts/Footer.tsx +0 -41
  604. package/src/cli/ui/components/parts/Header.test.tsx +0 -75
  605. package/src/cli/ui/components/parts/MergeStatusList.tsx +0 -75
  606. package/src/cli/ui/components/parts/ProgressBar.tsx +0 -73
  607. package/src/cli/ui/components/parts/Stats.tsx +0 -88
  608. package/src/cli/ui/components/screens/BatchMergeProgressScreen.tsx +0 -74
  609. package/src/cli/ui/components/screens/BatchMergeResultScreen.tsx +0 -108
  610. package/src/cli/ui/components/screens/BranchCreatorScreen.tsx +0 -242
  611. package/src/cli/ui/components/screens/BranchListScreen.tsx +0 -744
  612. package/src/cli/ui/components/screens/BranchQuickStartScreen.tsx +0 -244
  613. package/src/cli/ui/components/screens/CodingAgentSelectorScreen.tsx +0 -159
  614. package/src/cli/ui/components/screens/EnvironmentProfileScreen.tsx +0 -928
  615. package/src/cli/ui/components/screens/ExecutionModeSelectorScreen.tsx +0 -176
  616. package/src/cli/ui/components/screens/LogDatePickerScreen.tsx +0 -83
  617. package/src/cli/ui/components/screens/LogDetailScreen.tsx +0 -67
  618. package/src/cli/ui/components/screens/LogListScreen.tsx +0 -192
  619. package/src/cli/ui/components/screens/ModelSelectorScreen.tsx +0 -320
  620. package/src/cli/ui/components/screens/PRCleanupScreen.tsx +0 -171
  621. package/src/cli/ui/components/screens/SessionSelectorScreen.tsx +0 -135
  622. package/src/cli/ui/hooks/useAppInput.ts +0 -172
  623. package/src/cli/ui/hooks/useBatchMerge.ts +0 -96
  624. package/src/cli/ui/hooks/useGitData.ts +0 -347
  625. package/src/cli/ui/hooks/useProfiles.ts +0 -211
  626. package/src/cli/ui/hooks/useScreenState.ts +0 -44
  627. package/src/cli/ui/hooks/useTerminalSize.ts +0 -33
  628. package/src/cli/ui/hooks/useToolStatus.ts +0 -68
  629. package/src/cli/ui/screens/BranchActionSelectorScreen.tsx +0 -111
  630. package/src/cli/ui/screens/__tests__/BranchActionSelectorScreen.test.tsx +0 -264
@@ -0,0 +1,1084 @@
1
+ /** @jsxImportSource @opentui/solid */
2
+ import { describe, expect, it, mock, afterEach } from "bun:test";
3
+ import { testRender } from "@opentui/solid";
4
+ import type { BranchItem, Statistics } from "../../types.js";
5
+
6
+ if (!mock.module) {
7
+ mock.module = mock.module.bind(vi);
8
+ }
9
+
10
+ const makeStats = (overrides: Partial<Statistics> = {}): Statistics => ({
11
+ localCount: 0,
12
+ remoteCount: 0,
13
+ worktreeCount: 0,
14
+ changesCount: 0,
15
+ lastUpdated: new Date("2025-01-01T00:00:00Z"),
16
+ ...overrides,
17
+ });
18
+
19
+ const createBranch = (overrides: Partial<BranchItem> = {}): BranchItem => {
20
+ const base: BranchItem = {
21
+ name: "feature/cleanup-target",
22
+ type: "local",
23
+ branchType: "feature",
24
+ isCurrent: false,
25
+ icons: [],
26
+ hasChanges: false,
27
+ label: "feature/cleanup-target",
28
+ value: "feature/cleanup-target",
29
+ worktreeStatus: "active",
30
+ worktree: {
31
+ path: "/tmp/worktree",
32
+ locked: false,
33
+ prunable: false,
34
+ isAccessible: true,
35
+ },
36
+ mergedPR: { number: 1, mergedAt: "2025-01-01T00:00:00Z" },
37
+ hasUnpushedCommits: false,
38
+ syncStatus: "no-upstream",
39
+ };
40
+ return {
41
+ ...base,
42
+ ...overrides,
43
+ worktreeStatus: overrides.worktreeStatus ?? base.worktreeStatus,
44
+ };
45
+ };
46
+
47
+ afterEach(() => {
48
+ mock.restore();
49
+ mock.restore();
50
+ });
51
+
52
+ describe("AppSolid cleanup command", () => {
53
+ it("runs cleanup for selected branches when pressing c", async () => {
54
+ const deleteBranchMock = mock(async () => {});
55
+ const removeWorktreeMock = mock(async () => {});
56
+ const getCleanupStatusMock = mock(async () => [
57
+ {
58
+ worktreePath: "/tmp/worktree",
59
+ branch: "feature/cleanup-target",
60
+ hasUncommittedChanges: false,
61
+ hasUnpushedCommits: false,
62
+ cleanupType: "worktree-and-branch",
63
+ hasRemoteBranch: true,
64
+ hasUniqueCommits: false,
65
+ hasUpstream: true,
66
+ upstream: "origin/feature/cleanup-target",
67
+ isAccessible: true,
68
+ reasons: ["no-diff-with-base"],
69
+ },
70
+ ]);
71
+
72
+ mock.module?.("../../../../worktree.js", () => ({
73
+ listAdditionalWorktrees: mock(async () => []),
74
+ repairWorktrees: mock(async () => ({
75
+ repairedCount: 0,
76
+ failedCount: 0,
77
+ failures: [],
78
+ })),
79
+ removeWorktree: removeWorktreeMock,
80
+ getCleanupStatus: getCleanupStatusMock,
81
+ isProtectedBranchName: mock(() => false),
82
+ }));
83
+
84
+ mock.module?.("../../../../git.js", () => ({
85
+ getRepositoryRoot: mock(async () => "/repo"),
86
+ getAllBranches: mock(async () => []),
87
+ getLocalBranches: mock(async () => []),
88
+ getCurrentBranch: mock(async () => "main"),
89
+ deleteBranch: deleteBranchMock,
90
+ }));
91
+
92
+ mock.module?.("../../../../config/index.js", () => ({
93
+ getConfig: mock(async () => ({ defaultBaseBranch: "main" })),
94
+ getLastToolUsageMap: mock(async () => new Map()),
95
+ loadSession: mock(async () => null),
96
+ }));
97
+
98
+ mock.module?.("../../../../config/tools.js", () => ({
99
+ getAllCodingAgents: mock(async () => [
100
+ { id: "codex-cli", displayName: "Codex CLI" },
101
+ ]),
102
+ }));
103
+
104
+ mock.module?.("../../../../config/profiles.js", () => ({
105
+ loadProfiles: mock(async () => ({ profiles: {}, activeProfile: null })),
106
+ createProfile: mock(async () => {}),
107
+ updateProfile: mock(async () => {}),
108
+ deleteProfile: mock(async () => {}),
109
+ setActiveProfile: mock(async () => {}),
110
+ }));
111
+
112
+ const { AppSolid } = await import("../../App.solid.js");
113
+
114
+ const branch = createBranch();
115
+ const stats = makeStats({ localCount: 1, worktreeCount: 1 });
116
+
117
+ const testSetup = await testRender(
118
+ () => (
119
+ <AppSolid
120
+ branches={[branch]}
121
+ stats={stats}
122
+ version={null}
123
+ toolStatuses={[]}
124
+ />
125
+ ),
126
+ { width: 80, height: 24 },
127
+ );
128
+ await testSetup.renderOnce();
129
+
130
+ try {
131
+ await testSetup.mockInput.typeText(" ");
132
+ await testSetup.renderOnce();
133
+
134
+ await testSetup.mockInput.typeText("c");
135
+ await testSetup.renderOnce();
136
+ await new Promise((resolve) => setTimeout(resolve, 0));
137
+
138
+ expect(removeWorktreeMock).toHaveBeenCalledWith("/tmp/worktree", false);
139
+ expect(deleteBranchMock).toHaveBeenCalledWith(
140
+ "feature/cleanup-target",
141
+ true,
142
+ );
143
+ } finally {
144
+ testSetup.renderer.destroy();
145
+ }
146
+ });
147
+
148
+ it("maps cleanup safety indicators from cleanup candidates", async () => {
149
+ const getCleanupStatusMock = mock(async () => [
150
+ {
151
+ worktreePath: "/tmp/safe",
152
+ branch: "feature/safe",
153
+ hasUncommittedChanges: false,
154
+ hasUnpushedCommits: false,
155
+ cleanupType: "worktree-and-branch",
156
+ hasRemoteBranch: true,
157
+ hasUniqueCommits: false,
158
+ hasUpstream: true,
159
+ upstream: "origin/feature/safe",
160
+ isAccessible: true,
161
+ reasons: ["no-diff-with-base"],
162
+ },
163
+ {
164
+ worktreePath: null,
165
+ branch: "feature/unsafe",
166
+ hasUncommittedChanges: false,
167
+ hasUnpushedCommits: false,
168
+ cleanupType: "branch-only",
169
+ hasRemoteBranch: false,
170
+ hasUniqueCommits: false,
171
+ hasUpstream: false,
172
+ upstream: null,
173
+ reasons: [],
174
+ },
175
+ ]);
176
+
177
+ mock.module?.("../../../../worktree.js", () => ({
178
+ listAdditionalWorktrees: mock(async () => []),
179
+ repairWorktrees: mock(async () => ({
180
+ repairedCount: 0,
181
+ failedCount: 0,
182
+ failures: [],
183
+ })),
184
+ removeWorktree: mock(async () => {}),
185
+ getCleanupStatus: getCleanupStatusMock,
186
+ isProtectedBranchName: mock(() => false),
187
+ }));
188
+
189
+ mock.module?.("../../../../git.js", () => ({
190
+ getRepositoryRoot: mock(async () => "/repo"),
191
+ getAllBranches: mock(async () => []),
192
+ getLocalBranches: mock(async () => []),
193
+ getCurrentBranch: mock(async () => "main"),
194
+ deleteBranch: mock(async () => {}),
195
+ }));
196
+
197
+ mock.module?.("../../../../config/index.js", () => ({
198
+ getConfig: mock(async () => ({ defaultBaseBranch: "main" })),
199
+ getLastToolUsageMap: mock(async () => new Map()),
200
+ loadSession: mock(async () => null),
201
+ }));
202
+
203
+ mock.module?.("../../../../config/tools.js", () => ({
204
+ getAllCodingAgents: mock(async () => [
205
+ { id: "codex-cli", displayName: "Codex CLI" },
206
+ ]),
207
+ }));
208
+
209
+ mock.module?.("../../../../config/profiles.js", () => ({
210
+ loadProfiles: mock(async () => ({ profiles: {}, activeProfile: null })),
211
+ createProfile: mock(async () => {}),
212
+ updateProfile: mock(async () => {}),
213
+ deleteProfile: mock(async () => {}),
214
+ setActiveProfile: mock(async () => {}),
215
+ }));
216
+
217
+ const { AppSolid } = await import("../../App.solid.js");
218
+
219
+ const safeBranch = createBranch({
220
+ name: "feature/safe",
221
+ label: "feature/safe",
222
+ value: "feature/safe",
223
+ });
224
+ const unsafeBranch = createBranch({
225
+ name: "feature/unsafe",
226
+ label: "feature/unsafe",
227
+ value: "feature/unsafe",
228
+ worktree: undefined,
229
+ worktreeStatus: undefined,
230
+ });
231
+
232
+ const testSetup = await testRender(
233
+ () => (
234
+ <AppSolid
235
+ branches={[safeBranch, unsafeBranch]}
236
+ stats={makeStats({ localCount: 2, worktreeCount: 1 })}
237
+ version={null}
238
+ toolStatuses={[]}
239
+ />
240
+ ),
241
+ { width: 80, height: 24 },
242
+ );
243
+ await testSetup.renderOnce();
244
+ await new Promise((resolve) => setTimeout(resolve, 0));
245
+ await testSetup.renderOnce();
246
+
247
+ try {
248
+ const frame = testSetup.captureCharFrame();
249
+ expect(frame).toMatch(/\[ \] w o feature\/safe/);
250
+ expect(frame).toContain("[ ] w ! feature/unsafe");
251
+ } finally {
252
+ testSetup.renderer.destroy();
253
+ }
254
+ });
255
+
256
+ it("updates safety icons as each branch check completes", async () => {
257
+ let releaseSecond: (() => void) | null = null;
258
+ const progressGate = new Promise<void>((resolve) => {
259
+ releaseSecond = resolve;
260
+ });
261
+ const getCleanupStatusMock = mock(
262
+ async ({
263
+ onProgress,
264
+ }: { onProgress?: (status: { branch: string }) => void } = {}) => {
265
+ const firstStatus = {
266
+ worktreePath: "/tmp/first",
267
+ branch: "feature/first",
268
+ hasUncommittedChanges: false,
269
+ hasUnpushedCommits: false,
270
+ cleanupType: "worktree-and-branch",
271
+ hasRemoteBranch: true,
272
+ hasUniqueCommits: false,
273
+ hasUpstream: true,
274
+ upstream: "origin/feature/first",
275
+ isAccessible: true,
276
+ reasons: ["no-diff-with-base"],
277
+ };
278
+ const secondStatus = {
279
+ worktreePath: "/tmp/second",
280
+ branch: "feature/second",
281
+ hasUncommittedChanges: false,
282
+ hasUnpushedCommits: false,
283
+ cleanupType: "worktree-and-branch",
284
+ hasRemoteBranch: true,
285
+ hasUniqueCommits: true,
286
+ hasUpstream: true,
287
+ upstream: "origin/feature/second",
288
+ isAccessible: true,
289
+ reasons: ["remote-synced"],
290
+ };
291
+
292
+ onProgress?.(firstStatus);
293
+ await progressGate;
294
+ onProgress?.(secondStatus);
295
+ return [firstStatus, secondStatus];
296
+ },
297
+ );
298
+
299
+ mock.module?.("../../../../worktree.js", () => ({
300
+ listAdditionalWorktrees: mock(async () => []),
301
+ repairWorktrees: mock(async () => ({
302
+ repairedCount: 0,
303
+ failedCount: 0,
304
+ failures: [],
305
+ })),
306
+ removeWorktree: mock(async () => {}),
307
+ getCleanupStatus: getCleanupStatusMock,
308
+ isProtectedBranchName: mock(() => false),
309
+ }));
310
+
311
+ mock.module?.("../../../../git.js", () => ({
312
+ getRepositoryRoot: mock(async () => "/repo"),
313
+ getAllBranches: mock(async () => []),
314
+ getLocalBranches: mock(async () => []),
315
+ getCurrentBranch: mock(async () => "main"),
316
+ deleteBranch: mock(async () => {}),
317
+ }));
318
+
319
+ mock.module?.("../../../../config/index.js", () => ({
320
+ getConfig: mock(async () => ({ defaultBaseBranch: "main" })),
321
+ getLastToolUsageMap: mock(async () => new Map()),
322
+ loadSession: mock(async () => null),
323
+ }));
324
+
325
+ mock.module?.("../../../../config/tools.js", () => ({
326
+ getAllCodingAgents: mock(async () => [
327
+ { id: "codex-cli", displayName: "Codex CLI" },
328
+ ]),
329
+ }));
330
+
331
+ mock.module?.("../../../../config/profiles.js", () => ({
332
+ loadProfiles: mock(async () => ({ profiles: {}, activeProfile: null })),
333
+ createProfile: mock(async () => {}),
334
+ updateProfile: mock(async () => {}),
335
+ deleteProfile: mock(async () => {}),
336
+ setActiveProfile: mock(async () => {}),
337
+ }));
338
+
339
+ const { AppSolid } = await import("../../App.solid.js");
340
+
341
+ const firstBranch = createBranch({
342
+ name: "feature/first",
343
+ label: "feature/first",
344
+ value: "feature/first",
345
+ worktree: { path: "/tmp/first", locked: false, prunable: false },
346
+ });
347
+ const secondBranch = createBranch({
348
+ name: "feature/second",
349
+ label: "feature/second",
350
+ value: "feature/second",
351
+ worktree: { path: "/tmp/second", locked: false, prunable: false },
352
+ });
353
+
354
+ const testSetup = await testRender(
355
+ () => (
356
+ <AppSolid
357
+ branches={[firstBranch, secondBranch]}
358
+ stats={makeStats({ localCount: 2, worktreeCount: 2 })}
359
+ version={null}
360
+ toolStatuses={[]}
361
+ />
362
+ ),
363
+ { width: 80, height: 24 },
364
+ );
365
+ await testSetup.renderOnce();
366
+ await new Promise((resolve) => setTimeout(resolve, 0));
367
+ await testSetup.renderOnce();
368
+
369
+ try {
370
+ let frame = testSetup.captureCharFrame();
371
+ expect(frame).toMatch(/\[ \] w o feature\/first/);
372
+ expect(frame).toMatch(/\[ \] w [-\\|/] feature\/second/);
373
+
374
+ releaseSecond?.();
375
+ await new Promise((resolve) => setTimeout(resolve, 0));
376
+ await testSetup.renderOnce();
377
+
378
+ frame = testSetup.captureCharFrame();
379
+ expect(frame).toMatch(/\[ \] w \* feature\/second/);
380
+ } finally {
381
+ testSetup.renderer.destroy();
382
+ }
383
+ });
384
+ });
385
+
386
+ describe("AppSolid unsafe selection confirm", () => {
387
+ it("shows confirm and cancels selection on Cancel", async () => {
388
+ const getCleanupStatusMock = mock(async () => [
389
+ {
390
+ worktreePath: "/tmp/worktree",
391
+ branch: "feature/unsafe",
392
+ hasUncommittedChanges: false,
393
+ hasUnpushedCommits: true,
394
+ cleanupType: "worktree-and-branch",
395
+ hasRemoteBranch: true,
396
+ hasUniqueCommits: false,
397
+ hasUpstream: true,
398
+ upstream: "origin/feature/unsafe",
399
+ isAccessible: true,
400
+ reasons: ["no-diff-with-base"],
401
+ },
402
+ ]);
403
+
404
+ mock.module?.("../../../../worktree.js", () => ({
405
+ listAdditionalWorktrees: mock(async () => []),
406
+ repairWorktrees: mock(async () => ({
407
+ repairedCount: 0,
408
+ failedCount: 0,
409
+ failures: [],
410
+ })),
411
+ removeWorktree: mock(async () => {}),
412
+ getCleanupStatus: getCleanupStatusMock,
413
+ isProtectedBranchName: mock(() => false),
414
+ }));
415
+
416
+ mock.module?.("../../../../git.js", () => ({
417
+ getRepositoryRoot: mock(async () => "/repo"),
418
+ getAllBranches: mock(async () => []),
419
+ getLocalBranches: mock(async () => []),
420
+ getCurrentBranch: mock(async () => "main"),
421
+ deleteBranch: mock(async () => {}),
422
+ }));
423
+
424
+ mock.module?.("../../../../config/index.js", () => ({
425
+ getConfig: mock(async () => ({ defaultBaseBranch: "main" })),
426
+ getLastToolUsageMap: mock(async () => new Map()),
427
+ loadSession: mock(async () => null),
428
+ }));
429
+
430
+ mock.module?.("../../../../config/tools.js", () => ({
431
+ getAllCodingAgents: mock(async () => [
432
+ { id: "codex-cli", displayName: "Codex CLI" },
433
+ ]),
434
+ }));
435
+
436
+ mock.module?.("../../../../config/profiles.js", () => ({
437
+ loadProfiles: mock(async () => ({ profiles: {}, activeProfile: null })),
438
+ createProfile: mock(async () => {}),
439
+ updateProfile: mock(async () => {}),
440
+ deleteProfile: mock(async () => {}),
441
+ setActiveProfile: mock(async () => {}),
442
+ }));
443
+
444
+ const { AppSolid } = await import("../../App.solid.js");
445
+
446
+ const branch = createBranch({
447
+ name: "feature/unsafe",
448
+ label: "feature/unsafe",
449
+ value: "feature/unsafe",
450
+ });
451
+ const stats = makeStats({ localCount: 1, worktreeCount: 1 });
452
+
453
+ const testSetup = await testRender(
454
+ () => (
455
+ <AppSolid
456
+ branches={[branch]}
457
+ stats={stats}
458
+ version={null}
459
+ toolStatuses={[]}
460
+ />
461
+ ),
462
+ { width: 80, height: 24 },
463
+ );
464
+ await testSetup.renderOnce();
465
+ await new Promise((resolve) => setTimeout(resolve, 0));
466
+ await testSetup.renderOnce();
467
+
468
+ try {
469
+ await testSetup.mockInput.typeText(" ");
470
+ await testSetup.renderOnce();
471
+
472
+ let frame = testSetup.captureCharFrame();
473
+ expect(frame).toContain("Unsafe branch selected. Select anyway?");
474
+ expect(frame).toContain("OK");
475
+ expect(frame).toContain("Cancel");
476
+
477
+ await testSetup.mockInput.typeText("n");
478
+ await testSetup.renderOnce();
479
+
480
+ frame = testSetup.captureCharFrame();
481
+ expect(frame).toContain("[ ] w");
482
+ expect(frame).toContain("feature/unsafe");
483
+ expect(frame).not.toContain("Unsafe branch selected. Select anyway?");
484
+ } finally {
485
+ testSetup.renderer.destroy();
486
+ }
487
+ });
488
+
489
+ it("does not propagate Enter from confirm to branch selection", async () => {
490
+ const getCleanupStatusMock = mock(async () => [
491
+ {
492
+ worktreePath: "/tmp/worktree",
493
+ branch: "feature/unsafe-enter",
494
+ hasUncommittedChanges: false,
495
+ hasUnpushedCommits: true,
496
+ cleanupType: "worktree-and-branch",
497
+ hasRemoteBranch: true,
498
+ hasUniqueCommits: false,
499
+ hasUpstream: true,
500
+ upstream: "origin/feature/unsafe-enter",
501
+ isAccessible: true,
502
+ reasons: ["no-diff-with-base"],
503
+ },
504
+ ]);
505
+
506
+ mock.module?.("../../../../worktree.js", () => ({
507
+ listAdditionalWorktrees: mock(async () => []),
508
+ repairWorktrees: mock(async () => ({
509
+ repairedCount: 0,
510
+ failedCount: 0,
511
+ failures: [],
512
+ })),
513
+ removeWorktree: mock(async () => {}),
514
+ getCleanupStatus: getCleanupStatusMock,
515
+ isProtectedBranchName: mock(() => false),
516
+ }));
517
+
518
+ mock.module?.("../../../../git.js", () => ({
519
+ getRepositoryRoot: mock(async () => "/repo"),
520
+ getAllBranches: mock(async () => []),
521
+ getLocalBranches: mock(async () => []),
522
+ getCurrentBranch: mock(async () => "main"),
523
+ deleteBranch: mock(async () => {}),
524
+ }));
525
+
526
+ mock.module?.("../../../../config/index.js", () => ({
527
+ getConfig: mock(async () => ({ defaultBaseBranch: "main" })),
528
+ getLastToolUsageMap: mock(async () => new Map()),
529
+ loadSession: mock(async () => null),
530
+ }));
531
+
532
+ mock.module?.("../../../../config/tools.js", () => ({
533
+ getAllCodingAgents: mock(async () => [
534
+ { id: "codex-cli", displayName: "Codex CLI" },
535
+ ]),
536
+ }));
537
+
538
+ mock.module?.("../../../../config/profiles.js", () => ({
539
+ loadProfiles: mock(async () => ({ profiles: {}, activeProfile: null })),
540
+ createProfile: mock(async () => {}),
541
+ updateProfile: mock(async () => {}),
542
+ deleteProfile: mock(async () => {}),
543
+ setActiveProfile: mock(async () => {}),
544
+ }));
545
+
546
+ const { AppSolid } = await import("../../App.solid.js");
547
+
548
+ const branch = createBranch({
549
+ name: "feature/unsafe-enter",
550
+ label: "feature/unsafe-enter",
551
+ value: "feature/unsafe-enter",
552
+ });
553
+ const stats = makeStats({ localCount: 1, worktreeCount: 1 });
554
+
555
+ const testSetup = await testRender(
556
+ () => (
557
+ <AppSolid
558
+ branches={[branch]}
559
+ stats={stats}
560
+ version={null}
561
+ toolStatuses={[]}
562
+ />
563
+ ),
564
+ { width: 80, height: 24 },
565
+ );
566
+ await testSetup.renderOnce();
567
+ await new Promise((resolve) => setTimeout(resolve, 0));
568
+ await testSetup.renderOnce();
569
+
570
+ try {
571
+ await testSetup.mockInput.typeText(" ");
572
+ await testSetup.renderOnce();
573
+
574
+ testSetup.mockInput.pressEnter();
575
+ await testSetup.renderOnce();
576
+
577
+ const frame = testSetup.captureCharFrame();
578
+ expect(frame).not.toContain("Unsafe branch selected. Select anyway?");
579
+ expect(frame).toContain("[ ] w");
580
+ expect(frame).not.toContain("Open existing worktree");
581
+ } finally {
582
+ testSetup.renderer.destroy();
583
+ }
584
+ });
585
+
586
+ it("selects unsafe branch on OK", async () => {
587
+ const getCleanupStatusMock = mock(async () => [
588
+ {
589
+ worktreePath: "/tmp/worktree",
590
+ branch: "feature/unsafe-ok",
591
+ hasUncommittedChanges: false,
592
+ hasUnpushedCommits: true,
593
+ cleanupType: "worktree-and-branch",
594
+ hasRemoteBranch: true,
595
+ hasUniqueCommits: false,
596
+ hasUpstream: true,
597
+ upstream: "origin/feature/unsafe-ok",
598
+ isAccessible: true,
599
+ reasons: ["no-diff-with-base"],
600
+ },
601
+ ]);
602
+
603
+ mock.module?.("../../../../worktree.js", () => ({
604
+ listAdditionalWorktrees: mock(async () => []),
605
+ repairWorktrees: mock(async () => ({
606
+ repairedCount: 0,
607
+ failedCount: 0,
608
+ failures: [],
609
+ })),
610
+ removeWorktree: mock(async () => {}),
611
+ getCleanupStatus: getCleanupStatusMock,
612
+ isProtectedBranchName: mock(() => false),
613
+ }));
614
+
615
+ mock.module?.("../../../../git.js", () => ({
616
+ getRepositoryRoot: mock(async () => "/repo"),
617
+ getAllBranches: mock(async () => []),
618
+ getLocalBranches: mock(async () => []),
619
+ getCurrentBranch: mock(async () => "main"),
620
+ deleteBranch: mock(async () => {}),
621
+ }));
622
+
623
+ mock.module?.("../../../../config/index.js", () => ({
624
+ getConfig: mock(async () => ({ defaultBaseBranch: "main" })),
625
+ getLastToolUsageMap: mock(async () => new Map()),
626
+ loadSession: mock(async () => null),
627
+ }));
628
+
629
+ mock.module?.("../../../../config/tools.js", () => ({
630
+ getAllCodingAgents: mock(async () => [
631
+ { id: "codex-cli", displayName: "Codex CLI" },
632
+ ]),
633
+ }));
634
+
635
+ mock.module?.("../../../../config/profiles.js", () => ({
636
+ loadProfiles: mock(async () => ({ profiles: {}, activeProfile: null })),
637
+ createProfile: mock(async () => {}),
638
+ updateProfile: mock(async () => {}),
639
+ deleteProfile: mock(async () => {}),
640
+ setActiveProfile: mock(async () => {}),
641
+ }));
642
+
643
+ const { AppSolid } = await import("../../App.solid.js");
644
+
645
+ const branch = createBranch({
646
+ name: "feature/unsafe-ok",
647
+ label: "feature/unsafe-ok",
648
+ value: "feature/unsafe-ok",
649
+ });
650
+ const stats = makeStats({ localCount: 1, worktreeCount: 1 });
651
+
652
+ const testSetup = await testRender(
653
+ () => (
654
+ <AppSolid
655
+ branches={[branch]}
656
+ stats={stats}
657
+ version={null}
658
+ toolStatuses={[]}
659
+ />
660
+ ),
661
+ { width: 80, height: 24 },
662
+ );
663
+ await testSetup.renderOnce();
664
+ await new Promise((resolve) => setTimeout(resolve, 0));
665
+ await testSetup.renderOnce();
666
+
667
+ try {
668
+ await testSetup.mockInput.typeText(" ");
669
+ await testSetup.renderOnce();
670
+
671
+ await testSetup.mockInput.typeText("y");
672
+ await testSetup.renderOnce();
673
+
674
+ const frame = testSetup.captureCharFrame();
675
+ expect(frame).toContain("[*]");
676
+ expect(frame).toContain("feature/unsafe-ok");
677
+ expect(frame).not.toContain("Unsafe branch selected. Select anyway?");
678
+ } finally {
679
+ testSetup.renderer.destroy();
680
+ }
681
+ });
682
+ });
683
+
684
+ describe("AppSolid selected cleanup targets", () => {
685
+ it("cleans unsafe branch when confirmed and selected", async () => {
686
+ const deleteBranchMock = mock(async () => {});
687
+ const removeWorktreeMock = mock(async () => {});
688
+ const getCleanupStatusMock = mock(async () => [
689
+ {
690
+ worktreePath: "/tmp/worktree",
691
+ branch: "feature/unsafe-clean",
692
+ hasUncommittedChanges: false,
693
+ hasUnpushedCommits: true,
694
+ cleanupType: "worktree-and-branch",
695
+ hasRemoteBranch: true,
696
+ hasUniqueCommits: false,
697
+ hasUpstream: true,
698
+ upstream: "origin/feature/unsafe-clean",
699
+ isAccessible: true,
700
+ reasons: ["no-diff-with-base"],
701
+ },
702
+ ]);
703
+
704
+ mock.module?.("../../../../worktree.js", () => ({
705
+ listAdditionalWorktrees: mock(async () => []),
706
+ repairWorktrees: mock(async () => ({
707
+ repairedCount: 0,
708
+ failedCount: 0,
709
+ failures: [],
710
+ })),
711
+ removeWorktree: removeWorktreeMock,
712
+ getCleanupStatus: getCleanupStatusMock,
713
+ isProtectedBranchName: mock(() => false),
714
+ }));
715
+
716
+ mock.module?.("../../../../git.js", () => ({
717
+ getRepositoryRoot: mock(async () => "/repo"),
718
+ getAllBranches: mock(async () => []),
719
+ getLocalBranches: mock(async () => []),
720
+ getCurrentBranch: mock(async () => "main"),
721
+ deleteBranch: deleteBranchMock,
722
+ }));
723
+
724
+ mock.module?.("../../../../config/index.js", () => ({
725
+ getConfig: mock(async () => ({ defaultBaseBranch: "main" })),
726
+ getLastToolUsageMap: mock(async () => new Map()),
727
+ loadSession: mock(async () => null),
728
+ }));
729
+
730
+ mock.module?.("../../../../config/tools.js", () => ({
731
+ getAllCodingAgents: mock(async () => [
732
+ { id: "codex-cli", displayName: "Codex CLI" },
733
+ ]),
734
+ }));
735
+
736
+ mock.module?.("../../../../config/profiles.js", () => ({
737
+ loadProfiles: mock(async () => ({ profiles: {}, activeProfile: null })),
738
+ createProfile: mock(async () => {}),
739
+ updateProfile: mock(async () => {}),
740
+ deleteProfile: mock(async () => {}),
741
+ setActiveProfile: mock(async () => {}),
742
+ }));
743
+
744
+ const { AppSolid } = await import("../../App.solid.js");
745
+
746
+ const branch = createBranch({
747
+ name: "feature/unsafe-clean",
748
+ label: "feature/unsafe-clean",
749
+ value: "feature/unsafe-clean",
750
+ });
751
+ const stats = makeStats({ localCount: 1, worktreeCount: 1 });
752
+
753
+ const testSetup = await testRender(
754
+ () => (
755
+ <AppSolid
756
+ branches={[branch]}
757
+ stats={stats}
758
+ version={null}
759
+ toolStatuses={[]}
760
+ />
761
+ ),
762
+ { width: 80, height: 24 },
763
+ );
764
+ await testSetup.renderOnce();
765
+ await new Promise((resolve) => setTimeout(resolve, 0));
766
+ await testSetup.renderOnce();
767
+
768
+ try {
769
+ await testSetup.mockInput.typeText(" ");
770
+ await testSetup.renderOnce();
771
+
772
+ await testSetup.mockInput.typeText("y");
773
+ await testSetup.renderOnce();
774
+
775
+ await testSetup.mockInput.typeText("c");
776
+ await testSetup.renderOnce();
777
+ await new Promise((resolve) => setTimeout(resolve, 0));
778
+
779
+ expect(removeWorktreeMock).toHaveBeenCalledWith("/tmp/worktree", false);
780
+ expect(deleteBranchMock).toHaveBeenCalledWith(
781
+ "feature/unsafe-clean",
782
+ true,
783
+ );
784
+ } finally {
785
+ testSetup.renderer.destroy();
786
+ }
787
+ });
788
+
789
+ it("includes protected branch when selected", async () => {
790
+ const deleteBranchMock = mock(async () => {});
791
+ const removeWorktreeMock = mock(async () => {});
792
+ const getCleanupStatusMock = mock(async () => [
793
+ {
794
+ worktreePath: "/tmp/worktree",
795
+ branch: "develop",
796
+ hasUncommittedChanges: false,
797
+ hasUnpushedCommits: false,
798
+ cleanupType: "worktree-and-branch",
799
+ hasRemoteBranch: true,
800
+ hasUniqueCommits: false,
801
+ hasUpstream: true,
802
+ upstream: "origin/develop",
803
+ isAccessible: true,
804
+ reasons: ["no-diff-with-base"],
805
+ },
806
+ ]);
807
+
808
+ mock.module?.("../../../../worktree.js", () => ({
809
+ listAdditionalWorktrees: mock(async () => []),
810
+ repairWorktrees: mock(async () => ({
811
+ repairedCount: 0,
812
+ failedCount: 0,
813
+ failures: [],
814
+ })),
815
+ removeWorktree: removeWorktreeMock,
816
+ getCleanupStatus: getCleanupStatusMock,
817
+ isProtectedBranchName: mock(() => true),
818
+ }));
819
+
820
+ mock.module?.("../../../../git.js", () => ({
821
+ getRepositoryRoot: mock(async () => "/repo"),
822
+ getAllBranches: mock(async () => []),
823
+ getLocalBranches: mock(async () => []),
824
+ getCurrentBranch: mock(async () => "main"),
825
+ deleteBranch: deleteBranchMock,
826
+ }));
827
+
828
+ mock.module?.("../../../../config/index.js", () => ({
829
+ getConfig: mock(async () => ({ defaultBaseBranch: "main" })),
830
+ getLastToolUsageMap: mock(async () => new Map()),
831
+ loadSession: mock(async () => null),
832
+ }));
833
+
834
+ mock.module?.("../../../../config/tools.js", () => ({
835
+ getAllCodingAgents: mock(async () => [
836
+ { id: "codex-cli", displayName: "Codex CLI" },
837
+ ]),
838
+ }));
839
+
840
+ mock.module?.("../../../../config/profiles.js", () => ({
841
+ loadProfiles: mock(async () => ({ profiles: {}, activeProfile: null })),
842
+ createProfile: mock(async () => {}),
843
+ updateProfile: mock(async () => {}),
844
+ deleteProfile: mock(async () => {}),
845
+ setActiveProfile: mock(async () => {}),
846
+ }));
847
+
848
+ const { AppSolid } = await import("../../App.solid.js");
849
+
850
+ const branch = createBranch({
851
+ name: "develop",
852
+ label: "develop",
853
+ value: "develop",
854
+ });
855
+ const stats = makeStats({ localCount: 1, worktreeCount: 1 });
856
+
857
+ const testSetup = await testRender(
858
+ () => (
859
+ <AppSolid
860
+ branches={[branch]}
861
+ stats={stats}
862
+ version={null}
863
+ toolStatuses={[]}
864
+ />
865
+ ),
866
+ { width: 80, height: 24 },
867
+ );
868
+ await testSetup.renderOnce();
869
+ await new Promise((resolve) => setTimeout(resolve, 0));
870
+ await testSetup.renderOnce();
871
+
872
+ try {
873
+ await testSetup.mockInput.typeText(" ");
874
+ await testSetup.renderOnce();
875
+
876
+ await testSetup.mockInput.typeText("c");
877
+ await testSetup.renderOnce();
878
+ await new Promise((resolve) => setTimeout(resolve, 0));
879
+
880
+ expect(deleteBranchMock).toHaveBeenCalledWith("develop", true);
881
+ expect(removeWorktreeMock).toHaveBeenCalledWith("/tmp/worktree", false);
882
+ } finally {
883
+ testSetup.renderer.destroy();
884
+ }
885
+ });
886
+
887
+ it("excludes current branch even when selected", async () => {
888
+ const deleteBranchMock = mock(async () => {});
889
+ const removeWorktreeMock = mock(async () => {});
890
+ const getCleanupStatusMock = mock(async () => [
891
+ {
892
+ worktreePath: "/tmp/worktree",
893
+ branch: "main",
894
+ hasUncommittedChanges: false,
895
+ hasUnpushedCommits: false,
896
+ cleanupType: "worktree-and-branch",
897
+ hasRemoteBranch: true,
898
+ hasUniqueCommits: false,
899
+ hasUpstream: true,
900
+ upstream: "origin/main",
901
+ isAccessible: true,
902
+ reasons: ["no-diff-with-base"],
903
+ },
904
+ ]);
905
+
906
+ mock.module?.("../../../../worktree.js", () => ({
907
+ listAdditionalWorktrees: mock(async () => []),
908
+ repairWorktrees: mock(async () => ({
909
+ repairedCount: 0,
910
+ failedCount: 0,
911
+ failures: [],
912
+ })),
913
+ removeWorktree: removeWorktreeMock,
914
+ getCleanupStatus: getCleanupStatusMock,
915
+ isProtectedBranchName: mock(() => true),
916
+ }));
917
+
918
+ mock.module?.("../../../../git.js", () => ({
919
+ getRepositoryRoot: mock(async () => "/repo"),
920
+ getAllBranches: mock(async () => []),
921
+ getLocalBranches: mock(async () => []),
922
+ getCurrentBranch: mock(async () => "main"),
923
+ deleteBranch: deleteBranchMock,
924
+ }));
925
+
926
+ mock.module?.("../../../../config/index.js", () => ({
927
+ getConfig: mock(async () => ({ defaultBaseBranch: "main" })),
928
+ getLastToolUsageMap: mock(async () => new Map()),
929
+ loadSession: mock(async () => null),
930
+ }));
931
+
932
+ mock.module?.("../../../../config/tools.js", () => ({
933
+ getAllCodingAgents: mock(async () => [
934
+ { id: "codex-cli", displayName: "Codex CLI" },
935
+ ]),
936
+ }));
937
+
938
+ mock.module?.("../../../../config/profiles.js", () => ({
939
+ loadProfiles: mock(async () => ({ profiles: {}, activeProfile: null })),
940
+ createProfile: mock(async () => {}),
941
+ updateProfile: mock(async () => {}),
942
+ deleteProfile: mock(async () => {}),
943
+ setActiveProfile: mock(async () => {}),
944
+ }));
945
+
946
+ const { AppSolid } = await import("../../App.solid.js");
947
+
948
+ const branch = createBranch({
949
+ name: "main",
950
+ label: "main",
951
+ value: "main",
952
+ isCurrent: true,
953
+ });
954
+ const stats = makeStats({ localCount: 1, worktreeCount: 1 });
955
+
956
+ const testSetup = await testRender(
957
+ () => (
958
+ <AppSolid
959
+ branches={[branch]}
960
+ stats={stats}
961
+ version={null}
962
+ toolStatuses={[]}
963
+ />
964
+ ),
965
+ { width: 80, height: 24 },
966
+ );
967
+ await testSetup.renderOnce();
968
+ await new Promise((resolve) => setTimeout(resolve, 0));
969
+ await testSetup.renderOnce();
970
+
971
+ try {
972
+ await testSetup.mockInput.typeText(" ");
973
+ await testSetup.renderOnce();
974
+
975
+ await testSetup.mockInput.typeText("c");
976
+ await testSetup.renderOnce();
977
+ await new Promise((resolve) => setTimeout(resolve, 0));
978
+
979
+ expect(deleteBranchMock).not.toHaveBeenCalled();
980
+ expect(removeWorktreeMock).not.toHaveBeenCalled();
981
+ } finally {
982
+ testSetup.renderer.destroy();
983
+ }
984
+ });
985
+
986
+ it("repairs selected branch even when worktree is accessible", async () => {
987
+ const repairWorktreesMock = mock(async () => ({
988
+ repairedCount: 1,
989
+ failedCount: 0,
990
+ failures: [],
991
+ }));
992
+ const getCleanupStatusMock = mock(async () => [
993
+ {
994
+ worktreePath: "/tmp/worktree",
995
+ branch: "feature/repair-target",
996
+ hasUncommittedChanges: false,
997
+ hasUnpushedCommits: false,
998
+ cleanupType: "worktree-and-branch",
999
+ hasRemoteBranch: true,
1000
+ hasUniqueCommits: false,
1001
+ hasUpstream: true,
1002
+ upstream: "origin/feature/repair-target",
1003
+ isAccessible: true,
1004
+ reasons: ["no-diff-with-base"],
1005
+ },
1006
+ ]);
1007
+
1008
+ mock.module?.("../../../../worktree.js", () => ({
1009
+ listAdditionalWorktrees: mock(async () => []),
1010
+ repairWorktrees: repairWorktreesMock,
1011
+ removeWorktree: mock(async () => {}),
1012
+ getCleanupStatus: getCleanupStatusMock,
1013
+ isProtectedBranchName: mock(() => false),
1014
+ }));
1015
+
1016
+ mock.module?.("../../../../git.js", () => ({
1017
+ getRepositoryRoot: mock(async () => "/repo"),
1018
+ getAllBranches: mock(async () => []),
1019
+ getLocalBranches: mock(async () => []),
1020
+ getCurrentBranch: mock(async () => "main"),
1021
+ deleteBranch: mock(async () => {}),
1022
+ }));
1023
+
1024
+ mock.module?.("../../../../config/index.js", () => ({
1025
+ getConfig: mock(async () => ({ defaultBaseBranch: "main" })),
1026
+ getLastToolUsageMap: mock(async () => new Map()),
1027
+ loadSession: mock(async () => null),
1028
+ }));
1029
+
1030
+ mock.module?.("../../../../config/tools.js", () => ({
1031
+ getAllCodingAgents: mock(async () => [
1032
+ { id: "codex-cli", displayName: "Codex CLI" },
1033
+ ]),
1034
+ }));
1035
+
1036
+ mock.module?.("../../../../config/profiles.js", () => ({
1037
+ loadProfiles: mock(async () => ({ profiles: {}, activeProfile: null })),
1038
+ createProfile: mock(async () => {}),
1039
+ updateProfile: mock(async () => {}),
1040
+ deleteProfile: mock(async () => {}),
1041
+ setActiveProfile: mock(async () => {}),
1042
+ }));
1043
+
1044
+ const { AppSolid } = await import("../../App.solid.js");
1045
+
1046
+ const branch = createBranch({
1047
+ name: "feature/repair-target",
1048
+ label: "feature/repair-target",
1049
+ value: "feature/repair-target",
1050
+ worktreeStatus: "active",
1051
+ });
1052
+ const stats = makeStats({ localCount: 1, worktreeCount: 1 });
1053
+
1054
+ const testSetup = await testRender(
1055
+ () => (
1056
+ <AppSolid
1057
+ branches={[branch]}
1058
+ stats={stats}
1059
+ version={null}
1060
+ toolStatuses={[]}
1061
+ />
1062
+ ),
1063
+ { width: 80, height: 24 },
1064
+ );
1065
+ await testSetup.renderOnce();
1066
+ await new Promise((resolve) => setTimeout(resolve, 0));
1067
+ await testSetup.renderOnce();
1068
+
1069
+ try {
1070
+ await testSetup.mockInput.typeText(" ");
1071
+ await testSetup.renderOnce();
1072
+
1073
+ await testSetup.mockInput.typeText("x");
1074
+ await testSetup.renderOnce();
1075
+ await new Promise((resolve) => setTimeout(resolve, 0));
1076
+
1077
+ expect(repairWorktreesMock).toHaveBeenCalledWith([
1078
+ "feature/repair-target",
1079
+ ]);
1080
+ } finally {
1081
+ testSetup.renderer.destroy();
1082
+ }
1083
+ });
1084
+ });