command.go 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. package commands
  2. import (
  3. "encoding/json"
  4. "slices"
  5. "strings"
  6. tea "github.com/charmbracelet/bubbletea/v2"
  7. "github.com/sst/opencode-sdk-go"
  8. )
  9. type ExecuteCommandMsg Command
  10. type ExecuteCommandsMsg []Command
  11. type CommandExecutedMsg Command
  12. type Keybinding struct {
  13. RequiresLeader bool
  14. Key string
  15. }
  16. func (k Keybinding) Matches(msg tea.KeyPressMsg, leader bool) bool {
  17. key := k.Key
  18. key = strings.TrimSpace(key)
  19. return key == msg.String() && (k.RequiresLeader == leader)
  20. }
  21. type CommandName string
  22. type Command struct {
  23. Name CommandName
  24. Description string
  25. Keybindings []Keybinding
  26. Trigger []string
  27. }
  28. func (c Command) Keys() []string {
  29. var keys []string
  30. for _, k := range c.Keybindings {
  31. keys = append(keys, k.Key)
  32. }
  33. return keys
  34. }
  35. func (c Command) HasTrigger() bool {
  36. return len(c.Trigger) > 0
  37. }
  38. func (c Command) PrimaryTrigger() string {
  39. if len(c.Trigger) > 0 {
  40. return c.Trigger[0]
  41. }
  42. return ""
  43. }
  44. func (c Command) MatchesTrigger(trigger string) bool {
  45. return slices.Contains(c.Trigger, trigger)
  46. }
  47. type CommandRegistry map[CommandName]Command
  48. func (r CommandRegistry) Sorted() []Command {
  49. var commands []Command
  50. for _, command := range r {
  51. commands = append(commands, command)
  52. }
  53. slices.SortFunc(commands, func(a, b Command) int {
  54. if a.Name == AppExitCommand {
  55. return 1
  56. }
  57. if b.Name == AppExitCommand {
  58. return -1
  59. }
  60. return strings.Compare(string(a.Name), string(b.Name))
  61. })
  62. return commands
  63. }
  64. func (r CommandRegistry) Matches(msg tea.KeyPressMsg, leader bool) []Command {
  65. var matched []Command
  66. for _, command := range r.Sorted() {
  67. if command.Matches(msg, leader) {
  68. matched = append(matched, command)
  69. }
  70. }
  71. return matched
  72. }
  73. const (
  74. AppHelpCommand CommandName = "app_help"
  75. SwitchModeCommand CommandName = "switch_mode"
  76. SwitchModeReverseCommand CommandName = "switch_mode_reverse"
  77. EditorOpenCommand CommandName = "editor_open"
  78. SessionNewCommand CommandName = "session_new"
  79. SessionListCommand CommandName = "session_list"
  80. SessionShareCommand CommandName = "session_share"
  81. SessionUnshareCommand CommandName = "session_unshare"
  82. SessionInterruptCommand CommandName = "session_interrupt"
  83. SessionCompactCommand CommandName = "session_compact"
  84. SessionExportCommand CommandName = "session_export"
  85. ToolDetailsCommand CommandName = "tool_details"
  86. ModelListCommand CommandName = "model_list"
  87. ThemeListCommand CommandName = "theme_list"
  88. FileListCommand CommandName = "file_list"
  89. FileCloseCommand CommandName = "file_close"
  90. FileSearchCommand CommandName = "file_search"
  91. FileDiffToggleCommand CommandName = "file_diff_toggle"
  92. ProjectInitCommand CommandName = "project_init"
  93. InputClearCommand CommandName = "input_clear"
  94. InputPasteCommand CommandName = "input_paste"
  95. InputSubmitCommand CommandName = "input_submit"
  96. InputNewlineCommand CommandName = "input_newline"
  97. MessagesPageUpCommand CommandName = "messages_page_up"
  98. MessagesPageDownCommand CommandName = "messages_page_down"
  99. MessagesHalfPageUpCommand CommandName = "messages_half_page_up"
  100. MessagesHalfPageDownCommand CommandName = "messages_half_page_down"
  101. MessagesPreviousCommand CommandName = "messages_previous"
  102. MessagesNextCommand CommandName = "messages_next"
  103. MessagesFirstCommand CommandName = "messages_first"
  104. MessagesLastCommand CommandName = "messages_last"
  105. MessagesLayoutToggleCommand CommandName = "messages_layout_toggle"
  106. MessagesCopyCommand CommandName = "messages_copy"
  107. MessagesRevertCommand CommandName = "messages_revert"
  108. AppExitCommand CommandName = "app_exit"
  109. )
  110. func (k Command) Matches(msg tea.KeyPressMsg, leader bool) bool {
  111. for _, binding := range k.Keybindings {
  112. if binding.Matches(msg, leader) {
  113. return true
  114. }
  115. }
  116. return false
  117. }
  118. func parseBindings(bindings ...string) []Keybinding {
  119. var parsedBindings []Keybinding
  120. for _, binding := range bindings {
  121. for p := range strings.SplitSeq(binding, ",") {
  122. requireLeader := strings.HasPrefix(p, "<leader>")
  123. keybinding := strings.ReplaceAll(p, "<leader>", "")
  124. keybinding = strings.TrimSpace(keybinding)
  125. parsedBindings = append(parsedBindings, Keybinding{
  126. RequiresLeader: requireLeader,
  127. Key: keybinding,
  128. })
  129. }
  130. }
  131. return parsedBindings
  132. }
  133. func LoadFromConfig(config *opencode.Config) CommandRegistry {
  134. defaults := []Command{
  135. {
  136. Name: AppHelpCommand,
  137. Description: "show help",
  138. Keybindings: parseBindings("<leader>h"),
  139. Trigger: []string{"help"},
  140. },
  141. {
  142. Name: SwitchModeCommand,
  143. Description: "next mode",
  144. Keybindings: parseBindings("tab"),
  145. },
  146. {
  147. Name: SwitchModeReverseCommand,
  148. Description: "previous mode",
  149. Keybindings: parseBindings("shift+tab"),
  150. },
  151. {
  152. Name: EditorOpenCommand,
  153. Description: "open editor",
  154. Keybindings: parseBindings("<leader>e"),
  155. Trigger: []string{"editor"},
  156. },
  157. {
  158. Name: SessionExportCommand,
  159. Description: "export conversation",
  160. Keybindings: parseBindings("<leader>x"),
  161. Trigger: []string{"export"},
  162. },
  163. {
  164. Name: SessionNewCommand,
  165. Description: "new session",
  166. Keybindings: parseBindings("<leader>n"),
  167. Trigger: []string{"new", "clear"},
  168. },
  169. {
  170. Name: SessionListCommand,
  171. Description: "list sessions",
  172. Keybindings: parseBindings("<leader>l"),
  173. Trigger: []string{"sessions", "resume", "continue"},
  174. },
  175. {
  176. Name: SessionShareCommand,
  177. Description: "share session",
  178. Keybindings: parseBindings("<leader>s"),
  179. Trigger: []string{"share"},
  180. },
  181. {
  182. Name: SessionUnshareCommand,
  183. Description: "unshare session",
  184. Keybindings: parseBindings("<leader>u"),
  185. Trigger: []string{"unshare"},
  186. },
  187. {
  188. Name: SessionInterruptCommand,
  189. Description: "interrupt session",
  190. Keybindings: parseBindings("esc"),
  191. },
  192. {
  193. Name: SessionCompactCommand,
  194. Description: "compact the session",
  195. Keybindings: parseBindings("<leader>c"),
  196. Trigger: []string{"compact", "summarize"},
  197. },
  198. {
  199. Name: ToolDetailsCommand,
  200. Description: "toggle tool details",
  201. Keybindings: parseBindings("<leader>d"),
  202. Trigger: []string{"details"},
  203. },
  204. {
  205. Name: ModelListCommand,
  206. Description: "list models",
  207. Keybindings: parseBindings("<leader>m"),
  208. Trigger: []string{"models"},
  209. },
  210. {
  211. Name: ThemeListCommand,
  212. Description: "list themes",
  213. Keybindings: parseBindings("<leader>t"),
  214. Trigger: []string{"themes"},
  215. },
  216. // {
  217. // Name: FileListCommand,
  218. // Description: "list files",
  219. // Keybindings: parseBindings("<leader>f"),
  220. // Trigger: []string{"files"},
  221. // },
  222. {
  223. Name: FileCloseCommand,
  224. Description: "close file",
  225. Keybindings: parseBindings("esc"),
  226. },
  227. {
  228. Name: FileSearchCommand,
  229. Description: "search file",
  230. Keybindings: parseBindings("<leader>/"),
  231. },
  232. {
  233. Name: FileDiffToggleCommand,
  234. Description: "split/unified diff",
  235. Keybindings: parseBindings("<leader>v"),
  236. },
  237. {
  238. Name: ProjectInitCommand,
  239. Description: "create/update AGENTS.md",
  240. Keybindings: parseBindings("<leader>i"),
  241. Trigger: []string{"init"},
  242. },
  243. {
  244. Name: InputClearCommand,
  245. Description: "clear input",
  246. Keybindings: parseBindings("ctrl+c"),
  247. },
  248. {
  249. Name: InputPasteCommand,
  250. Description: "paste content",
  251. Keybindings: parseBindings("ctrl+v", "super+v"),
  252. },
  253. {
  254. Name: InputSubmitCommand,
  255. Description: "submit message",
  256. Keybindings: parseBindings("enter"),
  257. },
  258. {
  259. Name: InputNewlineCommand,
  260. Description: "insert newline",
  261. Keybindings: parseBindings("shift+enter", "ctrl+j"),
  262. },
  263. {
  264. Name: MessagesPageUpCommand,
  265. Description: "page up",
  266. Keybindings: parseBindings("pgup"),
  267. },
  268. {
  269. Name: MessagesPageDownCommand,
  270. Description: "page down",
  271. Keybindings: parseBindings("pgdown"),
  272. },
  273. {
  274. Name: MessagesHalfPageUpCommand,
  275. Description: "half page up",
  276. Keybindings: parseBindings("ctrl+alt+u"),
  277. },
  278. {
  279. Name: MessagesHalfPageDownCommand,
  280. Description: "half page down",
  281. Keybindings: parseBindings("ctrl+alt+d"),
  282. },
  283. {
  284. Name: MessagesPreviousCommand,
  285. Description: "previous message",
  286. Keybindings: parseBindings("ctrl+up"),
  287. },
  288. {
  289. Name: MessagesNextCommand,
  290. Description: "next message",
  291. Keybindings: parseBindings("ctrl+down"),
  292. },
  293. {
  294. Name: MessagesFirstCommand,
  295. Description: "first message",
  296. Keybindings: parseBindings("ctrl+g"),
  297. },
  298. {
  299. Name: MessagesLastCommand,
  300. Description: "last message",
  301. Keybindings: parseBindings("ctrl+alt+g"),
  302. },
  303. {
  304. Name: MessagesLayoutToggleCommand,
  305. Description: "toggle layout",
  306. Keybindings: parseBindings("<leader>p"),
  307. },
  308. {
  309. Name: MessagesCopyCommand,
  310. Description: "copy message",
  311. Keybindings: parseBindings("<leader>y"),
  312. },
  313. {
  314. Name: MessagesRevertCommand,
  315. Description: "revert message",
  316. Keybindings: parseBindings("<leader>r"),
  317. },
  318. {
  319. Name: AppExitCommand,
  320. Description: "exit the app",
  321. Keybindings: parseBindings("ctrl+c", "<leader>q"),
  322. Trigger: []string{"exit", "quit"},
  323. },
  324. }
  325. registry := make(CommandRegistry)
  326. keybinds := map[string]string{}
  327. marshalled, _ := json.Marshal(config.Keybinds)
  328. json.Unmarshal(marshalled, &keybinds)
  329. for _, command := range defaults {
  330. if keybind, ok := keybinds[string(command.Name)]; ok && keybind != "" {
  331. command.Keybindings = parseBindings(keybind)
  332. }
  333. registry[command.Name] = command
  334. }
  335. return registry
  336. }