@sleepy-ai/cli 0.1.6 → 0.1.8

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 (799) hide show
  1. package/{src/skill/compose/LICENSE-superpowers → LICENSE} +3 -6
  2. package/README.md +139 -6
  3. package/package.json +35 -199
  4. package/postinstall.mjs +102 -0
  5. package/.sleepycode-test-fixtures-50158/sleepycode-test-eqsxv84ythu/.watcher-b67v64k2mnv +0 -1
  6. package/Dockerfile +0 -18
  7. package/drizzle.config.ts +0 -10
  8. package/git +0 -0
  9. package/migration/20260127222353_familiar_lady_ursula/migration.sql +0 -90
  10. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +0 -796
  11. package/migration/20260211171708_add_project_commands/migration.sql +0 -1
  12. package/migration/20260211171708_add_project_commands/snapshot.json +0 -806
  13. package/migration/20260213144116_wakeful_the_professor/migration.sql +0 -11
  14. package/migration/20260213144116_wakeful_the_professor/snapshot.json +0 -897
  15. package/migration/20260225215848_workspace/migration.sql +0 -7
  16. package/migration/20260225215848_workspace/snapshot.json +0 -959
  17. package/migration/20260227213759_add_session_workspace_id/migration.sql +0 -2
  18. package/migration/20260227213759_add_session_workspace_id/snapshot.json +0 -983
  19. package/migration/20260228203230_blue_harpoon/migration.sql +0 -17
  20. package/migration/20260228203230_blue_harpoon/snapshot.json +0 -1102
  21. package/migration/20260303231226_add_workspace_fields/migration.sql +0 -5
  22. package/migration/20260303231226_add_workspace_fields/snapshot.json +0 -1013
  23. package/migration/20260309230000_move_org_to_state/migration.sql +0 -3
  24. package/migration/20260309230000_move_org_to_state/snapshot.json +0 -1156
  25. package/migration/20260312043431_session_message_cursor/migration.sql +0 -4
  26. package/migration/20260312043431_session_message_cursor/snapshot.json +0 -1168
  27. package/migration/20260323234822_events/migration.sql +0 -13
  28. package/migration/20260323234822_events/snapshot.json +0 -1271
  29. package/migration/20260410174513_workspace-name/migration.sql +0 -16
  30. package/migration/20260410174513_workspace-name/snapshot.json +0 -1271
  31. package/migration/20260413175956_chief_energizer/migration.sql +0 -13
  32. package/migration/20260413175956_chief_energizer/snapshot.json +0 -1399
  33. package/migration/20260422160000_context_inheritance/migration.sql +0 -3
  34. package/migration/20260422170000_task_registry/migration.sql +0 -18
  35. package/migration/20260423145421_remove_session_entry/migration.sql +0 -4
  36. package/migration/20260515000000_actor_rename/migration.sql +0 -7
  37. package/migration/20260515010000_memory_fts/migration.sql +0 -33
  38. package/migration/20260515020000_user_task/migration.sql +0 -29
  39. package/migration/20260519000000_last_checkpoint_message_id/migration.sql +0 -1
  40. package/migration/20260521000000_message_agent_id/migration.sql +0 -2
  41. package/migration/20260521000100_actor_registry_v6/migration.sql +0 -25
  42. package/migration/20260521010000_memory_fts_v6/migration.sql +0 -33
  43. package/migration/20260521020000_memory_fts_triggers/migration.sql +0 -17
  44. package/migration/20260526000000_agent_id_main/migration.sql +0 -14
  45. package/migration/20260527000000_actor_lifecycle/migration.sql +0 -8
  46. package/migration/20260527000100_inbox/migration.sql +0 -12
  47. package/migration/20260529000000_task_todo_redesign/migration.sql +0 -16
  48. package/migration/20260603000000_task_in_progress_owner/migration.sql +0 -1
  49. package/migration/20260603000000_workflow_run/migration.sql +0 -17
  50. package/migration/20260604000000_workflow_script_sha/migration.sql +0 -1
  51. package/migration/20260608000000_claude_import/migration.sql +0 -7
  52. package/migration/20260608010000_claude_import_message_ids/migration.sql +0 -1
  53. package/migration/20260609000000_history_fts/migration.sql +0 -29
  54. package/migration/20260609230000_workflow_agent_timeout/migration.sql +0 -1
  55. package/migration/20260612000000_external_import/migration.sql +0 -16
  56. package/parsers-config.ts +0 -290
  57. package/src/account/account.sql.ts +0 -39
  58. package/src/account/account.ts +0 -456
  59. package/src/account/repo.ts +0 -166
  60. package/src/account/schema.ts +0 -99
  61. package/src/account/url.ts +0 -8
  62. package/src/acp/README.md +0 -174
  63. package/src/acp/agent.ts +0 -1787
  64. package/src/acp/session.ts +0 -116
  65. package/src/acp/types.ts +0 -24
  66. package/src/actor/actor.sql.ts +0 -38
  67. package/src/actor/events.ts +0 -67
  68. package/src/actor/index.ts +0 -2
  69. package/src/actor/registry.ts +0 -412
  70. package/src/actor/return-header.ts +0 -24
  71. package/src/actor/schema.ts +0 -47
  72. package/src/actor/spawn-ref.ts +0 -16
  73. package/src/actor/spawn.ts +0 -756
  74. package/src/actor/turn.ts +0 -49
  75. package/src/actor/waiter.ts +0 -166
  76. package/src/agent/agent.ts +0 -574
  77. package/src/agent/config.ts +0 -5
  78. package/src/agent/generate.txt +0 -75
  79. package/src/agent/prompt/checkpoint-writer.txt +0 -167
  80. package/src/agent/prompt/compaction.txt +0 -9
  81. package/src/agent/prompt/distill.txt +0 -199
  82. package/src/agent/prompt/dream.txt +0 -155
  83. package/src/agent/prompt/explore.txt +0 -18
  84. package/src/agent/prompt/summary.txt +0 -11
  85. package/src/agent/prompt/title.txt +0 -44
  86. package/src/audio.d.ts +0 -9
  87. package/src/auth/index.ts +0 -97
  88. package/src/bus/bus-event.ts +0 -33
  89. package/src/bus/global.ts +0 -12
  90. package/src/bus/index.ts +0 -193
  91. package/src/cli/bootstrap.ts +0 -33
  92. package/src/cli/cmd/account.ts +0 -258
  93. package/src/cli/cmd/acp.ts +0 -70
  94. package/src/cli/cmd/agent.ts +0 -248
  95. package/src/cli/cmd/cmd.ts +0 -7
  96. package/src/cli/cmd/db.ts +0 -120
  97. package/src/cli/cmd/debug/agent.ts +0 -192
  98. package/src/cli/cmd/debug/config.ts +0 -17
  99. package/src/cli/cmd/debug/file.ts +0 -100
  100. package/src/cli/cmd/debug/index.ts +0 -48
  101. package/src/cli/cmd/debug/lsp.ts +0 -61
  102. package/src/cli/cmd/debug/ripgrep.ts +0 -105
  103. package/src/cli/cmd/debug/scrap.ts +0 -16
  104. package/src/cli/cmd/debug/skill.ts +0 -23
  105. package/src/cli/cmd/debug/snapshot.ts +0 -53
  106. package/src/cli/cmd/doctor.ts +0 -116
  107. package/src/cli/cmd/export.ts +0 -306
  108. package/src/cli/cmd/generate.ts +0 -50
  109. package/src/cli/cmd/github.ts +0 -1647
  110. package/src/cli/cmd/import.ts +0 -208
  111. package/src/cli/cmd/init.ts +0 -97
  112. package/src/cli/cmd/login.ts +0 -402
  113. package/src/cli/cmd/mcp.ts +0 -812
  114. package/src/cli/cmd/models.ts +0 -88
  115. package/src/cli/cmd/plug.ts +0 -233
  116. package/src/cli/cmd/pr.ts +0 -138
  117. package/src/cli/cmd/providers.ts +0 -713
  118. package/src/cli/cmd/run-completion.ts +0 -77
  119. package/src/cli/cmd/run.ts +0 -694
  120. package/src/cli/cmd/serve.ts +0 -30
  121. package/src/cli/cmd/session.ts +0 -181
  122. package/src/cli/cmd/stats.ts +0 -413
  123. package/src/cli/cmd/tui/app.tsx +0 -1181
  124. package/src/cli/cmd/tui/asset/TEN_VAD_LICENSE +0 -12
  125. package/src/cli/cmd/tui/asset/charge.wav +0 -0
  126. package/src/cli/cmd/tui/asset/pulse-a.wav +0 -0
  127. package/src/cli/cmd/tui/asset/pulse-b.wav +0 -0
  128. package/src/cli/cmd/tui/asset/pulse-c.wav +0 -0
  129. package/src/cli/cmd/tui/asset/ten_vad.wasm +0 -0
  130. package/src/cli/cmd/tui/asset/ten_vad_loader.js +0 -30
  131. package/src/cli/cmd/tui/attach.ts +0 -84
  132. package/src/cli/cmd/tui/component/background-image.tsx +0 -150
  133. package/src/cli/cmd/tui/component/banner-session-expired.tsx +0 -48
  134. package/src/cli/cmd/tui/component/bg-pulse.tsx +0 -130
  135. package/src/cli/cmd/tui/component/border.tsx +0 -21
  136. package/src/cli/cmd/tui/component/dialog-agent.tsx +0 -31
  137. package/src/cli/cmd/tui/component/dialog-agreement.tsx +0 -111
  138. package/src/cli/cmd/tui/component/dialog-command.tsx +0 -219
  139. package/src/cli/cmd/tui/component/dialog-console-org.tsx +0 -103
  140. package/src/cli/cmd/tui/component/dialog-go-upsell.tsx +0 -157
  141. package/src/cli/cmd/tui/component/dialog-image-list.tsx +0 -111
  142. package/src/cli/cmd/tui/component/dialog-logo-design.tsx +0 -37
  143. package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
  144. package/src/cli/cmd/tui/component/dialog-model.tsx +0 -299
  145. package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -449
  146. package/src/cli/cmd/tui/component/dialog-session-delete-failed.tsx +0 -101
  147. package/src/cli/cmd/tui/component/dialog-session-list.tsx +0 -269
  148. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +0 -31
  149. package/src/cli/cmd/tui/component/dialog-skill.tsx +0 -42
  150. package/src/cli/cmd/tui/component/dialog-sleepy-login.tsx +0 -288
  151. package/src/cli/cmd/tui/component/dialog-stash.tsx +0 -87
  152. package/src/cli/cmd/tui/component/dialog-status.tsx +0 -170
  153. package/src/cli/cmd/tui/component/dialog-tag.tsx +0 -44
  154. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +0 -50
  155. package/src/cli/cmd/tui/component/dialog-token-plan.tsx +0 -77
  156. package/src/cli/cmd/tui/component/dialog-variant.tsx +0 -39
  157. package/src/cli/cmd/tui/component/dialog-workflows.tsx +0 -51
  158. package/src/cli/cmd/tui/component/dialog-workspace-create.tsx +0 -289
  159. package/src/cli/cmd/tui/component/dialog-workspace-unavailable.tsx +0 -81
  160. package/src/cli/cmd/tui/component/dialog-worktree.tsx +0 -90
  161. package/src/cli/cmd/tui/component/error-component.tsx +0 -92
  162. package/src/cli/cmd/tui/component/logo.tsx +0 -961
  163. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +0 -14
  164. package/src/cli/cmd/tui/component/prompt/autocomplete-detect.ts +0 -34
  165. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -668
  166. package/src/cli/cmd/tui/component/prompt/cwd.ts +0 -0
  167. package/src/cli/cmd/tui/component/prompt/frecency.tsx +0 -90
  168. package/src/cli/cmd/tui/component/prompt/history.tsx +0 -108
  169. package/src/cli/cmd/tui/component/prompt/index.tsx +0 -1869
  170. package/src/cli/cmd/tui/component/prompt/offset.ts +0 -45
  171. package/src/cli/cmd/tui/component/prompt/part.ts +0 -36
  172. package/src/cli/cmd/tui/component/prompt/stash.tsx +0 -101
  173. package/src/cli/cmd/tui/component/spinner.tsx +0 -24
  174. package/src/cli/cmd/tui/component/starry-background.tsx +0 -305
  175. package/src/cli/cmd/tui/component/startup-loading.tsx +0 -67
  176. package/src/cli/cmd/tui/component/task-item.tsx +0 -63
  177. package/src/cli/cmd/tui/component/textarea-keybindings.ts +0 -73
  178. package/src/cli/cmd/tui/component/todo-item.tsx +0 -32
  179. package/src/cli/cmd/tui/component/workflow-tree.tsx +0 -177
  180. package/src/cli/cmd/tui/config/cwd.ts +0 -5
  181. package/src/cli/cmd/tui/config/tui-migrate.ts +0 -151
  182. package/src/cli/cmd/tui/config/tui-schema.ts +0 -38
  183. package/src/cli/cmd/tui/config/tui.ts +0 -219
  184. package/src/cli/cmd/tui/context/args.tsx +0 -16
  185. package/src/cli/cmd/tui/context/directory.ts +0 -15
  186. package/src/cli/cmd/tui/context/event.ts +0 -45
  187. package/src/cli/cmd/tui/context/exit.tsx +0 -65
  188. package/src/cli/cmd/tui/context/helper.tsx +0 -25
  189. package/src/cli/cmd/tui/context/keybind.tsx +0 -105
  190. package/src/cli/cmd/tui/context/kv.tsx +0 -86
  191. package/src/cli/cmd/tui/context/language.tsx +0 -91
  192. package/src/cli/cmd/tui/context/local.tsx +0 -469
  193. package/src/cli/cmd/tui/context/plugin-keybinds.ts +0 -41
  194. package/src/cli/cmd/tui/context/project.tsx +0 -109
  195. package/src/cli/cmd/tui/context/prompt.tsx +0 -18
  196. package/src/cli/cmd/tui/context/route.tsx +0 -68
  197. package/src/cli/cmd/tui/context/sdk.tsx +0 -150
  198. package/src/cli/cmd/tui/context/sync.tsx +0 -884
  199. package/src/cli/cmd/tui/context/theme/aura.json +0 -69
  200. package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
  201. package/src/cli/cmd/tui/context/theme/carbonfox.json +0 -248
  202. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +0 -230
  203. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -230
  204. package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
  205. package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -225
  206. package/src/cli/cmd/tui/context/theme/cursor.json +0 -249
  207. package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
  208. package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
  209. package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
  210. package/src/cli/cmd/tui/context/theme/github.json +0 -233
  211. package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -242
  212. package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
  213. package/src/cli/cmd/tui/context/theme/lucent-orng.json +0 -234
  214. package/src/cli/cmd/tui/context/theme/material.json +0 -235
  215. package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
  216. package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
  217. package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
  218. package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
  219. package/src/cli/cmd/tui/context/theme/nord.json +0 -223
  220. package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
  221. package/src/cli/cmd/tui/context/theme/orng.json +0 -249
  222. package/src/cli/cmd/tui/context/theme/osaka-jade.json +0 -93
  223. package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
  224. package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
  225. package/src/cli/cmd/tui/context/theme/sleepycode.json +0 -245
  226. package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
  227. package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
  228. package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
  229. package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
  230. package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
  231. package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
  232. package/src/cli/cmd/tui/context/theme.tsx +0 -1298
  233. package/src/cli/cmd/tui/context/thinking.ts +0 -48
  234. package/src/cli/cmd/tui/context/tui-config.tsx +0 -9
  235. package/src/cli/cmd/tui/event.ts +0 -62
  236. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +0 -93
  237. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +0 -193
  238. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +0 -54
  239. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +0 -267
  240. package/src/cli/cmd/tui/feature-plugins/sidebar/cwd.tsx +0 -45
  241. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +0 -62
  242. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +0 -93
  243. package/src/cli/cmd/tui/feature-plugins/sidebar/goal.tsx +0 -84
  244. package/src/cli/cmd/tui/feature-plugins/sidebar/instructions.tsx +0 -54
  245. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +0 -66
  246. package/src/cli/cmd/tui/feature-plugins/sidebar/mascot.tsx +0 -69
  247. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +0 -98
  248. package/src/cli/cmd/tui/feature-plugins/sidebar/task.tsx +0 -95
  249. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +0 -51
  250. package/src/cli/cmd/tui/feature-plugins/sidebar/tps.ts +0 -31
  251. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +0 -274
  252. package/src/cli/cmd/tui/i18n/en.ts +0 -465
  253. package/src/cli/cmd/tui/i18n/es.ts +0 -508
  254. package/src/cli/cmd/tui/i18n/fr.ts +0 -515
  255. package/src/cli/cmd/tui/i18n/ja.ts +0 -463
  256. package/src/cli/cmd/tui/i18n/locales.ts +0 -82
  257. package/src/cli/cmd/tui/i18n/ru.ts +0 -527
  258. package/src/cli/cmd/tui/i18n/slash-command.ts +0 -11
  259. package/src/cli/cmd/tui/i18n/zh.ts +0 -454
  260. package/src/cli/cmd/tui/i18n/zht.ts +0 -430
  261. package/src/cli/cmd/tui/layer.ts +0 -6
  262. package/src/cli/cmd/tui/plugin/api.tsx +0 -402
  263. package/src/cli/cmd/tui/plugin/index.ts +0 -3
  264. package/src/cli/cmd/tui/plugin/internal.ts +0 -37
  265. package/src/cli/cmd/tui/plugin/runtime.ts +0 -1057
  266. package/src/cli/cmd/tui/plugin/slots.tsx +0 -60
  267. package/src/cli/cmd/tui/routes/home.tsx +0 -165
  268. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +0 -76
  269. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -116
  270. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +0 -47
  271. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -47
  272. package/src/cli/cmd/tui/routes/session/footer.tsx +0 -91
  273. package/src/cli/cmd/tui/routes/session/index.tsx +0 -3073
  274. package/src/cli/cmd/tui/routes/session/permission.tsx +0 -697
  275. package/src/cli/cmd/tui/routes/session/question.tsx +0 -488
  276. package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -97
  277. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +0 -143
  278. package/src/cli/cmd/tui/thread.ts +0 -324
  279. package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -61
  280. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -95
  281. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +0 -223
  282. package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -42
  283. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -123
  284. package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -467
  285. package/src/cli/cmd/tui/ui/dialog.tsx +0 -207
  286. package/src/cli/cmd/tui/ui/link.tsx +0 -28
  287. package/src/cli/cmd/tui/ui/spinner.ts +0 -378
  288. package/src/cli/cmd/tui/ui/toast.tsx +0 -102
  289. package/src/cli/cmd/tui/util/clipboard.ts +0 -203
  290. package/src/cli/cmd/tui/util/editor.ts +0 -36
  291. package/src/cli/cmd/tui/util/image-protocol.ts +0 -35
  292. package/src/cli/cmd/tui/util/index.ts +0 -6
  293. package/src/cli/cmd/tui/util/model.ts +0 -23
  294. package/src/cli/cmd/tui/util/pinyin.ts +0 -20
  295. package/src/cli/cmd/tui/util/provider-origin.ts +0 -7
  296. package/src/cli/cmd/tui/util/revert-diff.ts +0 -18
  297. package/src/cli/cmd/tui/util/scroll.ts +0 -23
  298. package/src/cli/cmd/tui/util/selection.ts +0 -23
  299. package/src/cli/cmd/tui/util/signal.ts +0 -41
  300. package/src/cli/cmd/tui/util/sound.ts +0 -154
  301. package/src/cli/cmd/tui/util/system-locale.ts +0 -209
  302. package/src/cli/cmd/tui/util/terminal.ts +0 -110
  303. package/src/cli/cmd/tui/util/transcript.ts +0 -112
  304. package/src/cli/cmd/tui/util/vad.ts +0 -229
  305. package/src/cli/cmd/tui/util/voice.ts +0 -450
  306. package/src/cli/cmd/tui/win32.ts +0 -130
  307. package/src/cli/cmd/tui/worker.ts +0 -104
  308. package/src/cli/cmd/uninstall.ts +0 -351
  309. package/src/cli/cmd/upgrade.ts +0 -79
  310. package/src/cli/cmd/web.ts +0 -96
  311. package/src/cli/effect/prompt.ts +0 -25
  312. package/src/cli/error.ts +0 -82
  313. package/src/cli/heap.ts +0 -59
  314. package/src/cli/i18n.ts +0 -15
  315. package/src/cli/logo.ts +0 -53
  316. package/src/cli/network.ts +0 -68
  317. package/src/cli/ui.ts +0 -133
  318. package/src/cli/upgrade.ts +0 -41
  319. package/src/command/index.ts +0 -276
  320. package/src/command/template/initialize.txt +0 -66
  321. package/src/command/template/review.txt +0 -101
  322. package/src/config/agent.ts +0 -197
  323. package/src/config/command.ts +0 -69
  324. package/src/config/compose.ts +0 -26
  325. package/src/config/config.ts +0 -1047
  326. package/src/config/console-state.ts +0 -16
  327. package/src/config/entry-name.ts +0 -16
  328. package/src/config/error.ts +0 -21
  329. package/src/config/formatter.ts +0 -17
  330. package/src/config/history.ts +0 -21
  331. package/src/config/index.ts +0 -17
  332. package/src/config/keybinds.ts +0 -127
  333. package/src/config/layout.ts +0 -10
  334. package/src/config/lsp.ts +0 -45
  335. package/src/config/managed.ts +0 -70
  336. package/src/config/markdown.ts +0 -97
  337. package/src/config/mcp.ts +0 -172
  338. package/src/config/model-id.ts +0 -14
  339. package/src/config/parse.ts +0 -44
  340. package/src/config/paths.ts +0 -85
  341. package/src/config/permission.ts +0 -76
  342. package/src/config/plugin.ts +0 -88
  343. package/src/config/provider.ts +0 -118
  344. package/src/config/server.ts +0 -20
  345. package/src/config/skills.ts +0 -16
  346. package/src/config/variable.ts +0 -90
  347. package/src/control-plane/adaptors/index.ts +0 -52
  348. package/src/control-plane/adaptors/worktree.ts +0 -47
  349. package/src/control-plane/dev/debug-workspace-plugin.ts +0 -73
  350. package/src/control-plane/schema.ts +0 -19
  351. package/src/control-plane/sse.ts +0 -66
  352. package/src/control-plane/types.ts +0 -34
  353. package/src/control-plane/util.ts +0 -37
  354. package/src/control-plane/workspace-context.ts +0 -26
  355. package/src/control-plane/workspace.sql.ts +0 -17
  356. package/src/control-plane/workspace.ts +0 -615
  357. package/src/effect/app-runtime.ts +0 -146
  358. package/src/effect/bootstrap-runtime.ts +0 -33
  359. package/src/effect/bridge.ts +0 -48
  360. package/src/effect/cross-spawn-spawner.ts +0 -514
  361. package/src/effect/index.ts +0 -5
  362. package/src/effect/instance-ref.ts +0 -11
  363. package/src/effect/instance-registry.ts +0 -12
  364. package/src/effect/instance-state.ts +0 -81
  365. package/src/effect/logger.ts +0 -73
  366. package/src/effect/memo-map.ts +0 -3
  367. package/src/effect/observability.ts +0 -107
  368. package/src/effect/run-service.ts +0 -52
  369. package/src/effect/runner.ts +0 -210
  370. package/src/effect/runtime.ts +0 -19
  371. package/src/env/index.ts +0 -37
  372. package/src/file/ignore.ts +0 -81
  373. package/src/file/index.ts +0 -664
  374. package/src/file/protected.ts +0 -59
  375. package/src/file/ripgrep.ts +0 -485
  376. package/src/file/watcher.ts +0 -163
  377. package/src/flag/flag.ts +0 -185
  378. package/src/format/formatter.ts +0 -403
  379. package/src/format/index.ts +0 -203
  380. package/src/git/index.ts +0 -260
  381. package/src/global/index.ts +0 -54
  382. package/src/history/backfill.ts +0 -162
  383. package/src/history/extract.ts +0 -67
  384. package/src/history/fts-query.ts +0 -15
  385. package/src/history/fts.sql.ts +0 -20
  386. package/src/history/index.ts +0 -10
  387. package/src/history/resolve.ts +0 -65
  388. package/src/history/service.ts +0 -258
  389. package/src/history/writer.ts +0 -112
  390. package/src/id/id.ts +0 -87
  391. package/src/ide/index.ts +0 -73
  392. package/src/inbox/inbox-ref.ts +0 -38
  393. package/src/inbox/inbox.sql.ts +0 -26
  394. package/src/inbox/inbox.ts +0 -223
  395. package/src/inbox/index.ts +0 -3
  396. package/src/inbox/render.ts +0 -40
  397. package/src/index.ts +0 -268
  398. package/src/installation/index.ts +0 -362
  399. package/src/installation/version.ts +0 -8
  400. package/src/lsp/client.ts +0 -249
  401. package/src/lsp/diagnostic.ts +0 -29
  402. package/src/lsp/index.ts +0 -3
  403. package/src/lsp/language.ts +0 -120
  404. package/src/lsp/launch.ts +0 -21
  405. package/src/lsp/lsp.ts +0 -519
  406. package/src/lsp/server.ts +0 -1956
  407. package/src/mcp/auth.ts +0 -144
  408. package/src/mcp/index.ts +0 -939
  409. package/src/mcp/oauth-callback.ts +0 -236
  410. package/src/mcp/oauth-provider.ts +0 -214
  411. package/src/memory/fts-query.ts +0 -37
  412. package/src/memory/fts.sql.ts +0 -19
  413. package/src/memory/index.ts +0 -1
  414. package/src/memory/paths.ts +0 -116
  415. package/src/memory/reconcile.ts +0 -144
  416. package/src/memory/service.ts +0 -144
  417. package/src/metrics/client.ts +0 -40
  418. package/src/metrics/event.ts +0 -43
  419. package/src/metrics/index.ts +0 -5
  420. package/src/metrics/installation.ts +0 -18
  421. package/src/metrics/subscriber.ts +0 -58
  422. package/src/metrics/util.ts +0 -9
  423. package/src/node.ts +0 -6
  424. package/src/npm/config.ts +0 -0
  425. package/src/npm/index.ts +0 -293
  426. package/src/npmcli-config.d.ts +0 -43
  427. package/src/patch/index.ts +0 -680
  428. package/src/permission/arity.ts +0 -163
  429. package/src/permission/evaluate.ts +0 -15
  430. package/src/permission/index.ts +0 -382
  431. package/src/permission/schema.ts +0 -17
  432. package/src/plugin/checkpoint-splitover.ts +0 -60
  433. package/src/plugin/cloudflare.ts +0 -76
  434. package/src/plugin/codex.ts +0 -611
  435. package/src/plugin/github-copilot/copilot.ts +0 -368
  436. package/src/plugin/github-copilot/models.ts +0 -153
  437. package/src/plugin/index.ts +0 -637
  438. package/src/plugin/install.ts +0 -439
  439. package/src/plugin/loader.ts +0 -216
  440. package/src/plugin/matcher.ts +0 -33
  441. package/src/plugin/meta.ts +0 -188
  442. package/src/plugin/shared.ts +0 -323
  443. package/src/plugin/sleepy.ts +0 -244
  444. package/src/plugin/subagent-progress-checker.ts +0 -147
  445. package/src/project/bootstrap.ts +0 -59
  446. package/src/project/index.ts +0 -2
  447. package/src/project/instance.ts +0 -215
  448. package/src/project/project-id.ts +0 -48
  449. package/src/project/project.sql.ts +0 -16
  450. package/src/project/project.ts +0 -522
  451. package/src/project/schema.ts +0 -15
  452. package/src/project/vcs.ts +0 -227
  453. package/src/project/workspace-trust.ts +0 -67
  454. package/src/provider/auth.ts +0 -234
  455. package/src/provider/error.ts +0 -216
  456. package/src/provider/index.ts +0 -5
  457. package/src/provider/models-snapshot.d.ts +0 -2
  458. package/src/provider/models-snapshot.js +0 -3
  459. package/src/provider/models.ts +0 -180
  460. package/src/provider/provider.ts +0 -2098
  461. package/src/provider/schema.ts +0 -36
  462. package/src/provider/sdk/copilot/README.md +0 -5
  463. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +0 -170
  464. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +0 -15
  465. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +0 -19
  466. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +0 -64
  467. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +0 -815
  468. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +0 -28
  469. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +0 -44
  470. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +0 -83
  471. package/src/provider/sdk/copilot/copilot-provider.ts +0 -100
  472. package/src/provider/sdk/copilot/index.ts +0 -2
  473. package/src/provider/sdk/copilot/openai-compatible-error.ts +0 -27
  474. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +0 -335
  475. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +0 -22
  476. package/src/provider/sdk/copilot/responses/openai-config.ts +0 -18
  477. package/src/provider/sdk/copilot/responses/openai-error.ts +0 -22
  478. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +0 -214
  479. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +0 -1770
  480. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +0 -173
  481. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +0 -1
  482. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +0 -87
  483. package/src/provider/sdk/copilot/responses/tool/file-search.ts +0 -127
  484. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +0 -114
  485. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +0 -64
  486. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +0 -103
  487. package/src/provider/sdk/copilot/responses/tool/web-search.ts +0 -102
  488. package/src/provider/session-check.ts +0 -163
  489. package/src/provider/transform.ts +0 -1376
  490. package/src/pty/index.ts +0 -364
  491. package/src/pty/pty.bun.ts +0 -26
  492. package/src/pty/pty.node.ts +0 -27
  493. package/src/pty/pty.ts +0 -25
  494. package/src/pty/schema.ts +0 -17
  495. package/src/question/index.ts +0 -252
  496. package/src/question/schema.ts +0 -17
  497. package/src/server/adapter.bun.ts +0 -40
  498. package/src/server/adapter.node.ts +0 -66
  499. package/src/server/adapter.ts +0 -21
  500. package/src/server/auth.ts +0 -16
  501. package/src/server/error.ts +0 -53
  502. package/src/server/event.ts +0 -7
  503. package/src/server/fence.ts +0 -81
  504. package/src/server/mdns.ts +0 -60
  505. package/src/server/middleware.ts +0 -94
  506. package/src/server/projectors.ts +0 -28
  507. package/src/server/proxy.ts +0 -171
  508. package/src/server/pty-ticket.ts +0 -42
  509. package/src/server/rate-limit.ts +0 -38
  510. package/src/server/routes/control/index.ts +0 -160
  511. package/src/server/routes/control/workspace.ts +0 -203
  512. package/src/server/routes/global.ts +0 -367
  513. package/src/server/routes/instance/bash-interactive.ts +0 -82
  514. package/src/server/routes/instance/config.ts +0 -89
  515. package/src/server/routes/instance/event.ts +0 -108
  516. package/src/server/routes/instance/experimental.ts +0 -408
  517. package/src/server/routes/instance/file.ts +0 -190
  518. package/src/server/routes/instance/httpapi/config.ts +0 -51
  519. package/src/server/routes/instance/httpapi/permission.ts +0 -72
  520. package/src/server/routes/instance/httpapi/project.ts +0 -62
  521. package/src/server/routes/instance/httpapi/provider.ts +0 -142
  522. package/src/server/routes/instance/httpapi/question.ts +0 -121
  523. package/src/server/routes/instance/httpapi/server.ts +0 -153
  524. package/src/server/routes/instance/index.ts +0 -301
  525. package/src/server/routes/instance/mcp.ts +0 -260
  526. package/src/server/routes/instance/middleware.ts +0 -44
  527. package/src/server/routes/instance/permission.ts +0 -73
  528. package/src/server/routes/instance/project.ts +0 -122
  529. package/src/server/routes/instance/provider.ts +0 -158
  530. package/src/server/routes/instance/pty.ts +0 -302
  531. package/src/server/routes/instance/question.ts +0 -162
  532. package/src/server/routes/instance/session.ts +0 -1328
  533. package/src/server/routes/instance/sync.ts +0 -143
  534. package/src/server/routes/instance/trace.ts +0 -59
  535. package/src/server/routes/instance/tui.ts +0 -384
  536. package/src/server/routes/instance/workflows.ts +0 -142
  537. package/src/server/routes/ui.ts +0 -37
  538. package/src/server/server.ts +0 -146
  539. package/src/server/workspace.ts +0 -122
  540. package/src/session/auto-dream.ts +0 -123
  541. package/src/session/boundary.ts +0 -77
  542. package/src/session/budgeted-read.ts +0 -118
  543. package/src/session/checkpoint-align.ts +0 -29
  544. package/src/session/checkpoint-context.ts +0 -36
  545. package/src/session/checkpoint-paths.ts +0 -86
  546. package/src/session/checkpoint-progress-reconcile.ts +0 -111
  547. package/src/session/checkpoint-retry.ts +0 -192
  548. package/src/session/checkpoint-templates.ts +0 -114
  549. package/src/session/checkpoint-validator.ts +0 -259
  550. package/src/session/checkpoint.ts +0 -1560
  551. package/src/session/classify.ts +0 -117
  552. package/src/session/claude-import.ts +0 -381
  553. package/src/session/codex-import.ts +0 -416
  554. package/src/session/compaction.ts +0 -545
  555. package/src/session/external-import.sql.ts +0 -18
  556. package/src/session/external-import.ts +0 -136
  557. package/src/session/goal.ts +0 -232
  558. package/src/session/index.ts +0 -1
  559. package/src/session/instruction.ts +0 -276
  560. package/src/session/last-message-info.ts +0 -32
  561. package/src/session/llm-request-prefix.ts +0 -82
  562. package/src/session/llm.ts +0 -737
  563. package/src/session/max-mode.ts +0 -410
  564. package/src/session/message-v2.ts +0 -1138
  565. package/src/session/message.ts +0 -191
  566. package/src/session/opencode-import.ts +0 -281
  567. package/src/session/overflow.ts +0 -53
  568. package/src/session/prefix-capture-ref.ts +0 -48
  569. package/src/session/processor.ts +0 -983
  570. package/src/session/projectors.ts +0 -137
  571. package/src/session/prompt/anthropic.txt +0 -154
  572. package/src/session/prompt/beast.txt +0 -155
  573. package/src/session/prompt/build-switch.txt +0 -5
  574. package/src/session/prompt/codex.txt +0 -79
  575. package/src/session/prompt/compose.txt +0 -119
  576. package/src/session/prompt/copilot-gpt-5.txt +0 -143
  577. package/src/session/prompt/deepseek.txt +0 -131
  578. package/src/session/prompt/default.old.txt +0 -151
  579. package/src/session/prompt/default.txt +0 -172
  580. package/src/session/prompt/gemini.txt +0 -155
  581. package/src/session/prompt/glm.txt +0 -51
  582. package/src/session/prompt/gpt.txt +0 -107
  583. package/src/session/prompt/kimi.txt +0 -95
  584. package/src/session/prompt/max-steps.txt +0 -16
  585. package/src/session/prompt/minimax.txt +0 -140
  586. package/src/session/prompt/text-loop-recovery.ts +0 -40
  587. package/src/session/prompt/text-ngram-detection.ts +0 -87
  588. package/src/session/prompt/trinity.txt +0 -97
  589. package/src/session/prompt.ts +0 -3878
  590. package/src/session/prune.ts +0 -481
  591. package/src/session/retry.ts +0 -178
  592. package/src/session/revert.ts +0 -161
  593. package/src/session/run-state.ts +0 -135
  594. package/src/session/schema.ts +0 -36
  595. package/src/session/session.sql.ts +0 -110
  596. package/src/session/session.ts +0 -908
  597. package/src/session/status.ts +0 -89
  598. package/src/session/summary.ts +0 -163
  599. package/src/session/system.ts +0 -96
  600. package/src/session/todo.ts +0 -77
  601. package/src/session/trajectory.ts +0 -98
  602. package/src/share/index.ts +0 -2
  603. package/src/share/session.ts +0 -57
  604. package/src/share/share-next.ts +0 -381
  605. package/src/share/share.sql.ts +0 -13
  606. package/src/shell/shell.ts +0 -124
  607. package/src/skill/builtin/.bundle/self-extend/SKILL.md +0 -131
  608. package/src/skill/builtin/.bundle/self-extend/reference/hook-api.md +0 -242
  609. package/src/skill/builtin/.bundle/self-extend/reference/skill-api.md +0 -114
  610. package/src/skill/builtin/.bundle/self-extend/reference/tool-api.md +0 -115
  611. package/src/skill/builtin/.bundle/self-extend/reference/tui-api.md +0 -258
  612. package/src/skill/builtin/bundle.macro.ts +0 -30
  613. package/src/skill/builtin/extract.ts +0 -41
  614. package/src/skill/compose/.bundle/ask/SKILL.md +0 -58
  615. package/src/skill/compose/.bundle/brainstorm/SKILL.md +0 -200
  616. package/src/skill/compose/.bundle/brainstorm/scripts/frame-template.html +0 -214
  617. package/src/skill/compose/.bundle/brainstorm/scripts/helper.js +0 -88
  618. package/src/skill/compose/.bundle/brainstorm/scripts/server.cjs +0 -354
  619. package/src/skill/compose/.bundle/brainstorm/scripts/start-server.sh +0 -148
  620. package/src/skill/compose/.bundle/brainstorm/scripts/stop-server.sh +0 -56
  621. package/src/skill/compose/.bundle/brainstorm/spec-document-reviewer-prompt.md +0 -50
  622. package/src/skill/compose/.bundle/brainstorm/visual-companion.md +0 -258
  623. package/src/skill/compose/.bundle/debug/CREATION-LOG.md +0 -119
  624. package/src/skill/compose/.bundle/debug/SKILL.md +0 -297
  625. package/src/skill/compose/.bundle/debug/condition-based-waiting-example.ts +0 -158
  626. package/src/skill/compose/.bundle/debug/condition-based-waiting.md +0 -106
  627. package/src/skill/compose/.bundle/debug/defense-in-depth.md +0 -122
  628. package/src/skill/compose/.bundle/debug/find-polluter.sh +0 -63
  629. package/src/skill/compose/.bundle/debug/root-cause-tracing.md +0 -144
  630. package/src/skill/compose/.bundle/debug/test-academic.md +0 -14
  631. package/src/skill/compose/.bundle/debug/test-pressure-1.md +0 -58
  632. package/src/skill/compose/.bundle/debug/test-pressure-2.md +0 -68
  633. package/src/skill/compose/.bundle/debug/test-pressure-3.md +0 -69
  634. package/src/skill/compose/.bundle/execute/SKILL.md +0 -71
  635. package/src/skill/compose/.bundle/feedback/SKILL.md +0 -214
  636. package/src/skill/compose/.bundle/merge/SKILL.md +0 -252
  637. package/src/skill/compose/.bundle/new-skill/SKILL.md +0 -211
  638. package/src/skill/compose/.bundle/parallel/SKILL.md +0 -168
  639. package/src/skill/compose/.bundle/plan/SKILL.md +0 -183
  640. package/src/skill/compose/.bundle/plan/plan-document-reviewer-prompt.md +0 -50
  641. package/src/skill/compose/.bundle/report/SKILL.md +0 -180
  642. package/src/skill/compose/.bundle/review/SKILL.md +0 -104
  643. package/src/skill/compose/.bundle/review/code-reviewer.md +0 -178
  644. package/src/skill/compose/.bundle/subagent/SKILL.md +0 -325
  645. package/src/skill/compose/.bundle/subagent/code-quality-reviewer-prompt.md +0 -26
  646. package/src/skill/compose/.bundle/subagent/implementer-prompt.md +0 -128
  647. package/src/skill/compose/.bundle/subagent/spec-reviewer-prompt.md +0 -118
  648. package/src/skill/compose/.bundle/tdd/SKILL.md +0 -360
  649. package/src/skill/compose/.bundle/tdd/testing-anti-patterns.md +0 -299
  650. package/src/skill/compose/.bundle/verify/SKILL.md +0 -140
  651. package/src/skill/compose/.bundle/worktree/SKILL.md +0 -234
  652. package/src/skill/compose/LICENSE-karpathy +0 -28
  653. package/src/skill/compose/bundle.macro.ts +0 -30
  654. package/src/skill/compose/extract.ts +0 -85
  655. package/src/skill/discovery.ts +0 -114
  656. package/src/skill/index.ts +0 -328
  657. package/src/snapshot/index.ts +0 -777
  658. package/src/sql.d.ts +0 -4
  659. package/src/storage/db.bun.ts +0 -8
  660. package/src/storage/db.node.ts +0 -8
  661. package/src/storage/db.ts +0 -172
  662. package/src/storage/index.ts +0 -26
  663. package/src/storage/json-migration.ts +0 -426
  664. package/src/storage/read-sqlite.bun.ts +0 -11
  665. package/src/storage/read-sqlite.node.ts +0 -13
  666. package/src/storage/read-sqlite.ts +0 -10
  667. package/src/storage/schema.sql.ts +0 -10
  668. package/src/storage/schema.ts +0 -7
  669. package/src/storage/storage.ts +0 -331
  670. package/src/sync/README.md +0 -179
  671. package/src/sync/event.sql.ts +0 -16
  672. package/src/sync/index.ts +0 -278
  673. package/src/sync/schema.ts +0 -14
  674. package/src/task/events.ts +0 -28
  675. package/src/task/gate-state.ts +0 -54
  676. package/src/task/gate.ts +0 -116
  677. package/src/task/index.ts +0 -1
  678. package/src/task/registry.ts +0 -394
  679. package/src/task/schema.ts +0 -43
  680. package/src/task/task.sql.ts +0 -50
  681. package/src/team/events.ts +0 -22
  682. package/src/team/index.ts +0 -113
  683. package/src/team/schema.ts +0 -31
  684. package/src/temporary.ts +0 -33
  685. package/src/tool/actor.shell.txt +0 -72
  686. package/src/tool/actor.ts +0 -804
  687. package/src/tool/actor.txt +0 -103
  688. package/src/tool/apply_patch.ts +0 -308
  689. package/src/tool/apply_patch.txt +0 -33
  690. package/src/tool/bash-interactive.ts +0 -183
  691. package/src/tool/bash.ts +0 -704
  692. package/src/tool/bash.txt +0 -123
  693. package/src/tool/change-directory.ts +0 -91
  694. package/src/tool/codesearch.ts +0 -63
  695. package/src/tool/codesearch.txt +0 -12
  696. package/src/tool/edit.ts +0 -693
  697. package/src/tool/edit.txt +0 -10
  698. package/src/tool/external-directory.ts +0 -132
  699. package/src/tool/glob.ts +0 -100
  700. package/src/tool/glob.txt +0 -6
  701. package/src/tool/grep.ts +0 -145
  702. package/src/tool/grep.txt +0 -8
  703. package/src/tool/history.ts +0 -146
  704. package/src/tool/history.txt +0 -17
  705. package/src/tool/index.ts +0 -4
  706. package/src/tool/invalid.ts +0 -20
  707. package/src/tool/invocation-style.ts +0 -17
  708. package/src/tool/lsp.ts +0 -91
  709. package/src/tool/lsp.txt +0 -19
  710. package/src/tool/mcp-exa.ts +0 -78
  711. package/src/tool/memory-path-guard.ts +0 -162
  712. package/src/tool/memory.ts +0 -81
  713. package/src/tool/memory.txt +0 -69
  714. package/src/tool/multiedit.ts +0 -54
  715. package/src/tool/multiedit.txt +0 -41
  716. package/src/tool/notebook-edit.ts +0 -225
  717. package/src/tool/notebook-edit.txt +0 -10
  718. package/src/tool/plan-enter.txt +0 -16
  719. package/src/tool/plan-exit.txt +0 -14
  720. package/src/tool/plan.ts +0 -179
  721. package/src/tool/question.ts +0 -67
  722. package/src/tool/question.txt +0 -10
  723. package/src/tool/read-state.ts +0 -44
  724. package/src/tool/read.ts +0 -327
  725. package/src/tool/read.txt +0 -14
  726. package/src/tool/recoverable.ts +0 -35
  727. package/src/tool/registry.ts +0 -429
  728. package/src/tool/schema.ts +0 -17
  729. package/src/tool/session-cwd.ts +0 -35
  730. package/src/tool/shell-tokenize.ts +0 -374
  731. package/src/tool/shell-wrap.ts +0 -235
  732. package/src/tool/skill.ts +0 -76
  733. package/src/tool/skill.txt +0 -5
  734. package/src/tool/task.shell.txt +0 -57
  735. package/src/tool/task.ts +0 -456
  736. package/src/tool/task.txt +0 -56
  737. package/src/tool/tool.ts +0 -166
  738. package/src/tool/truncate.ts +0 -201
  739. package/src/tool/truncation-dir.ts +0 -4
  740. package/src/tool/webfetch.ts +0 -208
  741. package/src/tool/webfetch.txt +0 -13
  742. package/src/tool/websearch/index.ts +0 -104
  743. package/src/tool/websearch/sleepy.ts +0 -118
  744. package/src/tool/websearch/websearch.txt +0 -14
  745. package/src/tool/workflow.ts +0 -357
  746. package/src/tool/workflow.txt +0 -25
  747. package/src/tool/write.ts +0 -88
  748. package/src/tool/write.txt +0 -10
  749. package/src/util/abort.ts +0 -35
  750. package/src/util/archive.ts +0 -15
  751. package/src/util/color.ts +0 -17
  752. package/src/util/data-url.ts +0 -9
  753. package/src/util/defer.ts +0 -10
  754. package/src/util/effect-http-client.ts +0 -11
  755. package/src/util/effect-zod.ts +0 -367
  756. package/src/util/env-info.ts +0 -62
  757. package/src/util/error.ts +0 -78
  758. package/src/util/filesystem.ts +0 -243
  759. package/src/util/fn.ts +0 -21
  760. package/src/util/format.ts +0 -20
  761. package/src/util/iife.ts +0 -3
  762. package/src/util/index.ts +0 -14
  763. package/src/util/keybind.ts +0 -101
  764. package/src/util/lazy.ts +0 -18
  765. package/src/util/local-context.ts +0 -23
  766. package/src/util/locale.ts +0 -79
  767. package/src/util/lock.ts +0 -96
  768. package/src/util/log.ts +0 -234
  769. package/src/util/media.ts +0 -26
  770. package/src/util/network.ts +0 -9
  771. package/src/util/process.ts +0 -174
  772. package/src/util/provider-priority.ts +0 -48
  773. package/src/util/queue.ts +0 -60
  774. package/src/util/record.ts +0 -3
  775. package/src/util/rpc.ts +0 -64
  776. package/src/util/schema.ts +0 -53
  777. package/src/util/scrap.ts +0 -10
  778. package/src/util/signal.ts +0 -12
  779. package/src/util/sleepy-process.ts +0 -24
  780. package/src/util/ssrf.ts +0 -116
  781. package/src/util/timeout.ts +0 -14
  782. package/src/util/token.ts +0 -5
  783. package/src/util/tool-compat.ts +0 -144
  784. package/src/util/update-schema.ts +0 -13
  785. package/src/util/which.ts +0 -14
  786. package/src/util/wildcard.ts +0 -57
  787. package/src/workflow/builtin/compose.js +0 -749
  788. package/src/workflow/builtin/deep-research.js +0 -398
  789. package/src/workflow/builtin.ts +0 -59
  790. package/src/workflow/events.ts +0 -72
  791. package/src/workflow/meta.ts +0 -335
  792. package/src/workflow/persistence.ts +0 -312
  793. package/src/workflow/resolve.ts +0 -45
  794. package/src/workflow/runtime-ref.ts +0 -18
  795. package/src/workflow/runtime.ts +0 -1447
  796. package/src/workflow/sandbox.ts +0 -286
  797. package/src/workflow/workflow.sql.ts +0 -31
  798. package/src/workflow/workspace.ts +0 -69
  799. package/src/worktree/index.ts +0 -629
@@ -1,1869 +0,0 @@
1
- import { BoxRenderable, RGBA, TextareaRenderable, MouseEvent, PasteEvent, decodePasteBytes } from "@opentui/core"
2
- import { createEffect, createMemo, onMount, createSignal, onCleanup, on, Show, Switch, Match } from "solid-js"
3
- import "opentui-spinner/solid"
4
- import path from "path"
5
- import { fileURLToPath } from "url"
6
- import { Filesystem } from "@/util"
7
- import { useLocal } from "@tui/context/local"
8
- import { tint, useTheme } from "@tui/context/theme"
9
- import { EmptyBorder, SplitBorder } from "@tui/component/border"
10
- import { useSDK } from "@tui/context/sdk"
11
- import { useRoute } from "@tui/context/route"
12
- import { useProject } from "@tui/context/project"
13
- import { useSync } from "@tui/context/sync"
14
- import { useEvent } from "@tui/context/event"
15
- import { MessageID, PartID } from "@/session/schema"
16
- import { createStore, produce, unwrap } from "solid-js/store"
17
- import { useKeybind } from "@tui/context/keybind"
18
- import { usePromptHistory, type PromptInfo } from "./history"
19
- import { assign, expandPlaceholders } from "./part"
20
- import { usePromptStash } from "./stash"
21
- import { DialogStash } from "../dialog-stash"
22
- import { type AutocompleteRef, Autocomplete } from "./autocomplete"
23
- import { useCommandDialog } from "../dialog-command"
24
- import { useLanguage } from "@tui/context/language"
25
- import { useRenderer, type JSX } from "@opentui/solid"
26
- import * as Editor from "@tui/util/editor"
27
- import * as Voice from "@tui/util/voice"
28
- import { useExit } from "../../context/exit"
29
- import * as Clipboard from "../../util/clipboard"
30
- import type { AssistantMessage, FilePart, UserMessage } from "@sleepy-ai/sdk/v2"
31
- import { TuiEvent } from "../../event"
32
- import { iife } from "@/util/iife"
33
- import { Locale } from "@/util"
34
- import { formatDuration } from "@/util/format"
35
- import { createColors, createFrames } from "../../ui/spinner.ts"
36
- import { useDialog } from "@tui/ui/dialog"
37
- import { DialogProvider as DialogProviderConnect } from "../dialog-provider"
38
- import { DialogAlert } from "../../ui/dialog-alert"
39
- import { useToast } from "../../ui/toast"
40
- import { useKV } from "../../context/kv"
41
- import { createFadeIn } from "../../util/signal"
42
- import { useTextareaKeybindings } from "../textarea-keybindings"
43
- import { DialogSkill } from "../dialog-skill"
44
- import { DialogWorkspaceCreate, restoreWorkspaceSession } from "../dialog-workspace-create"
45
- import { DialogWorkspaceUnavailable } from "../dialog-workspace-unavailable"
46
- import { DialogAgreement, FREE_AGREEMENT_KEY, FREE_MODEL_IDS } from "../dialog-agreement"
47
- import { useArgs } from "@tui/context/args"
48
-
49
- export type PromptProps = {
50
- sessionID?: string
51
- workspaceID?: string
52
- visible?: boolean
53
- disabled?: boolean
54
- onSubmit?: () => void
55
- ref?: (ref: PromptRef | undefined) => void
56
- hint?: JSX.Element
57
- right?: JSX.Element
58
- showPlaceholder?: boolean
59
- placeholders?: {
60
- normal?: string[]
61
- shell?: string[]
62
- }
63
- }
64
-
65
- export type PromptRef = {
66
- focused: boolean
67
- current: PromptInfo
68
- set(prompt: PromptInfo): void
69
- reset(): void
70
- blur(): void
71
- focus(): void
72
- submit(): void
73
- paste(): void
74
- }
75
-
76
- const money = new Intl.NumberFormat("en-US", {
77
- style: "currency",
78
- currency: "USD",
79
- })
80
-
81
- function randomIndex(count: number) {
82
- if (count <= 0) return 0
83
- return Math.floor(Math.random() * count)
84
- }
85
-
86
- function fadeColor(color: RGBA, alpha: number) {
87
- return RGBA.fromValues(color.r, color.g, color.b, color.a * alpha)
88
- }
89
-
90
- let stashed: { prompt: PromptInfo; cursor: number } | undefined
91
-
92
- // Module-level voice state: survives component remounts and route changes
93
- let activeVoice: {
94
- handle: Voice.StreamingHandle
95
- pending: number
96
- appendText: (text: string) => void
97
- setText: (text: string) => void
98
- getPlainText: () => string
99
- switchAgent: (name: string) => void
100
- submit: () => Promise<unknown>
101
- setState: (type: "listening" | "speaking" | "processing" | "finishing" | "idle") => void
102
- showError: (msg: string) => void
103
- } | undefined
104
-
105
- export function Prompt(props: PromptProps) {
106
- let input: TextareaRenderable
107
- let anchor: BoxRenderable
108
- let autocomplete: AutocompleteRef
109
-
110
- const keybind = useKeybind()
111
- const local = useLocal()
112
- const args = useArgs()
113
- const sdk = useSDK()
114
- const route = useRoute()
115
- const project = useProject()
116
- const sync = useSync()
117
- const dialog = useDialog()
118
- const toast = useToast()
119
- const status = createMemo(() => sync.data.session_status?.[props.sessionID ?? ""] ?? { type: "idle" })
120
- const history = usePromptHistory()
121
- const stash = usePromptStash()
122
- const command = useCommandDialog()
123
- const t = useLanguage().t
124
- const renderer = useRenderer()
125
- const { theme, syntax } = useTheme()
126
- const kv = useKV()
127
- const animationsEnabled = createMemo(() => kv.get("animations_enabled", true))
128
- const voiceEnabled = createMemo(() => kv.get("voice_enabled", false))
129
- const voiceSendEnabled = createMemo(() => kv.get("voice_send_command", false))
130
- const voiceControlEnabled = createMemo(() => kv.get("voice_control_enabled", false))
131
- const currentProviderLabel = createMemo(() => local.model.parsed().provider)
132
- const [voiceState, setVoiceState] = createSignal<"idle" | "listening" | "speaking" | "processing" | "finishing">(
133
- activeVoice ? (activeVoice.pending > 0 ? "processing" : "listening") : "idle",
134
- )
135
- const [voiceElapsed, setVoiceElapsed] = createSignal(0)
136
-
137
- let voiceTimer: ReturnType<typeof setInterval> | undefined
138
- let voiceSegmentStart = 0
139
-
140
- function voiceTimerStart() {
141
- voiceTimerStop()
142
- voiceSegmentStart = Date.now()
143
- voiceTimer = setInterval(() => {
144
- setVoiceElapsed(Math.floor((Date.now() - voiceSegmentStart) / 1000))
145
- }, 200)
146
- }
147
- function voiceTimerStop() {
148
- if (voiceTimer) {
149
- clearInterval(voiceTimer)
150
- voiceTimer = undefined
151
- }
152
- setVoiceElapsed(0)
153
- }
154
-
155
- function voiceAppendText(text: string) {
156
- if (!input || input.isDestroyed) return
157
- const current = store.prompt.input
158
- if (current.length > 0 && /[.?!]$/.test(current) && text.length > 0 && text[0] !== " ") {
159
- input.insertText(" " + text)
160
- setStore("prompt", "input", current + " " + text)
161
- } else {
162
- input.insertText(text)
163
- setStore("prompt", "input", current + text)
164
- }
165
- setTimeout(() => {
166
- if (!input || input.isDestroyed) return
167
- input.getLayoutNode().markDirty()
168
- input.gotoBufferEnd()
169
- renderer.requestRender()
170
- }, 0)
171
- }
172
-
173
- function voiceSetText(text: string) {
174
- if (!input || input.isDestroyed) return
175
- input.clear()
176
- input.insertText(text)
177
- setStore("prompt", "input", text)
178
- setTimeout(() => {
179
- if (!input || input.isDestroyed) return
180
- input.getLayoutNode().markDirty()
181
- input.gotoBufferEnd()
182
- renderer.requestRender()
183
- }, 0)
184
- }
185
-
186
- function voiceGetPlainText() {
187
- return store.prompt.input
188
- }
189
-
190
- function voiceSwitchAgent(name: string) {
191
- const match = local.agent.list().find((x) => x.name.toLowerCase() === name.toLowerCase())
192
- if (match) local.agent.set(match.name)
193
- else toast.show({ message: t("tui.voice.error.unknown_agent", { name: name }), variant: "error", duration: 3000 })
194
- }
195
-
196
- function voiceSetState(type: "idle" | "listening" | "speaking" | "processing" | "finishing") {
197
- setVoiceState(type)
198
- if (type === "speaking") voiceTimerStart()
199
- if (type === "idle" || type === "listening" || type === "processing") voiceTimerStop()
200
- }
201
-
202
- // Wire module-level callbacks to current component instance
203
- if (activeVoice) {
204
- activeVoice.appendText = voiceAppendText
205
- activeVoice.setText = voiceSetText
206
- activeVoice.getPlainText = voiceGetPlainText
207
- activeVoice.switchAgent = voiceSwitchAgent
208
- activeVoice.submit = () => submit()
209
- activeVoice.setState = voiceSetState
210
- activeVoice.showError = (msg) => toast.show({ message: msg, variant: "error", duration: 3000 })
211
- }
212
- onCleanup(() => {
213
- voiceTimerStop()
214
- })
215
-
216
- async function voiceToggle() {
217
- const state = voiceState()
218
- if (state === "listening" || state === "speaking" || state === "processing") {
219
- voiceTimerStop()
220
- setVoiceState("finishing")
221
- if (activeVoice) {
222
- const handle = activeVoice.handle
223
- const av = activeVoice
224
- activeVoice = undefined
225
- await Voice.stopStreaming(handle)
226
- if (av.pending <= 0) setVoiceState("idle")
227
- }
228
- return
229
- }
230
- if (state === "finishing") return
231
- // Start streaming — only validate the active mode's provider
232
- const voiceConfig = sync.data.config.voice
233
- const resolved = Voice.resolveVoiceConfig(voiceConfig)
234
- const activeConfig = voiceControlEnabled() ? resolved.control : resolved.asr
235
- const creds = Voice.resolveCredentials(sync.data.provider, activeConfig)
236
- if ("error" in creds) {
237
- const vars = { provider: creds.providerID, model: creds.model }
238
- const msg = !voiceConfig ? t("tui.voice.error.no_auth")
239
- : creds.error === "not_found" ? t("tui.voice.error.provider_not_found", vars)
240
- : creds.error === "no_url" ? t("tui.voice.error.no_url", vars)
241
- : t("tui.voice.error.no_auth_provider", vars)
242
- toast.show({ message: msg, variant: "error" })
243
- return
244
- }
245
- if (!Voice.isAvailable()) {
246
- toast.show({ message: t("tui.voice.error.no_recorder"), variant: "error" })
247
- return
248
- }
249
-
250
- const av: NonNullable<typeof activeVoice> = {
251
- handle: undefined!,
252
- pending: 0,
253
- appendText: voiceAppendText,
254
- setText: voiceSetText,
255
- getPlainText: voiceGetPlainText,
256
- switchAgent: voiceSwitchAgent,
257
- submit: () => submit(),
258
- setState: voiceSetState,
259
- showError: (msg) => toast.show({ message: msg, variant: "error", duration: 3000 }),
260
- }
261
-
262
- let voiceControlChain: Promise<void> = Promise.resolve()
263
-
264
- const handle = Voice.startStreaming({
265
- onSegment: (segment) => {
266
- av.pending++
267
- av.setState("processing")
268
-
269
- if (voiceControlEnabled()) {
270
- voiceControlChain = voiceControlChain.then(async () => {
271
- try {
272
- if (!activeVoice) return
273
- av.setState("processing")
274
- const currentText = av.getPlainText()
275
- const currentAgent = local.agent.current()?.name ?? ""
276
- const availableAgents = local.agent.list().map((x) => x.name)
277
-
278
- const ctrl = await Voice.processVoiceControl({
279
- audio: segment.audio,
280
- apiKey: creds.apiKey,
281
- baseUrl: creds.baseUrl,
282
- model: resolved.control.model,
283
- currentText,
284
- currentAgent,
285
- availableAgents,
286
- sendEnabled: voiceSendEnabled(),
287
- })
288
-
289
- if (ctrl) {
290
- for (const action of ctrl.actions) {
291
- if (action.action === "edit") av.setText(action.text)
292
- else if (action.action === "send") {
293
- if (voiceSendEnabled() && av.getPlainText().trim()) await av.submit()
294
- else if (!av.getPlainText().trim()) av.showError(t("tui.voice.error.empty_send"))
295
- } else if (action.action === "agent") {
296
- av.switchAgent(action.agent)
297
- }
298
- }
299
- } else {
300
- av.showError(t("tui.voice.error.network"))
301
- }
302
- } finally {
303
- av.pending--
304
- if (activeVoice === av && voiceState() !== "speaking")
305
- av.setState(av.pending > 0 ? "processing" : "listening")
306
- if (!activeVoice && av.pending <= 0) av.setState("idle")
307
- }
308
- }).catch(() => {})
309
- } else {
310
- Voice.transcribeAudio({
311
- audio: segment.audio,
312
- apiKey: creds.apiKey,
313
- baseUrl: creds.baseUrl,
314
- model: resolved.asr.model,
315
- }).then((text) => {
316
- if (text) {
317
- if (voiceSendEnabled() && Voice.SEND_RE.test(text.replace(/[\s。.!!??,,]+$/g, "").trim())) {
318
- av.submit()
319
- } else {
320
- av.appendText(text.trim())
321
- }
322
- } else {
323
- av.showError(t("tui.voice.error.network"))
324
- }
325
- av.pending--
326
- if (activeVoice === av && voiceState() !== "speaking")
327
- av.setState(av.pending > 0 ? "processing" : "listening")
328
- if (!activeVoice && av.pending <= 0) av.setState("idle")
329
- }).catch(() => {
330
- av.pending--
331
- if (activeVoice === av && voiceState() !== "speaking")
332
- av.setState(av.pending > 0 ? "processing" : "listening")
333
- if (!activeVoice && av.pending <= 0) av.setState("idle")
334
- })
335
- }
336
- },
337
- onActiveChange: (active) => {
338
- if (active && activeVoice === av) av.setState("speaking")
339
- },
340
- onError: (err) => {
341
- const msg = err.message || ""
342
- if (msg.includes("no default audio") || msg.includes("not found") || msg.includes("Cannot open") || msg.includes("ALSA")) {
343
- av.showError(t("tui.voice.error.no_device"))
344
- } else {
345
- av.showError(`${t("tui.voice.error.recorder_failed")}: ${msg}`)
346
- }
347
- activeVoice = undefined
348
- av.setState("idle")
349
- },
350
- })
351
- if (!handle) {
352
- toast.show({ message: t("tui.voice.error.no_recorder"), variant: "error" })
353
- return
354
- }
355
- av.handle = handle
356
- activeVoice = av
357
- setVoiceState("listening")
358
- }
359
-
360
- const list = createMemo(() => props.placeholders?.normal ?? [])
361
- const shell = createMemo(() => props.placeholders?.shell ?? [])
362
- const [auto, setAuto] = createSignal<AutocompleteRef>()
363
- const [ghost, setGhost] = createSignal("")
364
- const hasRightContent = createMemo(() => Boolean(props.right))
365
-
366
- function promptModelWarning() {
367
- toast.show({
368
- variant: "warning",
369
- message: "Connect a provider to send prompts",
370
- duration: 3000,
371
- })
372
- if (sync.data.provider.length === 0) {
373
- dialog.replace(() => <DialogProviderConnect />)
374
- }
375
- }
376
-
377
- const textareaKeybindings = useTextareaKeybindings()
378
-
379
- const fileStyleId = syntax().getStyleId("extmark.file")!
380
- const agentStyleId = syntax().getStyleId("extmark.agent")!
381
- const pasteStyleId = syntax().getStyleId("extmark.paste")!
382
- let promptPartTypeId = 0
383
- const event = useEvent()
384
-
385
- event.on(TuiEvent.PromptAppend.type, (evt) => {
386
- if (!input || input.isDestroyed) return
387
- input.insertText(evt.properties.text)
388
- setTimeout(() => {
389
- // setTimeout is a workaround and needs to be addressed properly
390
- if (!input || input.isDestroyed) return
391
- input.getLayoutNode().markDirty()
392
- input.gotoBufferEnd()
393
- renderer.requestRender()
394
- }, 0)
395
- })
396
-
397
- createEffect(() => {
398
- if (props.disabled) input.cursorColor = theme.backgroundElement
399
- if (!props.disabled) input.cursorColor = theme.text
400
- })
401
-
402
- const lastUserMessage = createMemo(() => {
403
- if (!props.sessionID) return undefined
404
- const messages = sync.data.message[props.sessionID]?.["main"]
405
- if (!messages) return undefined
406
- return messages.findLast((m): m is UserMessage => m.role === "user")
407
- })
408
-
409
- // After the agent finishes a turn, predict the user's likely next prompt and
410
- // show it as ghost text in the empty input (accept with Tab). Only fires on
411
- // an idle transition while the input is empty so it never clobbers typing.
412
- let ghostRequest = 0
413
- async function fetchGhost(sessionID: string) {
414
- if (props.showPlaceholder === false) return
415
- const token = ++ghostRequest
416
- const userMessageID = lastUserMessage()?.id
417
- const res = await sdk.client.session.predict({ sessionID }).catch(() => undefined)
418
- const text = res?.data?.prediction?.trim()
419
- if (!text) return
420
- // Drop the result if anything that defined its context changed while the
421
- // request was in flight: superseded by a newer fetch, session switched, a
422
- // new run started, the conversation advanced, or the user began typing.
423
- if (token !== ghostRequest) return
424
- if (props.sessionID !== sessionID) return
425
- if (status().type !== "idle") return
426
- if (lastUserMessage()?.id !== userMessageID) return
427
- if (!input || input.isDestroyed || input.plainText !== "") return
428
- setGhost(text)
429
- }
430
- createEffect(
431
- on(
432
- () => status().type,
433
- (type, prev) => {
434
- if (type !== "idle") {
435
- // A new run started (or the session went non-idle): invalidate any
436
- // in-flight prediction and hide a stale suggestion.
437
- ghostRequest++
438
- if (ghost()) setGhost("")
439
- return
440
- }
441
- if (prev === "idle") return
442
- const sessionID = props.sessionID
443
- if (!sessionID || !input || input.isDestroyed || input.plainText !== "") return
444
- if (!lastUserMessage()) return
445
- fetchGhost(sessionID)
446
- },
447
- ),
448
- )
449
- // While a ghost suggestion is showing, suspend global command keybinds so Tab
450
- // reaches the textarea's onKeyDown (where we accept it) instead of being
451
- // consumed by the agent-cycle keybind. Global keyboard handlers run before
452
- // renderable handlers, so without this the suggestion can never be accepted.
453
- // The cleanup resumes keybinds on any dismissal (typing, accept, submit,
454
- // session change, status leaving idle).
455
- createEffect(() => {
456
- if (!ghost()) return
457
- command.keybinds(false)
458
- onCleanup(() => command.keybinds(true))
459
- })
460
-
461
- const usage = createMemo(() => {
462
- if (!props.sessionID) return
463
- const msg = sync.data.message[props.sessionID]?.["main"] ?? []
464
- const last = msg.findLast((item): item is AssistantMessage => item.role === "assistant" && item.tokens.output > 0)
465
- if (!last) return
466
-
467
- const tokens =
468
- last.tokens.input + last.tokens.output + last.tokens.reasoning + last.tokens.cache.read + last.tokens.cache.write
469
- if (tokens <= 0) return
470
-
471
- const model = sync.data.provider.find((item) => item.id === last.providerID)?.models[last.modelID]
472
- const pct = model?.limit.context ? `${Math.round((tokens / model.limit.context) * 100)}%` : undefined
473
- const cost = msg.reduce((sum, item) => sum + (item.role === "assistant" ? item.cost : 0), 0)
474
- return {
475
- context: pct ? `${Locale.number(tokens)} (${pct})` : Locale.number(tokens),
476
- cost: cost > 0 ? money.format(cost) : undefined,
477
- }
478
- })
479
-
480
- const [store, setStore] = createStore<{
481
- prompt: PromptInfo
482
- mode: "normal" | "shell"
483
- extmarkToPartIndex: Map<number, number>
484
- interrupt: number
485
- placeholder: number
486
- }>({
487
- placeholder: randomIndex(list().length),
488
- prompt: {
489
- input: "",
490
- parts: [],
491
- },
492
- mode: "normal",
493
- extmarkToPartIndex: new Map(),
494
- interrupt: 0,
495
- })
496
-
497
- createEffect(
498
- on(
499
- () => props.sessionID,
500
- () => {
501
- setGhost("")
502
- setStore("placeholder", randomIndex(list().length))
503
- },
504
- { defer: true },
505
- ),
506
- )
507
-
508
- // Initialize agent/model/variant from last user message when session changes
509
- let syncedSessionID: string | undefined
510
- createEffect(() => {
511
- const sessionID = props.sessionID
512
- const msg = lastUserMessage()
513
-
514
- if (sessionID !== syncedSessionID) {
515
- if (!sessionID || !msg) return
516
-
517
- syncedSessionID = sessionID
518
-
519
- // Only set agent if it's a primary agent (not a subagent)
520
- const isPrimaryAgent = local.agent.list().some((x) => x.name === msg.agent)
521
- if (msg.agent && isPrimaryAgent) {
522
- // Keep command line --agent if specified.
523
- if (!args.agent) local.agent.set(msg.agent)
524
- if (msg.model) {
525
- local.model.set(msg.model)
526
- local.model.variant.set(msg.model.variant)
527
- }
528
- }
529
- }
530
- })
531
-
532
- command.register(() => {
533
- return [
534
- {
535
- title: t("tui.command.prompt.clear.title"),
536
- value: "prompt.clear",
537
- category: "prompt",
538
- hidden: true,
539
- onSelect: (dialog) => {
540
- input.extmarks.clear()
541
- input.clear()
542
- dialog.clear()
543
- },
544
- },
545
- {
546
- title: t("tui.command.prompt.submit.title"),
547
- value: "prompt.submit",
548
- keybind: "input_submit",
549
- category: "prompt",
550
- hidden: true,
551
- onSelect: async (dialog) => {
552
- if (!input.focused) return
553
- const handled = await submit()
554
- if (!handled) return
555
-
556
- dialog.clear()
557
- },
558
- },
559
- {
560
- title: t("tui.command.prompt.paste.title"),
561
- value: "prompt.paste",
562
- keybind: "input_paste",
563
- category: "prompt",
564
- hidden: true,
565
- onSelect: async () => {
566
- await pasteFromClipboard()
567
- },
568
- },
569
- {
570
- title: t("tui.command.session.interrupt.title"),
571
- value: "session.interrupt",
572
- keybind: "session_interrupt",
573
- category: "session",
574
- hidden: true,
575
- enabled: status().type !== "idle",
576
- onSelect: (dialog) => {
577
- if (autocomplete.visible) return
578
- if (!input.focused) return
579
- // TODO: this should be its own command
580
- if (store.mode === "shell") {
581
- setStore("mode", "normal")
582
- return
583
- }
584
- if (!props.sessionID) return
585
-
586
- setStore("interrupt", store.interrupt + 1)
587
-
588
- setTimeout(() => {
589
- setStore("interrupt", 0)
590
- }, 5000)
591
-
592
- if (store.interrupt >= 2) {
593
- void sdk.client.session.abort({
594
- sessionID: props.sessionID,
595
- })
596
- setStore("interrupt", 0)
597
- }
598
- dialog.clear()
599
- },
600
- },
601
- {
602
- title: t("tui.command.prompt.editor.title"),
603
- category: "session",
604
- keybind: "editor_open",
605
- value: "prompt.editor",
606
- slash: {
607
- name: "editor",
608
- },
609
- onSelect: async (dialog) => {
610
- dialog.clear()
611
-
612
- // replace summarized text parts with the actual text
613
- const text = store.prompt.parts
614
- .filter((p) => p.type === "text")
615
- .reduce((acc, p) => {
616
- if (!p.source) return acc
617
- return acc.replace(p.source.text.value, p.text)
618
- }, store.prompt.input)
619
-
620
- const nonTextParts = store.prompt.parts.filter((p) => p.type !== "text")
621
-
622
- const value = text
623
- const content = await Editor.open({ value, renderer })
624
- if (!content) return
625
-
626
- input.setText(content)
627
-
628
- // Update positions for nonTextParts based on their location in new content
629
- // Filter out parts whose virtual text was deleted
630
- // this handles a case where the user edits the text in the editor
631
- // such that the virtual text moves around or is deleted
632
- const updatedNonTextParts = nonTextParts
633
- .map((part) => {
634
- let virtualText = ""
635
- if (part.type === "file" && part.source?.text) {
636
- virtualText = part.source.text.value
637
- } else if (part.type === "agent" && part.source) {
638
- virtualText = part.source.value
639
- }
640
-
641
- if (!virtualText) return part
642
-
643
- const newStart = content.indexOf(virtualText)
644
- // if the virtual text is deleted, remove the part
645
- if (newStart === -1) return null
646
-
647
- const newEnd = newStart + virtualText.length
648
-
649
- if (part.type === "file" && part.source?.text) {
650
- return {
651
- ...part,
652
- source: {
653
- ...part.source,
654
- text: {
655
- ...part.source.text,
656
- start: newStart,
657
- end: newEnd,
658
- },
659
- },
660
- }
661
- }
662
-
663
- if (part.type === "agent" && part.source) {
664
- return {
665
- ...part,
666
- source: {
667
- ...part.source,
668
- start: newStart,
669
- end: newEnd,
670
- },
671
- }
672
- }
673
-
674
- return part
675
- })
676
- .filter((part) => part !== null)
677
-
678
- setStore("prompt", {
679
- input: content,
680
- // keep only the non-text parts because the text parts were
681
- // already expanded inline
682
- parts: updatedNonTextParts,
683
- })
684
- restoreExtmarksFromParts(updatedNonTextParts)
685
- input.cursorOffset = Bun.stringWidth(content)
686
- },
687
- },
688
- {
689
- title: t("tui.command.prompt.skills.title"),
690
- value: "prompt.skills",
691
- category: "prompt",
692
- slash: {
693
- name: "skills",
694
- },
695
- onSelect: () => {
696
- dialog.replace(() => (
697
- <DialogSkill
698
- onSelect={(skill) => {
699
- input.setText(`/${skill} `)
700
- setStore("prompt", {
701
- input: `/${skill} `,
702
- parts: [],
703
- })
704
- input.gotoBufferEnd()
705
- }}
706
- />
707
- ))
708
- },
709
- },
710
- {
711
- title: t("tui.command.consent.revoke.title"),
712
- value: "consent.revoke",
713
- category: "prompt",
714
- slash: {
715
- name: "revoke-consent",
716
- },
717
- onSelect: (dialog) => {
718
- kv.delete(FREE_AGREEMENT_KEY)
719
- dialog.clear()
720
- toast.show({
721
- message: t("tui.consent.revoked"),
722
- variant: "info",
723
- duration: 3000,
724
- })
725
- },
726
- },
727
- {
728
- title: voiceEnabled() ? t("tui.command.voice.toggle.title_on") : t("tui.command.voice.toggle.title_off"),
729
- value: "voice.toggle",
730
- category: "prompt",
731
- slash: {
732
- name: "voice",
733
- },
734
- onSelect: () => {
735
- const next = !voiceEnabled()
736
- kv.set("voice_enabled", next)
737
- if (!next && activeVoice) void voiceToggle()
738
- toast.show({
739
- message: next ? t("tui.voice.enabled") : t("tui.voice.disabled"),
740
- variant: "info",
741
- duration: 3000,
742
- })
743
- },
744
- },
745
- {
746
- title: voiceSendEnabled() ? t("tui.command.voice.send.title_on") : t("tui.command.voice.send.title_off"),
747
- value: "voice.send",
748
- category: "prompt",
749
- slash: {
750
- name: "voice-send",
751
- },
752
- onSelect: () => {
753
- const next = !voiceSendEnabled()
754
- kv.set("voice_send_command", next)
755
- toast.show({
756
- message: next ? t("tui.voice.send.enabled") : t("tui.voice.send.disabled"),
757
- variant: "info",
758
- duration: 3000,
759
- })
760
- },
761
- },
762
- {
763
- title: voiceControlEnabled() ? t("tui.command.voice.control.title_on") : t("tui.command.voice.control.title_off"),
764
- value: "voice.control",
765
- category: "prompt",
766
- slash: {
767
- name: "voice-control",
768
- },
769
- onSelect: () => {
770
- const next = !voiceControlEnabled()
771
- kv.set("voice_control_enabled", next)
772
- toast.show({
773
- message: next ? t("tui.voice.control.enabled") : t("tui.voice.control.disabled"),
774
- variant: "info",
775
- duration: 3000,
776
- })
777
- },
778
- },
779
- ]
780
- })
781
-
782
- const ref: PromptRef = {
783
- get focused() {
784
- return input.focused
785
- },
786
- get current() {
787
- return store.prompt
788
- },
789
- focus() {
790
- input.focus()
791
- },
792
- blur() {
793
- input.blur()
794
- },
795
- set(prompt) {
796
- input.setText(prompt.input)
797
- setStore("prompt", prompt)
798
- restoreExtmarksFromParts(prompt.parts)
799
- input.gotoBufferEnd()
800
- },
801
- reset() {
802
- input.clear()
803
- input.extmarks.clear()
804
- setStore("prompt", {
805
- input: "",
806
- parts: [],
807
- })
808
- setStore("extmarkToPartIndex", new Map())
809
- },
810
- submit() {
811
- void submit()
812
- },
813
- paste() {
814
- void pasteFromClipboard()
815
- },
816
- }
817
-
818
- onMount(() => {
819
- const saved = stashed
820
- stashed = undefined
821
- if (store.prompt.input) return
822
- if (saved && saved.prompt.input) {
823
- input.setText(saved.prompt.input)
824
- setStore("prompt", saved.prompt)
825
- restoreExtmarksFromParts(saved.prompt.parts)
826
- input.cursorOffset = saved.cursor
827
- }
828
- })
829
-
830
- onCleanup(() => {
831
- if (store.prompt.input) {
832
- stashed = { prompt: unwrap(store.prompt), cursor: input.cursorOffset }
833
- }
834
- props.ref?.(undefined)
835
- })
836
-
837
- createEffect(() => {
838
- if (!input || input.isDestroyed) return
839
- if (props.visible === false || dialog.stack.length > 0) {
840
- if (input.focused) input.blur()
841
- return
842
- }
843
-
844
- // Slot/plugin updates can remount the background prompt while a dialog is open.
845
- // Keep focus with the dialog and let the prompt reclaim it after the dialog closes.
846
- if (!input.focused) input.focus()
847
- })
848
-
849
- createEffect(() => {
850
- if (!input || input.isDestroyed) return
851
- const capture =
852
- store.mode === "normal"
853
- ? auto()?.visible
854
- ? (["escape", "navigate", "submit", "tab"] as const)
855
- : (["tab"] as const)
856
- : undefined
857
- input.traits = {
858
- capture,
859
- suspend: !!props.disabled || store.mode === "shell",
860
- status: store.mode === "shell" ? "SHELL" : undefined,
861
- }
862
- })
863
-
864
- function restoreExtmarksFromParts(parts: PromptInfo["parts"]) {
865
- input.extmarks.clear()
866
- setStore("extmarkToPartIndex", new Map())
867
-
868
- parts.forEach((part, partIndex) => {
869
- let start = 0
870
- let end = 0
871
- let virtualText = ""
872
- let styleId: number | undefined
873
-
874
- if (part.type === "file" && part.source?.text) {
875
- start = part.source.text.start
876
- end = part.source.text.end
877
- virtualText = part.source.text.value
878
- styleId = fileStyleId
879
- } else if (part.type === "agent" && part.source) {
880
- start = part.source.start
881
- end = part.source.end
882
- virtualText = part.source.value
883
- styleId = agentStyleId
884
- } else if (part.type === "text" && part.source?.text) {
885
- start = part.source.text.start
886
- end = part.source.text.end
887
- virtualText = part.source.text.value
888
- styleId = pasteStyleId
889
- }
890
-
891
- if (virtualText) {
892
- const extmarkId = input.extmarks.create({
893
- start,
894
- end,
895
- virtual: true,
896
- styleId,
897
- typeId: promptPartTypeId,
898
- })
899
- setStore("extmarkToPartIndex", (map: Map<number, number>) => {
900
- const newMap = new Map(map)
901
- newMap.set(extmarkId, partIndex)
902
- return newMap
903
- })
904
- }
905
- })
906
- }
907
-
908
- function syncExtmarksWithPromptParts() {
909
- const allExtmarks = input.extmarks.getAllForTypeId(promptPartTypeId)
910
- setStore(
911
- produce((draft) => {
912
- const newMap = new Map<number, number>()
913
- const newParts: typeof draft.prompt.parts = []
914
-
915
- for (const extmark of allExtmarks) {
916
- const partIndex = draft.extmarkToPartIndex.get(extmark.id)
917
- if (partIndex !== undefined) {
918
- const part = draft.prompt.parts[partIndex]
919
- if (part) {
920
- if (part.type === "agent" && part.source) {
921
- part.source.start = extmark.start
922
- part.source.end = extmark.end
923
- } else if (part.type === "file" && part.source?.text) {
924
- part.source.text.start = extmark.start
925
- part.source.text.end = extmark.end
926
- } else if (part.type === "text" && part.source?.text) {
927
- part.source.text.start = extmark.start
928
- part.source.text.end = extmark.end
929
- }
930
- newMap.set(extmark.id, newParts.length)
931
- newParts.push(part)
932
- }
933
- }
934
- }
935
-
936
- draft.extmarkToPartIndex = newMap
937
- draft.prompt.parts = newParts
938
- }),
939
- )
940
- }
941
-
942
- command.register(() => [
943
- {
944
- title: t("tui.command.prompt.stash.title"),
945
- value: "prompt.stash",
946
- category: "prompt",
947
- enabled: !!store.prompt.input,
948
- onSelect: (dialog) => {
949
- if (!store.prompt.input) return
950
- stash.push({
951
- input: store.prompt.input,
952
- parts: store.prompt.parts,
953
- })
954
- input.extmarks.clear()
955
- input.clear()
956
- setStore("prompt", { input: "", parts: [] })
957
- setStore("extmarkToPartIndex", new Map())
958
- dialog.clear()
959
- },
960
- },
961
- {
962
- title: t("tui.command.prompt.stash.pop.title"),
963
- value: "prompt.stash.pop",
964
- category: "prompt",
965
- enabled: stash.list().length > 0,
966
- onSelect: (dialog) => {
967
- const entry = stash.pop()
968
- if (entry) {
969
- input.setText(entry.input)
970
- setStore("prompt", { input: entry.input, parts: entry.parts })
971
- restoreExtmarksFromParts(entry.parts)
972
- input.gotoBufferEnd()
973
- }
974
- dialog.clear()
975
- },
976
- },
977
- {
978
- title: t("tui.command.prompt.stash.list.title"),
979
- value: "prompt.stash.list",
980
- category: "prompt",
981
- enabled: stash.list().length > 0,
982
- onSelect: (dialog) => {
983
- dialog.replace(() => (
984
- <DialogStash
985
- onSelect={(entry) => {
986
- input.setText(entry.input)
987
- setStore("prompt", { input: entry.input, parts: entry.parts })
988
- restoreExtmarksFromParts(entry.parts)
989
- input.gotoBufferEnd()
990
- }}
991
- />
992
- ))
993
- },
994
- },
995
- ])
996
-
997
- // While the free-model agreement dialog is open, ignore any further submit()
998
- // calls. Enter triggers submit twice (the input_submit keybind plus the
999
- // textarea's deferred onSubmit), and without this guard the deferred call can
1000
- // interleave with the post-accept re-submit and drop the user's message.
1001
- let agreementPending = false
1002
- async function submit() {
1003
- if (agreementPending) return false
1004
- setGhost("")
1005
- // IME: double-defer may fire before onContentChange flushes the last
1006
- // composed character (e.g. Korean hangul) to the store, so read
1007
- // plainText directly and sync before any downstream reads.
1008
- if (input && !input.isDestroyed && input.plainText !== store.prompt.input) {
1009
- setStore("prompt", "input", input.plainText)
1010
- syncExtmarksWithPromptParts()
1011
- }
1012
- if (props.disabled) return false
1013
- if (autocomplete?.visible) return false
1014
- if (!store.prompt.input) return false
1015
- const agent = local.agent.current()
1016
- if (!agent) return false
1017
- const trimmed = store.prompt.input.trim()
1018
- if (trimmed === "exit" || trimmed === "quit" || trimmed === ":q") {
1019
- void exit()
1020
- return true
1021
- }
1022
- const selectedModel = local.model.current()
1023
- if (!selectedModel) {
1024
- void promptModelWarning()
1025
- return false
1026
- }
1027
-
1028
- // Free models require a one-time acknowledgment of the terms and privacy
1029
- // policy. Gate submission until the user accepts; the flag is stored in KV.
1030
- const isFreeModel = FREE_MODEL_IDS.has(selectedModel.modelID)
1031
- if (isFreeModel && !kv.get(FREE_AGREEMENT_KEY)) {
1032
- agreementPending = true
1033
- DialogAgreement.show(dialog, {
1034
- onConfirm: () => {
1035
- kv.set(FREE_AGREEMENT_KEY, true)
1036
- void submit()
1037
- },
1038
- // Fires on any dismissal (confirm, cancel, esc, click-outside). Reset
1039
- // the guard here so submission is unblocked once the dialog is gone.
1040
- onClose: () => {
1041
- agreementPending = false
1042
- },
1043
- })
1044
- return false
1045
- }
1046
-
1047
- const workspaceSession = props.sessionID ? sync.session.get(props.sessionID) : undefined
1048
- const workspaceID = workspaceSession?.workspaceID
1049
- const workspaceStatus = workspaceID ? (project.workspace.status(workspaceID) ?? "error") : undefined
1050
- if (props.sessionID && workspaceID && workspaceStatus !== "connected") {
1051
- dialog.replace(() => (
1052
- <DialogWorkspaceUnavailable
1053
- onRestore={() => {
1054
- dialog.replace(() => (
1055
- <DialogWorkspaceCreate
1056
- onSelect={(nextWorkspaceID) =>
1057
- restoreWorkspaceSession({
1058
- dialog,
1059
- sdk,
1060
- sync,
1061
- project,
1062
- toast,
1063
- workspaceID: nextWorkspaceID,
1064
- sessionID: props.sessionID!,
1065
- })
1066
- }
1067
- />
1068
- ))
1069
- }}
1070
- />
1071
- ))
1072
- return false
1073
- }
1074
-
1075
- let sessionID = props.sessionID
1076
- if (sessionID == null) {
1077
- const res = await sdk.client.session.create({ workspace: props.workspaceID })
1078
-
1079
- if (res.error) {
1080
- console.log("Creating a session failed:", res.error)
1081
-
1082
- toast.show({
1083
- message: "Creating a session failed. Open console for more details.",
1084
- variant: "error",
1085
- })
1086
-
1087
- return true
1088
- }
1089
-
1090
- sessionID = res.data.id
1091
- }
1092
-
1093
- const messageID = MessageID.ascending()
1094
-
1095
- // Expand pasted text inline before submitting. Extmark offsets are
1096
- // display-width based while plainText is UTF-16, so expandPlaceholders
1097
- // bridges the two coordinate systems (otherwise CJK content desyncs them).
1098
- const marks = input.extmarks
1099
- .getAllForTypeId(promptPartTypeId)
1100
- .flatMap((extmark: { id: number; start: number; end: number }) => {
1101
- const partIndex = store.extmarkToPartIndex.get(extmark.id)
1102
- if (partIndex === undefined) return []
1103
- const part = store.prompt.parts[partIndex]
1104
- if (part?.type !== "text" || !part.text) return []
1105
- return [{ start: extmark.start, end: extmark.end, text: part.text }]
1106
- })
1107
- const inputText = expandPlaceholders(store.prompt.input, marks)
1108
-
1109
- // Filter out text parts (pasted content) since they're now expanded inline
1110
- const nonTextParts = store.prompt.parts.filter((part) => part.type !== "text")
1111
-
1112
- // Capture mode before it gets reset
1113
- const currentMode = store.mode
1114
- const variant = local.model.variant.current()
1115
-
1116
- const clientSlash = inputText.startsWith("/")
1117
- ? command.slashes().find((s) => s.display === inputText.trim())
1118
- : undefined
1119
-
1120
- if (store.mode === "shell") {
1121
- void sdk.client.session.shell({
1122
- sessionID,
1123
- agent: agent.name,
1124
- model: {
1125
- providerID: selectedModel.providerID,
1126
- modelID: selectedModel.modelID,
1127
- },
1128
- command: inputText,
1129
- })
1130
- setStore("mode", "normal")
1131
- } else if (clientSlash) {
1132
- clientSlash.onSelect?.()
1133
- } else if (
1134
- inputText.startsWith("/") &&
1135
- iife(() => {
1136
- const firstLine = inputText.split("\n")[0]
1137
- const command = firstLine.split(" ")[0].slice(1)
1138
- return sync.data.command.some((x) => x.name === command)
1139
- })
1140
- ) {
1141
- // Parse command from first line, preserve multi-line content in arguments
1142
- const firstLineEnd = inputText.indexOf("\n")
1143
- const firstLine = firstLineEnd === -1 ? inputText : inputText.slice(0, firstLineEnd)
1144
- const [command, ...firstLineArgs] = firstLine.split(" ")
1145
- const restOfInput = firstLineEnd === -1 ? "" : inputText.slice(firstLineEnd + 1)
1146
- const args = firstLineArgs.join(" ") + (restOfInput ? "\n" + restOfInput : "")
1147
-
1148
- void sdk.client.session.command({
1149
- sessionID,
1150
- command: command.slice(1),
1151
- arguments: args,
1152
- agent: agent.name,
1153
- model: `${selectedModel.providerID}/${selectedModel.modelID}`,
1154
- messageID,
1155
- variant,
1156
- parts: nonTextParts
1157
- .filter((x) => x.type === "file")
1158
- .map((x) => ({
1159
- id: PartID.ascending(),
1160
- ...x,
1161
- })),
1162
- })
1163
- } else {
1164
- sdk.client.session
1165
- .promptAsync({
1166
- sessionID,
1167
- ...selectedModel,
1168
- messageID,
1169
- agent: agent.name,
1170
- model: selectedModel,
1171
- variant,
1172
- parts: [
1173
- {
1174
- id: PartID.ascending(),
1175
- type: "text",
1176
- text: inputText,
1177
- },
1178
- ...nonTextParts.map(assign),
1179
- ],
1180
- })
1181
- .catch((err) => {
1182
- toast.show({
1183
- message: err instanceof Error ? err.message : "Failed to send message",
1184
- variant: "error",
1185
- })
1186
- })
1187
- }
1188
- history.append({
1189
- ...store.prompt,
1190
- mode: currentMode,
1191
- })
1192
- input.extmarks.clear()
1193
- setStore("prompt", {
1194
- input: "",
1195
- parts: [],
1196
- })
1197
- setStore("extmarkToPartIndex", new Map())
1198
- props.onSubmit?.()
1199
-
1200
- // temporary hack to make sure the message is sent
1201
- if (!props.sessionID)
1202
- setTimeout(() => {
1203
- route.navigate({
1204
- type: "session",
1205
- sessionID,
1206
- })
1207
- }, 50)
1208
- input.clear()
1209
- return true
1210
- }
1211
- const exit = useExit()
1212
-
1213
- function pasteText(text: string, virtualText: string) {
1214
- const currentOffset = input.visualCursor.offset
1215
- const extmarkStart = currentOffset
1216
- const extmarkEnd = extmarkStart + virtualText.length
1217
-
1218
- input.insertText(virtualText + " ")
1219
-
1220
- const extmarkId = input.extmarks.create({
1221
- start: extmarkStart,
1222
- end: extmarkEnd,
1223
- virtual: true,
1224
- styleId: pasteStyleId,
1225
- typeId: promptPartTypeId,
1226
- })
1227
-
1228
- setStore(
1229
- produce((draft) => {
1230
- const partIndex = draft.prompt.parts.length
1231
- draft.prompt.parts.push({
1232
- type: "text" as const,
1233
- text,
1234
- source: {
1235
- text: {
1236
- start: extmarkStart,
1237
- end: extmarkEnd,
1238
- value: virtualText,
1239
- },
1240
- },
1241
- })
1242
- draft.extmarkToPartIndex.set(extmarkId, partIndex)
1243
- }),
1244
- )
1245
- }
1246
-
1247
- async function pastePlainText(normalizedText: string) {
1248
- const pastedContent = normalizedText.trim()
1249
- if (!pastedContent) return
1250
-
1251
- const filepath = iife(() => {
1252
- const raw = pastedContent.replace(/^['"]+|['"]+$/g, "")
1253
- if (raw.startsWith("file://")) {
1254
- try {
1255
- return fileURLToPath(raw)
1256
- } catch {}
1257
- }
1258
- if (process.platform === "win32") return raw
1259
- return raw.replace(/\\(.)/g, "$1")
1260
- })
1261
- const isUrl = /^(https?):\/\//.test(filepath)
1262
- if (!isUrl) {
1263
- try {
1264
- const mime = await Filesystem.mimeType(filepath)
1265
- const filename = path.basename(filepath)
1266
- // Handle SVG as raw text content, not as base64 image
1267
- if (mime === "image/svg+xml") {
1268
- const content = await Filesystem.readText(filepath).catch(() => {})
1269
- if (content) {
1270
- pasteText(content, `[SVG: ${filename ?? "image"}]`)
1271
- return
1272
- }
1273
- }
1274
- if (mime.startsWith("image/") || mime === "application/pdf") {
1275
- const content = await Filesystem.readArrayBuffer(filepath)
1276
- .then((buffer) => Buffer.from(buffer).toString("base64"))
1277
- .catch(() => {})
1278
- if (content) {
1279
- await pasteAttachment({
1280
- filename,
1281
- filepath,
1282
- mime,
1283
- content,
1284
- })
1285
- return
1286
- }
1287
- }
1288
- } catch {}
1289
- }
1290
-
1291
- const lineCount = (pastedContent.match(/\n/g)?.length ?? 0) + 1
1292
- if ((lineCount >= 3 || pastedContent.length > 150) && !sync.data.config.experimental?.disable_paste_summary) {
1293
- pasteText(pastedContent, `[Pasted ~${lineCount} lines]`)
1294
- return
1295
- }
1296
-
1297
- input.insertText(normalizedText)
1298
-
1299
- // Force layout update and render for the pasted content
1300
- setTimeout(() => {
1301
- // setTimeout is a workaround and needs to be addressed properly
1302
- if (!input || input.isDestroyed) return
1303
- input.getLayoutNode().markDirty()
1304
- renderer.requestRender()
1305
- }, 0)
1306
- }
1307
-
1308
- async function pasteFromClipboard() {
1309
- if (props.disabled) return
1310
- const content = await Clipboard.read()
1311
- if (!content) return
1312
- if (content.mime.startsWith("image/")) {
1313
- await pasteAttachment({
1314
- filename: "clipboard",
1315
- mime: content.mime,
1316
- content: content.data,
1317
- })
1318
- return
1319
- }
1320
- await pastePlainText(content.data.replace(/\r\n/g, "\n").replace(/\r/g, "\n"))
1321
- }
1322
-
1323
- async function pasteAttachment(file: { filename?: string; filepath?: string; content: string; mime: string }) {
1324
- const currentOffset = input.visualCursor.offset
1325
- const extmarkStart = currentOffset
1326
- const pdf = file.mime === "application/pdf"
1327
- const count = store.prompt.parts.filter((x) => {
1328
- if (x.type !== "file") return false
1329
- if (pdf) return x.mime === "application/pdf"
1330
- return x.mime.startsWith("image/")
1331
- }).length
1332
- const virtualText = pdf ? `[PDF ${count + 1}]` : `[Image ${count + 1}]`
1333
- const extmarkEnd = extmarkStart + virtualText.length
1334
- const textToInsert = virtualText + " "
1335
-
1336
- input.insertText(textToInsert)
1337
-
1338
- const extmarkId = input.extmarks.create({
1339
- start: extmarkStart,
1340
- end: extmarkEnd,
1341
- virtual: true,
1342
- styleId: pasteStyleId,
1343
- typeId: promptPartTypeId,
1344
- })
1345
-
1346
- const part: Omit<FilePart, "id" | "messageID" | "sessionID"> = {
1347
- type: "file" as const,
1348
- mime: file.mime,
1349
- filename: file.filename,
1350
- url: `data:${file.mime};base64,${file.content}`,
1351
- source: {
1352
- type: "file",
1353
- path: file.filepath ?? file.filename ?? "",
1354
- text: {
1355
- start: extmarkStart,
1356
- end: extmarkEnd,
1357
- value: virtualText,
1358
- },
1359
- },
1360
- }
1361
- setStore(
1362
- produce((draft) => {
1363
- const partIndex = draft.prompt.parts.length
1364
- draft.prompt.parts.push(part)
1365
- draft.extmarkToPartIndex.set(extmarkId, partIndex)
1366
- }),
1367
- )
1368
- return
1369
- }
1370
-
1371
- const highlight = createMemo(() => {
1372
- if (keybind.leader) return theme.border
1373
- if (store.mode === "shell") return theme.primary
1374
- const agent = local.agent.current()
1375
- if (!agent) return theme.border
1376
- return local.agent.color(agent.name)
1377
- })
1378
-
1379
- const showVariant = createMemo(() => {
1380
- const variants = local.model.variant.list()
1381
- if (variants.length === 0) return false
1382
- const current = local.model.variant.current()
1383
- return !!current
1384
- })
1385
-
1386
- const agentMetaAlpha = createFadeIn(() => !!local.agent.current(), animationsEnabled)
1387
- const modelMetaAlpha = createFadeIn(() => !!local.agent.current() && store.mode === "normal", animationsEnabled)
1388
- const variantMetaAlpha = createFadeIn(
1389
- () => !!local.agent.current() && store.mode === "normal" && showVariant(),
1390
- animationsEnabled,
1391
- )
1392
- const borderHighlight = createMemo(() => tint(theme.border, highlight(), agentMetaAlpha()))
1393
-
1394
- const placeholderText = createMemo(() => {
1395
- if (props.showPlaceholder === false) return undefined
1396
- if (store.mode === "normal" && ghost()) return t("tui.prompt.ghost", { prediction: ghost() })
1397
- if (store.mode === "shell") {
1398
- if (!shell().length) return undefined
1399
- return t("tui.prompt.placeholder.shell", { example: shell()[store.placeholder % shell().length] })
1400
- }
1401
- if (!list().length) return undefined
1402
- return t("tui.prompt.placeholder.normal", { example: list()[store.placeholder % list().length] })
1403
- })
1404
-
1405
- const spinnerDef = createMemo(() => {
1406
- const agent = local.agent.current()
1407
- const color = agent ? local.agent.color(agent.name) : theme.border
1408
- return {
1409
- frames: createFrames({
1410
- color,
1411
- style: "plane",
1412
- width: 14,
1413
- holdStart: 8,
1414
- holdEnd: 8,
1415
- inactiveFactor: 0.6,
1416
- // enableFading: false,
1417
- minAlpha: 0.3,
1418
- }),
1419
- color: createColors({
1420
- color,
1421
- style: "plane",
1422
- holdStart: 8,
1423
- holdEnd: 8,
1424
- inactiveFactor: 0.6,
1425
- // enableFading: false,
1426
- minAlpha: 0.3,
1427
- }),
1428
- }
1429
- })
1430
-
1431
- return (
1432
- <>
1433
- <Autocomplete
1434
- sessionID={props.sessionID}
1435
- ref={(r) => {
1436
- autocomplete = r
1437
- setAuto(() => r)
1438
- }}
1439
- anchor={() => anchor}
1440
- input={() => input}
1441
- setPrompt={(cb) => {
1442
- setStore("prompt", produce(cb))
1443
- }}
1444
- setExtmark={(partIndex, extmarkId) => {
1445
- setStore("extmarkToPartIndex", (map: Map<number, number>) => {
1446
- const newMap = new Map(map)
1447
- newMap.set(extmarkId, partIndex)
1448
- return newMap
1449
- })
1450
- }}
1451
- value={store.prompt.input}
1452
- fileStyleId={fileStyleId}
1453
- agentStyleId={agentStyleId}
1454
- promptPartTypeId={() => promptPartTypeId}
1455
- />
1456
- <box ref={(r) => (anchor = r)} visible={props.visible !== false}>
1457
- <box
1458
- border={["left"]}
1459
- borderColor={borderHighlight()}
1460
- customBorderChars={{
1461
- ...SplitBorder.customBorderChars,
1462
- bottomLeft: "╹",
1463
- }}
1464
- >
1465
- <box
1466
- paddingLeft={2}
1467
- paddingRight={2}
1468
- paddingTop={1}
1469
- flexShrink={0}
1470
- backgroundColor={theme.backgroundElement}
1471
- flexGrow={1}
1472
- >
1473
- <textarea
1474
- placeholder={placeholderText()}
1475
- placeholderColor={theme.textMuted}
1476
- textColor={keybind.leader ? theme.textMuted : theme.text}
1477
- focusedTextColor={keybind.leader ? theme.textMuted : theme.text}
1478
- minHeight={1}
1479
- maxHeight={6}
1480
- onContentChange={() => {
1481
- const value = input.plainText
1482
- if (value !== "" && ghost()) setGhost("")
1483
- setStore("prompt", "input", value)
1484
- autocomplete.onInput(value)
1485
- syncExtmarksWithPromptParts()
1486
- }}
1487
- keyBindings={textareaKeybindings()}
1488
- onKeyDown={async (e) => {
1489
- if (props.disabled) {
1490
- e.preventDefault()
1491
- return
1492
- }
1493
- // Handle Ctrl+V for terminals that forward it to the app as a raw
1494
- // keypress (common on macOS/Linux). The textarea has no built-in
1495
- // paste action, so without this nothing gets inserted. Terminals
1496
- // that handle paste natively (e.g. Windows Terminal 1.25+) emit a
1497
- // bracketed paste instead and never reach this path.
1498
- if (keybind.match("input_paste", e)) {
1499
- e.preventDefault()
1500
- await pasteFromClipboard()
1501
- return
1502
- }
1503
- if (keybind.match("input_clear", e) && store.prompt.input !== "") {
1504
- input.clear()
1505
- input.extmarks.clear()
1506
- setStore("prompt", {
1507
- input: "",
1508
- parts: [],
1509
- })
1510
- setStore("extmarkToPartIndex", new Map())
1511
- return
1512
- }
1513
- if (keybind.match("app_exit", e)) {
1514
- if (store.prompt.input === "") {
1515
- if (props.sessionID && status().type !== "idle") {
1516
- void sdk.client.session.abort({ sessionID: props.sessionID })
1517
- e.preventDefault()
1518
- return
1519
- }
1520
- await exit()
1521
- // Don't preventDefault - let textarea potentially handle the event
1522
- e.preventDefault()
1523
- return
1524
- }
1525
- }
1526
- if (e.name === "!" && input.visualCursor.offset === 0) {
1527
- setStore("placeholder", randomIndex(shell().length))
1528
- setStore("mode", "shell")
1529
- e.preventDefault()
1530
- return
1531
- }
1532
- if (store.mode === "shell") {
1533
- if ((e.name === "backspace" && input.visualCursor.offset === 0) || e.name === "escape") {
1534
- setStore("mode", "normal")
1535
- e.preventDefault()
1536
- return
1537
- }
1538
- }
1539
- if (ghost() && store.mode === "normal" && !autocomplete.visible && input.plainText === "") {
1540
- if (e.name === "tab") {
1541
- const text = ghost()
1542
- setGhost("")
1543
- input.setText(text)
1544
- setStore("prompt", "input", text)
1545
- input.gotoBufferEnd()
1546
- e.preventDefault()
1547
- return
1548
- }
1549
- if (e.name === "escape") {
1550
- setGhost("")
1551
- e.preventDefault()
1552
- return
1553
- }
1554
- }
1555
- if (store.mode === "normal") autocomplete.onKeyDown(e)
1556
- if (!autocomplete.visible) {
1557
- if (
1558
- (keybind.match("history_previous", e) && input.cursorOffset === 0) ||
1559
- (keybind.match("history_next", e) && input.cursorOffset === input.plainText.length)
1560
- ) {
1561
- const direction = keybind.match("history_previous", e) ? -1 : 1
1562
- const item = history.move(direction, input.plainText)
1563
-
1564
- if (item) {
1565
- input.setText(item.input)
1566
- setStore("prompt", item)
1567
- setStore("mode", item.mode ?? "normal")
1568
- restoreExtmarksFromParts(item.parts)
1569
- e.preventDefault()
1570
- if (direction === -1) input.cursorOffset = 0
1571
- if (direction === 1) input.cursorOffset = input.plainText.length
1572
- }
1573
- return
1574
- }
1575
-
1576
- if (keybind.match("history_previous", e) && input.visualCursor.visualRow === 0) input.cursorOffset = 0
1577
- if (keybind.match("history_next", e) && input.visualCursor.visualRow === input.height - 1)
1578
- input.cursorOffset = input.plainText.length
1579
- }
1580
- }}
1581
- onSubmit={() => {
1582
- // IME: double-defer so the last composed character (e.g. Korean
1583
- // hangul) is flushed to plainText before we read it for submission.
1584
- setTimeout(() => setTimeout(() => submit(), 0), 0)
1585
- }}
1586
- onPaste={async (event: PasteEvent) => {
1587
- if (props.disabled) {
1588
- event.preventDefault()
1589
- return
1590
- }
1591
-
1592
- // Normalize line endings at the boundary
1593
- // Windows ConPTY/Terminal often sends CR-only newlines in bracketed paste
1594
- // Replace CRLF first, then any remaining CR
1595
- const normalizedText = decodePasteBytes(event.bytes).replace(/\r\n/g, "\n").replace(/\r/g, "\n")
1596
-
1597
- // Windows Terminal <1.25 can surface image-only clipboard as an
1598
- // empty bracketed paste. Windows Terminal 1.25+ does not.
1599
- if (!normalizedText.trim()) {
1600
- command.trigger("prompt.paste")
1601
- return
1602
- }
1603
-
1604
- // Once we cross an async boundary below, the terminal may perform its
1605
- // default paste unless we suppress it first and handle insertion ourselves.
1606
- event.preventDefault()
1607
- await pastePlainText(normalizedText)
1608
- }}
1609
- ref={(r: TextareaRenderable) => {
1610
- input = r
1611
- if (promptPartTypeId === 0) {
1612
- promptPartTypeId = input.extmarks.registerType("prompt-part")
1613
- }
1614
- props.ref?.(ref)
1615
- setTimeout(() => {
1616
- // setTimeout is a workaround and needs to be addressed properly
1617
- if (!input || input.isDestroyed) return
1618
- input.cursorColor = theme.text
1619
- }, 0)
1620
- }}
1621
- onMouseDown={(r: MouseEvent) => r.target?.focus()}
1622
- focusedBackgroundColor={theme.backgroundElement}
1623
- cursorColor={theme.text}
1624
- syntaxStyle={syntax()}
1625
- />
1626
- <box flexDirection="row" flexShrink={0} paddingTop={1} gap={1} justifyContent="space-between">
1627
- <box flexDirection="row" gap={1}>
1628
- <Show when={local.agent.current()} fallback={<box height={1} />}>
1629
- {(agent) => (
1630
- <>
1631
- <text fg={fadeColor(highlight(), agentMetaAlpha())}>
1632
- {store.mode === "shell" ? "Shell" : Locale.titlecase(agent().name)}
1633
- </text>
1634
- <Show when={store.mode === "normal"}>
1635
- <box flexDirection="row" gap={1}>
1636
- <text fg={fadeColor(theme.textMuted, modelMetaAlpha())}>·</text>
1637
- <text
1638
- flexShrink={0}
1639
- fg={fadeColor(keybind.leader ? theme.textMuted : theme.text, modelMetaAlpha())}
1640
- >
1641
- {local.model.parsed().model}
1642
- </text>
1643
- {/* Hide provider label for sleepy-auto since model name already contains "Sleepy" */}
1644
- <Show when={!(local.model.current()?.providerID === "sleepy" && local.model.current()?.modelID === "sleepy-auto")}>
1645
- <text fg={fadeColor(theme.textMuted, modelMetaAlpha())}>
1646
- {currentProviderLabel()}
1647
- </text>
1648
- </Show>
1649
- <Show when={showVariant()}>
1650
- <text fg={fadeColor(theme.textMuted, variantMetaAlpha())}>·</text>
1651
- <text>
1652
- <span style={{ fg: fadeColor(theme.warning, variantMetaAlpha()), bold: true }}>
1653
- {local.model.variant.current()}
1654
- </span>
1655
- </text>
1656
- </Show>
1657
- </box>
1658
- </Show>
1659
- </>
1660
- )}
1661
- </Show>
1662
- <Show when={local.neverAsk.current()}>
1663
- <text>
1664
- <span style={{ fg: theme.error, bold: true }}>«never-ask»</span>
1665
- </text>
1666
- </Show>
1667
- </box>
1668
- <box flexDirection="row" gap={1} alignItems="center">
1669
- <Show when={hasRightContent()}>
1670
- {props.right}
1671
- </Show>
1672
- <Show when={voiceEnabled()}>
1673
- <Switch>
1674
- <Match when={voiceState() === "idle"}>
1675
- <text fg={theme.textMuted} selectable={false} onMouseUp={() => voiceToggle()}>
1676
- {"[ 🎙 Voice ]"}
1677
- </text>
1678
- </Match>
1679
- <Match when={voiceState() === "listening"}>
1680
- <text fg={theme.primary} selectable={false} onMouseUp={() => voiceToggle()}>
1681
- {"[ 🎙 -:-- ]"}
1682
- </text>
1683
- </Match>
1684
- <Match when={voiceState() === "speaking"}>
1685
- <text fg={theme.primary} selectable={false} onMouseUp={() => voiceToggle()}>
1686
- {`[ 🎙 ${Math.floor(voiceElapsed() / 60)}:${String(voiceElapsed() % 60).padStart(2, "0")} ]`}
1687
- </text>
1688
- </Match>
1689
- <Match when={voiceState() === "processing"}>
1690
- <text fg={theme.primary} selectable={false} onMouseUp={() => voiceToggle()}>
1691
- {"[ 🎙 .... ]"}
1692
- </text>
1693
- </Match>
1694
- <Match when={voiceState() === "finishing"}>
1695
- <text fg={theme.textMuted} selectable={false}>{"[ 🎙 .... ]"}</text>
1696
- </Match>
1697
- </Switch>
1698
- </Show>
1699
- </box>
1700
- </box>
1701
- </box>
1702
- </box>
1703
- <box
1704
- height={1}
1705
- border={["left"]}
1706
- borderColor={borderHighlight()}
1707
- customBorderChars={{
1708
- ...EmptyBorder,
1709
- vertical: theme.backgroundElement.a !== 0 ? "╹" : " ",
1710
- }}
1711
- >
1712
- <box
1713
- height={1}
1714
- border={["bottom"]}
1715
- borderColor={theme.backgroundElement}
1716
- customBorderChars={
1717
- theme.backgroundElement.a !== 0
1718
- ? {
1719
- ...EmptyBorder,
1720
- horizontal: "▀",
1721
- }
1722
- : {
1723
- ...EmptyBorder,
1724
- horizontal: " ",
1725
- }
1726
- }
1727
- />
1728
- </box>
1729
- <box width="100%" flexDirection="row" justifyContent="space-between">
1730
- <Show when={status().type !== "idle"} fallback={props.hint ?? <text />}>
1731
- <box
1732
- flexDirection="row"
1733
- gap={1}
1734
- flexGrow={1}
1735
- justifyContent={status().type === "retry" ? "space-between" : "flex-start"}
1736
- >
1737
- <box flexShrink={0} flexDirection="row" gap={1}>
1738
- <box marginLeft={1}>
1739
- <Show when={kv.get("animations_enabled", true)} fallback={<text fg={theme.textMuted}>[⋯]</text>}>
1740
- <spinner color={spinnerDef().color} frames={spinnerDef().frames} interval={40} />
1741
- </Show>
1742
- </box>
1743
- {(() => {
1744
- const busyMessage = createMemo(() => {
1745
- const s = status()
1746
- return s.type === "busy" ? s.message : undefined
1747
- })
1748
- return (
1749
- <Show when={busyMessage()}>
1750
- <text fg={theme.textMuted}>{busyMessage()}</text>
1751
- </Show>
1752
- )
1753
- })()}
1754
- <box flexDirection="row" gap={1} flexShrink={0}>
1755
- {(() => {
1756
- const retry = createMemo(() => {
1757
- const s = status()
1758
- if (s.type !== "retry") return
1759
- return s
1760
- })
1761
- const message = createMemo(() => {
1762
- const r = retry()
1763
- if (!r) return
1764
- if (r.message.includes("exceeded your current quota") && r.message.includes("gemini"))
1765
- return "gemini is way too hot right now"
1766
- if (r.message.length > 80) return r.message.slice(0, 80) + "..."
1767
- return r.message
1768
- })
1769
- const isTruncated = createMemo(() => {
1770
- const r = retry()
1771
- if (!r) return false
1772
- return r.message.length > 120
1773
- })
1774
- const [seconds, setSeconds] = createSignal(0)
1775
- onMount(() => {
1776
- const timer = setInterval(() => {
1777
- const next = retry()?.next
1778
- if (next) setSeconds(Math.round((next - Date.now()) / 1000))
1779
- }, 1000)
1780
-
1781
- onCleanup(() => {
1782
- clearInterval(timer)
1783
- })
1784
- })
1785
- const handleMessageClick = () => {
1786
- const r = retry()
1787
- if (!r) return
1788
- if (isTruncated()) {
1789
- void DialogAlert.show(dialog, "Retry Error", r.message)
1790
- }
1791
- }
1792
-
1793
- const retryText = () => {
1794
- const r = retry()
1795
- if (!r) return ""
1796
- const baseMessage = message()
1797
- const truncatedHint = isTruncated() ? " (click to expand)" : ""
1798
- const duration = formatDuration(seconds())
1799
- const retryInfo = ` [retrying ${duration ? `in ${duration} ` : ""}attempt #${r.attempt}]`
1800
- return baseMessage + truncatedHint + retryInfo
1801
- }
1802
-
1803
- return (
1804
- <Show when={retry()}>
1805
- <box onMouseUp={handleMessageClick}>
1806
- <text fg={theme.error}>{retryText()}</text>
1807
- </box>
1808
- </Show>
1809
- )
1810
- })()}
1811
- </box>
1812
- </box>
1813
- <text fg={store.interrupt > 0 ? theme.primary : theme.text}>
1814
- esc{" "}
1815
- <span style={{ fg: store.interrupt > 0 ? theme.primary : theme.textMuted }}>
1816
- {store.interrupt > 0 ? "again to interrupt" : "interrupt"}
1817
- </span>
1818
- </text>
1819
- </box>
1820
- </Show>
1821
- <Show when={status().type !== "retry"}>
1822
- <box gap={2} flexGrow={1} flexDirection="row" justifyContent="space-between">
1823
- <Switch>
1824
- <Match when={store.mode === "normal"}>
1825
- <box gap={2} flexDirection="row">
1826
- <Show when={usage()}>
1827
- {(item) => (
1828
- <text fg={theme.textMuted} wrapMode="none">
1829
- {[item().context, item().cost].filter(Boolean).join(" · ")}
1830
- </text>
1831
- )}
1832
- </Show>
1833
- <text fg={theme.text}>
1834
- {keybind.print("agent_cycle")} <span style={{ fg: theme.textMuted }}>{t("tui.prompt.hint.switch_mode")}</span>
1835
- </text>
1836
- <text fg={theme.text}>
1837
- {keybind.print("command_list")}{" "}
1838
- <span style={{ fg: theme.textMuted }}>{t("tui.prompt.hint.settings")}</span>
1839
- </text>
1840
- </box>
1841
- <Show when={status().type === "idle"}>
1842
- <box gap={2} flexDirection="row">
1843
- <text fg={theme.text}>
1844
- @ <span style={{ fg: theme.textMuted }}>{t("tui.prompt.hint.attach_file")}</span>
1845
- </text>
1846
- <text fg={theme.text}>
1847
- $ <span style={{ fg: theme.textMuted }}>{t("tui.prompt.hint.subagent")}</span>
1848
- </text>
1849
- <text fg={theme.text}>
1850
- / <span style={{ fg: theme.textMuted }}>{t("tui.prompt.hint.commands")}</span>
1851
- </text>
1852
- </box>
1853
- </Show>
1854
- </Match>
1855
- <Match when={store.mode === "shell"}>
1856
- <box flexGrow={1} flexDirection="row" justifyContent="flex-end">
1857
- <text fg={theme.text}>
1858
- esc <span style={{ fg: theme.textMuted }}>exit shell mode</span>
1859
- </text>
1860
- </box>
1861
- </Match>
1862
- </Switch>
1863
- </box>
1864
- </Show>
1865
- </box>
1866
- </box>
1867
- </>
1868
- )
1869
- }