chad-code 1.3.1

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 (338) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/README.npm.md +64 -0
  5. package/bin/chad-code +84 -0
  6. package/bunfig.toml +7 -0
  7. package/eslint.config.js +29 -0
  8. package/package.json +107 -0
  9. package/parsers-config.ts +253 -0
  10. package/script/build.ts +167 -0
  11. package/script/postinstall.mjs +122 -0
  12. package/script/publish-registries.ts +187 -0
  13. package/script/publish.ts +93 -0
  14. package/script/schema.ts +47 -0
  15. package/src/acp/README.md +164 -0
  16. package/src/acp/agent.ts +1086 -0
  17. package/src/acp/session.ts +101 -0
  18. package/src/acp/types.ts +22 -0
  19. package/src/agent/agent.ts +253 -0
  20. package/src/agent/generate.txt +75 -0
  21. package/src/agent/prompt/compaction.txt +12 -0
  22. package/src/agent/prompt/explore.txt +18 -0
  23. package/src/agent/prompt/summary.txt +11 -0
  24. package/src/agent/prompt/title.txt +36 -0
  25. package/src/auth/index.ts +70 -0
  26. package/src/bun/index.ts +130 -0
  27. package/src/bus/bus-event.ts +43 -0
  28. package/src/bus/global.ts +10 -0
  29. package/src/bus/index.ts +105 -0
  30. package/src/cli/bootstrap.ts +17 -0
  31. package/src/cli/cmd/acp.ts +69 -0
  32. package/src/cli/cmd/agent.ts +257 -0
  33. package/src/cli/cmd/auth.ts +132 -0
  34. package/src/cli/cmd/cmd.ts +7 -0
  35. package/src/cli/cmd/debug/agent.ts +28 -0
  36. package/src/cli/cmd/debug/config.ts +15 -0
  37. package/src/cli/cmd/debug/file.ts +91 -0
  38. package/src/cli/cmd/debug/index.ts +45 -0
  39. package/src/cli/cmd/debug/lsp.ts +48 -0
  40. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  41. package/src/cli/cmd/debug/scrap.ts +15 -0
  42. package/src/cli/cmd/debug/skill.ts +15 -0
  43. package/src/cli/cmd/debug/snapshot.ts +48 -0
  44. package/src/cli/cmd/export.ts +88 -0
  45. package/src/cli/cmd/generate.ts +38 -0
  46. package/src/cli/cmd/github.ts +32 -0
  47. package/src/cli/cmd/import.ts +98 -0
  48. package/src/cli/cmd/mcp.ts +670 -0
  49. package/src/cli/cmd/models.ts +42 -0
  50. package/src/cli/cmd/pr.ts +112 -0
  51. package/src/cli/cmd/run.ts +374 -0
  52. package/src/cli/cmd/serve.ts +16 -0
  53. package/src/cli/cmd/session.ts +135 -0
  54. package/src/cli/cmd/stats.ts +402 -0
  55. package/src/cli/cmd/tui/app.tsx +705 -0
  56. package/src/cli/cmd/tui/attach.ts +32 -0
  57. package/src/cli/cmd/tui/component/border.tsx +21 -0
  58. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  59. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  60. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  61. package/src/cli/cmd/tui/component/dialog-model.tsx +232 -0
  62. package/src/cli/cmd/tui/component/dialog-provider.tsx +228 -0
  63. package/src/cli/cmd/tui/component/dialog-session-list.tsx +115 -0
  64. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  65. package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
  66. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  67. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  68. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  69. package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
  70. package/src/cli/cmd/tui/component/logo.tsx +43 -0
  71. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +654 -0
  72. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  73. package/src/cli/cmd/tui/component/prompt/index.tsx +1078 -0
  74. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  75. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  76. package/src/cli/cmd/tui/component/tips.ts +92 -0
  77. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  78. package/src/cli/cmd/tui/context/args.tsx +14 -0
  79. package/src/cli/cmd/tui/context/directory.ts +13 -0
  80. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  81. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  82. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  83. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  84. package/src/cli/cmd/tui/context/local.tsx +392 -0
  85. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  86. package/src/cli/cmd/tui/context/route.tsx +46 -0
  87. package/src/cli/cmd/tui/context/sdk.tsx +75 -0
  88. package/src/cli/cmd/tui/context/sync.tsx +384 -0
  89. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  90. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  91. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  92. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  94. package/src/cli/cmd/tui/context/theme/chad.json +245 -0
  95. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  96. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  97. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  98. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  99. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  100. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  101. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  102. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  103. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  104. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  105. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  107. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  108. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  110. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  111. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  112. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  113. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  114. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  115. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  116. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  117. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  118. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  119. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  120. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  121. package/src/cli/cmd/tui/context/theme.tsx +1137 -0
  122. package/src/cli/cmd/tui/event.ts +46 -0
  123. package/src/cli/cmd/tui/routes/home.tsx +138 -0
  124. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  128. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  129. package/src/cli/cmd/tui/routes/session/header.tsx +125 -0
  130. package/src/cli/cmd/tui/routes/session/index.tsx +1814 -0
  131. package/src/cli/cmd/tui/routes/session/permission.tsx +416 -0
  132. package/src/cli/cmd/tui/routes/session/sidebar.tsx +318 -0
  133. package/src/cli/cmd/tui/spawn.ts +48 -0
  134. package/src/cli/cmd/tui/thread.ts +111 -0
  135. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  136. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  137. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  138. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  139. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  140. package/src/cli/cmd/tui/ui/dialog-select.tsx +345 -0
  141. package/src/cli/cmd/tui/ui/dialog.tsx +171 -0
  142. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  143. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  144. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  145. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  146. package/src/cli/cmd/tui/util/editor.ts +32 -0
  147. package/src/cli/cmd/tui/util/signal.ts +7 -0
  148. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  149. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  150. package/src/cli/cmd/tui/worker.ts +68 -0
  151. package/src/cli/cmd/uninstall.ts +344 -0
  152. package/src/cli/cmd/upgrade.ts +67 -0
  153. package/src/cli/cmd/web.ts +73 -0
  154. package/src/cli/error.ts +56 -0
  155. package/src/cli/network.ts +53 -0
  156. package/src/cli/ui.ts +87 -0
  157. package/src/cli/upgrade.ts +25 -0
  158. package/src/command/index.ts +131 -0
  159. package/src/command/template/initialize.txt +10 -0
  160. package/src/command/template/review.txt +97 -0
  161. package/src/config/config.ts +1124 -0
  162. package/src/config/markdown.ts +41 -0
  163. package/src/env/index.ts +26 -0
  164. package/src/file/ignore.ts +83 -0
  165. package/src/file/index.ts +411 -0
  166. package/src/file/ripgrep.ts +402 -0
  167. package/src/file/time.ts +64 -0
  168. package/src/file/watcher.ts +117 -0
  169. package/src/flag/flag.ts +52 -0
  170. package/src/format/formatter.ts +359 -0
  171. package/src/format/index.ts +137 -0
  172. package/src/global/index.ts +55 -0
  173. package/src/id/id.ts +73 -0
  174. package/src/ide/index.ts +77 -0
  175. package/src/index.ts +159 -0
  176. package/src/installation/index.ts +198 -0
  177. package/src/lsp/client.ts +252 -0
  178. package/src/lsp/index.ts +485 -0
  179. package/src/lsp/language.ts +119 -0
  180. package/src/lsp/server.ts +2023 -0
  181. package/src/mcp/auth.ts +135 -0
  182. package/src/mcp/index.ts +874 -0
  183. package/src/mcp/oauth-callback.ts +200 -0
  184. package/src/mcp/oauth-provider.ts +154 -0
  185. package/src/patch/index.ts +622 -0
  186. package/src/permission/arity.ts +163 -0
  187. package/src/permission/index.ts +210 -0
  188. package/src/permission/next.ts +268 -0
  189. package/src/plugin/index.ts +106 -0
  190. package/src/project/bootstrap.ts +31 -0
  191. package/src/project/instance.ts +78 -0
  192. package/src/project/project.ts +263 -0
  193. package/src/project/state.ts +65 -0
  194. package/src/project/vcs.ts +76 -0
  195. package/src/provider/auth.ts +143 -0
  196. package/src/provider/models-macro.ts +4 -0
  197. package/src/provider/models.ts +77 -0
  198. package/src/provider/provider.ts +516 -0
  199. package/src/provider/transform.ts +114 -0
  200. package/src/pty/index.ts +212 -0
  201. package/src/server/error.ts +36 -0
  202. package/src/server/mdns.ts +57 -0
  203. package/src/server/project.ts +79 -0
  204. package/src/server/server.ts +2866 -0
  205. package/src/server/tui.ts +71 -0
  206. package/src/session/compaction.ts +225 -0
  207. package/src/session/index.ts +469 -0
  208. package/src/session/llm.ts +213 -0
  209. package/src/session/message-v2.ts +742 -0
  210. package/src/session/message.ts +189 -0
  211. package/src/session/processor.ts +402 -0
  212. package/src/session/prompt/anthropic-20250930.txt +166 -0
  213. package/src/session/prompt/anthropic.txt +105 -0
  214. package/src/session/prompt/anthropic_spoof.txt +1 -0
  215. package/src/session/prompt/beast.txt +147 -0
  216. package/src/session/prompt/build-switch.txt +5 -0
  217. package/src/session/prompt/codex.txt +318 -0
  218. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  219. package/src/session/prompt/gemini.txt +155 -0
  220. package/src/session/prompt/max-steps.txt +16 -0
  221. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  222. package/src/session/prompt/plan.txt +26 -0
  223. package/src/session/prompt/qwen.txt +109 -0
  224. package/src/session/prompt.ts +1621 -0
  225. package/src/session/retry.ts +90 -0
  226. package/src/session/revert.ts +108 -0
  227. package/src/session/status.ts +76 -0
  228. package/src/session/summary.ts +194 -0
  229. package/src/session/system.ts +108 -0
  230. package/src/session/todo.ts +37 -0
  231. package/src/share/share-next.ts +194 -0
  232. package/src/share/share.ts +23 -0
  233. package/src/shell/shell.ts +67 -0
  234. package/src/skill/index.ts +1 -0
  235. package/src/skill/skill.ts +124 -0
  236. package/src/snapshot/index.ts +197 -0
  237. package/src/storage/storage.ts +226 -0
  238. package/src/tool/bash.ts +262 -0
  239. package/src/tool/bash.txt +116 -0
  240. package/src/tool/batch.ts +175 -0
  241. package/src/tool/batch.txt +24 -0
  242. package/src/tool/codesearch.ts +132 -0
  243. package/src/tool/codesearch.txt +12 -0
  244. package/src/tool/edit.ts +655 -0
  245. package/src/tool/edit.txt +10 -0
  246. package/src/tool/glob.ts +75 -0
  247. package/src/tool/glob.txt +6 -0
  248. package/src/tool/grep.ts +132 -0
  249. package/src/tool/grep.txt +8 -0
  250. package/src/tool/invalid.ts +17 -0
  251. package/src/tool/ls.ts +119 -0
  252. package/src/tool/ls.txt +1 -0
  253. package/src/tool/lsp.ts +94 -0
  254. package/src/tool/lsp.txt +19 -0
  255. package/src/tool/multiedit.ts +46 -0
  256. package/src/tool/multiedit.txt +41 -0
  257. package/src/tool/patch.ts +210 -0
  258. package/src/tool/patch.txt +1 -0
  259. package/src/tool/read.ts +191 -0
  260. package/src/tool/read.txt +12 -0
  261. package/src/tool/registry.ts +137 -0
  262. package/src/tool/skill.ts +77 -0
  263. package/src/tool/task.ts +167 -0
  264. package/src/tool/task.txt +60 -0
  265. package/src/tool/todo.ts +53 -0
  266. package/src/tool/todoread.txt +14 -0
  267. package/src/tool/todowrite.txt +167 -0
  268. package/src/tool/tool.ts +73 -0
  269. package/src/tool/webfetch.ts +182 -0
  270. package/src/tool/webfetch.txt +13 -0
  271. package/src/tool/websearch.ts +144 -0
  272. package/src/tool/websearch.txt +11 -0
  273. package/src/tool/write.ts +84 -0
  274. package/src/tool/write.txt +8 -0
  275. package/src/util/archive.ts +16 -0
  276. package/src/util/color.ts +19 -0
  277. package/src/util/context.ts +25 -0
  278. package/src/util/defer.ts +12 -0
  279. package/src/util/eventloop.ts +20 -0
  280. package/src/util/filesystem.ts +83 -0
  281. package/src/util/fn.ts +11 -0
  282. package/src/util/iife.ts +3 -0
  283. package/src/util/keybind.ts +102 -0
  284. package/src/util/lazy.ts +18 -0
  285. package/src/util/locale.ts +81 -0
  286. package/src/util/lock.ts +98 -0
  287. package/src/util/log.ts +180 -0
  288. package/src/util/queue.ts +32 -0
  289. package/src/util/rpc.ts +42 -0
  290. package/src/util/scrap.ts +10 -0
  291. package/src/util/signal.ts +12 -0
  292. package/src/util/timeout.ts +14 -0
  293. package/src/util/token.ts +7 -0
  294. package/src/util/wildcard.ts +54 -0
  295. package/src/worktree/index.ts +217 -0
  296. package/sst-env.d.ts +9 -0
  297. package/test/agent/agent.test.ts +448 -0
  298. package/test/bun.test.ts +53 -0
  299. package/test/cli/github-action.test.ts +129 -0
  300. package/test/cli/github-remote.test.ts +80 -0
  301. package/test/cli/tui/transcript.test.ts +297 -0
  302. package/test/config/agent-color.test.ts +66 -0
  303. package/test/config/config.test.ts +870 -0
  304. package/test/config/markdown.test.ts +89 -0
  305. package/test/file/ignore.test.ts +10 -0
  306. package/test/file/path-traversal.test.ts +115 -0
  307. package/test/fixture/fixture.ts +45 -0
  308. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  309. package/test/ide/ide.test.ts +82 -0
  310. package/test/keybind.test.ts +421 -0
  311. package/test/lsp/client.test.ts +95 -0
  312. package/test/mcp/headers.test.ts +153 -0
  313. package/test/patch/patch.test.ts +348 -0
  314. package/test/permission/arity.test.ts +33 -0
  315. package/test/permission/next.test.ts +652 -0
  316. package/test/preload.ts +63 -0
  317. package/test/project/project.test.ts +120 -0
  318. package/test/provider/amazon-bedrock.test.ts +236 -0
  319. package/test/provider/provider.test.ts +2127 -0
  320. package/test/provider/transform.test.ts +980 -0
  321. package/test/server/session-select.test.ts +78 -0
  322. package/test/session/compaction.test.ts +251 -0
  323. package/test/session/message-v2.test.ts +570 -0
  324. package/test/session/retry.test.ts +131 -0
  325. package/test/session/revert-compact.test.ts +285 -0
  326. package/test/session/session.test.ts +71 -0
  327. package/test/skill/skill.test.ts +185 -0
  328. package/test/snapshot/snapshot.test.ts +939 -0
  329. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  330. package/test/tool/bash.test.ts +232 -0
  331. package/test/tool/grep.test.ts +109 -0
  332. package/test/tool/patch.test.ts +261 -0
  333. package/test/tool/read.test.ts +167 -0
  334. package/test/util/iife.test.ts +36 -0
  335. package/test/util/lazy.test.ts +50 -0
  336. package/test/util/timeout.test.ts +21 -0
  337. package/test/util/wildcard.test.ts +55 -0
  338. package/tsconfig.json +16 -0
@@ -0,0 +1,1137 @@
1
+ import { SyntaxStyle, RGBA, type TerminalColors } from "@opentui/core"
2
+ import path from "path"
3
+ import { createEffect, createMemo, onMount } from "solid-js"
4
+ import { useSync } from "@tui/context/sync"
5
+ import { createSimpleContext } from "./helper"
6
+ import aura from "./theme/aura.json" with { type: "json" }
7
+ import ayu from "./theme/ayu.json" with { type: "json" }
8
+ import catppuccin from "./theme/catppuccin.json" with { type: "json" }
9
+ import catppuccinFrappe from "./theme/catppuccin-frappe.json" with { type: "json" }
10
+ import catppuccinMacchiato from "./theme/catppuccin-macchiato.json" with { type: "json" }
11
+ import cobalt2 from "./theme/cobalt2.json" with { type: "json" }
12
+ import cursor from "./theme/cursor.json" with { type: "json" }
13
+ import dracula from "./theme/dracula.json" with { type: "json" }
14
+ import everforest from "./theme/everforest.json" with { type: "json" }
15
+ import flexoki from "./theme/flexoki.json" with { type: "json" }
16
+ import github from "./theme/github.json" with { type: "json" }
17
+ import gruvbox from "./theme/gruvbox.json" with { type: "json" }
18
+ import kanagawa from "./theme/kanagawa.json" with { type: "json" }
19
+ import material from "./theme/material.json" with { type: "json" }
20
+ import matrix from "./theme/matrix.json" with { type: "json" }
21
+ import mercury from "./theme/mercury.json" with { type: "json" }
22
+ import monokai from "./theme/monokai.json" with { type: "json" }
23
+ import nightowl from "./theme/nightowl.json" with { type: "json" }
24
+ import nord from "./theme/nord.json" with { type: "json" }
25
+ import osakaJade from "./theme/osaka-jade.json" with { type: "json" }
26
+ import onedark from "./theme/one-dark.json" with { type: "json" }
27
+ import chad from "./theme/chad.json" with { type: "json" }
28
+ import orng from "./theme/orng.json" with { type: "json" }
29
+ import lucentOrng from "./theme/lucent-orng.json" with { type: "json" }
30
+ import palenight from "./theme/palenight.json" with { type: "json" }
31
+ import rosepine from "./theme/rosepine.json" with { type: "json" }
32
+ import solarized from "./theme/solarized.json" with { type: "json" }
33
+ import synthwave84 from "./theme/synthwave84.json" with { type: "json" }
34
+ import tokyonight from "./theme/tokyonight.json" with { type: "json" }
35
+ import vercel from "./theme/vercel.json" with { type: "json" }
36
+ import vesper from "./theme/vesper.json" with { type: "json" }
37
+ import zenburn from "./theme/zenburn.json" with { type: "json" }
38
+ import { useKV } from "./kv"
39
+ import { useRenderer } from "@opentui/solid"
40
+ import { createStore, produce } from "solid-js/store"
41
+ import { Global } from "@/global"
42
+ import { Filesystem } from "@/util/filesystem"
43
+
44
+ type ThemeColors = {
45
+ primary: RGBA
46
+ secondary: RGBA
47
+ accent: RGBA
48
+ error: RGBA
49
+ warning: RGBA
50
+ success: RGBA
51
+ info: RGBA
52
+ text: RGBA
53
+ textMuted: RGBA
54
+ selectedListItemText: RGBA
55
+ background: RGBA
56
+ backgroundPanel: RGBA
57
+ backgroundElement: RGBA
58
+ backgroundMenu: RGBA
59
+ border: RGBA
60
+ borderActive: RGBA
61
+ borderSubtle: RGBA
62
+ diffAdded: RGBA
63
+ diffRemoved: RGBA
64
+ diffContext: RGBA
65
+ diffHunkHeader: RGBA
66
+ diffHighlightAdded: RGBA
67
+ diffHighlightRemoved: RGBA
68
+ diffAddedBg: RGBA
69
+ diffRemovedBg: RGBA
70
+ diffContextBg: RGBA
71
+ diffLineNumber: RGBA
72
+ diffAddedLineNumberBg: RGBA
73
+ diffRemovedLineNumberBg: RGBA
74
+ markdownText: RGBA
75
+ markdownHeading: RGBA
76
+ markdownLink: RGBA
77
+ markdownLinkText: RGBA
78
+ markdownCode: RGBA
79
+ markdownBlockQuote: RGBA
80
+ markdownEmph: RGBA
81
+ markdownStrong: RGBA
82
+ markdownHorizontalRule: RGBA
83
+ markdownListItem: RGBA
84
+ markdownListEnumeration: RGBA
85
+ markdownImage: RGBA
86
+ markdownImageText: RGBA
87
+ markdownCodeBlock: RGBA
88
+ syntaxComment: RGBA
89
+ syntaxKeyword: RGBA
90
+ syntaxFunction: RGBA
91
+ syntaxVariable: RGBA
92
+ syntaxString: RGBA
93
+ syntaxNumber: RGBA
94
+ syntaxType: RGBA
95
+ syntaxOperator: RGBA
96
+ syntaxPunctuation: RGBA
97
+ }
98
+
99
+ type Theme = ThemeColors & {
100
+ _hasSelectedListItemText: boolean
101
+ thinkingOpacity: number
102
+ }
103
+
104
+ export function selectedForeground(theme: Theme): RGBA {
105
+ // If theme explicitly defines selectedListItemText, use it
106
+ if (theme._hasSelectedListItemText) {
107
+ return theme.selectedListItemText
108
+ }
109
+
110
+ // For transparent backgrounds, calculate contrast based on primary color
111
+ if (theme.background.a === 0) {
112
+ const { r, g, b } = theme.primary
113
+ const luminance = 0.299 * r + 0.587 * g + 0.114 * b
114
+ return luminance > 0.5 ? RGBA.fromInts(0, 0, 0) : RGBA.fromInts(255, 255, 255)
115
+ }
116
+
117
+ // Fall back to background color
118
+ return theme.background
119
+ }
120
+
121
+ type HexColor = `#${string}`
122
+ type RefName = string
123
+ type Variant = {
124
+ dark: HexColor | RefName
125
+ light: HexColor | RefName
126
+ }
127
+ type ColorValue = HexColor | RefName | Variant | RGBA
128
+ type ThemeJson = {
129
+ $schema?: string
130
+ defs?: Record<string, HexColor | RefName>
131
+ theme: Omit<Record<keyof ThemeColors, ColorValue>, "selectedListItemText" | "backgroundMenu"> & {
132
+ selectedListItemText?: ColorValue
133
+ backgroundMenu?: ColorValue
134
+ thinkingOpacity?: number
135
+ }
136
+ }
137
+
138
+ export const DEFAULT_THEMES: Record<string, ThemeJson> = {
139
+ aura,
140
+ ayu,
141
+ catppuccin,
142
+ ["catppuccin-frappe"]: catppuccinFrappe,
143
+ ["catppuccin-macchiato"]: catppuccinMacchiato,
144
+ cobalt2,
145
+ cursor,
146
+ dracula,
147
+ everforest,
148
+ flexoki,
149
+ github,
150
+ gruvbox,
151
+ kanagawa,
152
+ material,
153
+ matrix,
154
+ mercury,
155
+ monokai,
156
+ nightowl,
157
+ nord,
158
+ ["one-dark"]: onedark,
159
+ ["osaka-jade"]: osakaJade,
160
+ chad,
161
+ orng,
162
+ ["lucent-orng"]: lucentOrng,
163
+ palenight,
164
+ rosepine,
165
+ solarized,
166
+ synthwave84,
167
+ tokyonight,
168
+ vesper,
169
+ vercel,
170
+ zenburn,
171
+ }
172
+
173
+ function resolveTheme(theme: ThemeJson, mode: "dark" | "light") {
174
+ const defs = theme.defs ?? {}
175
+ function resolveColor(c: ColorValue): RGBA {
176
+ if (c instanceof RGBA) return c
177
+ if (typeof c === "string") {
178
+ if (c === "transparent" || c === "none") return RGBA.fromInts(0, 0, 0, 0)
179
+
180
+ if (c.startsWith("#")) return RGBA.fromHex(c)
181
+
182
+ if (defs[c] != null) {
183
+ return resolveColor(defs[c])
184
+ } else if (theme.theme[c as keyof ThemeColors] !== undefined) {
185
+ return resolveColor(theme.theme[c as keyof ThemeColors]!)
186
+ } else {
187
+ throw new Error(`Color reference "${c}" not found in defs or theme`)
188
+ }
189
+ }
190
+ if (typeof c === "number") {
191
+ return ansiToRgba(c)
192
+ }
193
+ return resolveColor(c[mode])
194
+ }
195
+
196
+ const resolved = Object.fromEntries(
197
+ Object.entries(theme.theme)
198
+ .filter(([key]) => key !== "selectedListItemText" && key !== "backgroundMenu" && key !== "thinkingOpacity")
199
+ .map(([key, value]) => {
200
+ return [key, resolveColor(value as ColorValue)]
201
+ }),
202
+ ) as Partial<ThemeColors>
203
+
204
+ // Handle selectedListItemText separately since it's optional
205
+ const hasSelectedListItemText = theme.theme.selectedListItemText !== undefined
206
+ if (hasSelectedListItemText) {
207
+ resolved.selectedListItemText = resolveColor(theme.theme.selectedListItemText!)
208
+ } else {
209
+ // Backward compatibility: if selectedListItemText is not defined, use background color
210
+ // This preserves the current behavior for all existing themes
211
+ resolved.selectedListItemText = resolved.background
212
+ }
213
+
214
+ // Handle backgroundMenu - optional with fallback to backgroundElement
215
+ if (theme.theme.backgroundMenu !== undefined) {
216
+ resolved.backgroundMenu = resolveColor(theme.theme.backgroundMenu)
217
+ } else {
218
+ resolved.backgroundMenu = resolved.backgroundElement
219
+ }
220
+
221
+ // Handle thinkingOpacity - optional with default of 0.6
222
+ const thinkingOpacity = theme.theme.thinkingOpacity ?? 0.6
223
+
224
+ return {
225
+ ...resolved,
226
+ _hasSelectedListItemText: hasSelectedListItemText,
227
+ thinkingOpacity,
228
+ } as Theme
229
+ }
230
+
231
+ function ansiToRgba(code: number): RGBA {
232
+ // Standard ANSI colors (0-15)
233
+ if (code < 16) {
234
+ const ansiColors = [
235
+ "#000000", // Black
236
+ "#800000", // Red
237
+ "#008000", // Green
238
+ "#808000", // Yellow
239
+ "#000080", // Blue
240
+ "#800080", // Magenta
241
+ "#008080", // Cyan
242
+ "#c0c0c0", // White
243
+ "#808080", // Bright Black
244
+ "#ff0000", // Bright Red
245
+ "#00ff00", // Bright Green
246
+ "#ffff00", // Bright Yellow
247
+ "#0000ff", // Bright Blue
248
+ "#ff00ff", // Bright Magenta
249
+ "#00ffff", // Bright Cyan
250
+ "#ffffff", // Bright White
251
+ ]
252
+ return RGBA.fromHex(ansiColors[code] ?? "#000000")
253
+ }
254
+
255
+ // 6x6x6 Color Cube (16-231)
256
+ if (code < 232) {
257
+ const index = code - 16
258
+ const b = index % 6
259
+ const g = Math.floor(index / 6) % 6
260
+ const r = Math.floor(index / 36)
261
+
262
+ const val = (x: number) => (x === 0 ? 0 : x * 40 + 55)
263
+ return RGBA.fromInts(val(r), val(g), val(b))
264
+ }
265
+
266
+ // Grayscale Ramp (232-255)
267
+ if (code < 256) {
268
+ const gray = (code - 232) * 10 + 8
269
+ return RGBA.fromInts(gray, gray, gray)
270
+ }
271
+
272
+ // Fallback for invalid codes
273
+ return RGBA.fromInts(0, 0, 0)
274
+ }
275
+
276
+ export const { use: useTheme, provider: ThemeProvider } = createSimpleContext({
277
+ name: "Theme",
278
+ init: (props: { mode: "dark" | "light" }) => {
279
+ const sync = useSync()
280
+ const kv = useKV()
281
+ const [store, setStore] = createStore({
282
+ themes: DEFAULT_THEMES,
283
+ mode: kv.get("theme_mode", props.mode),
284
+ active: (sync.data.config.theme ?? kv.get("theme", "chad")) as string,
285
+ ready: false,
286
+ })
287
+
288
+ createEffect(() => {
289
+ const theme = sync.data.config.theme
290
+ console.log("theme", theme)
291
+ if (theme) setStore("active", theme)
292
+ })
293
+
294
+ createEffect(() => {
295
+ getCustomThemes()
296
+ .then((custom) => {
297
+ setStore(
298
+ produce((draft) => {
299
+ Object.assign(draft.themes, custom)
300
+ }),
301
+ )
302
+ })
303
+ .catch(() => {
304
+ setStore("active", "chad")
305
+ })
306
+ .finally(() => {
307
+ if (store.active !== "system") {
308
+ setStore("ready", true)
309
+ }
310
+ })
311
+ })
312
+
313
+ const renderer = useRenderer()
314
+ renderer
315
+ .getPalette({
316
+ size: 16,
317
+ })
318
+ .then((colors) => {
319
+ if (!colors.palette[0]) {
320
+ if (store.active === "system") {
321
+ setStore(
322
+ produce((draft) => {
323
+ draft.active = "chad"
324
+ draft.ready = true
325
+ }),
326
+ )
327
+ }
328
+ return
329
+ }
330
+ setStore(
331
+ produce((draft) => {
332
+ draft.themes.system = generateSystem(colors, store.mode)
333
+ if (store.active === "system") {
334
+ draft.ready = true
335
+ }
336
+ }),
337
+ )
338
+ })
339
+
340
+ const values = createMemo(() => {
341
+ return resolveTheme(store.themes[store.active] ?? store.themes.chad, store.mode)
342
+ })
343
+
344
+ const syntax = createMemo(() => generateSyntax(values()))
345
+ const subtleSyntax = createMemo(() => generateSubtleSyntax(values()))
346
+
347
+ return {
348
+ theme: new Proxy(values(), {
349
+ get(_target, prop) {
350
+ // @ts-expect-error
351
+ return values()[prop]
352
+ },
353
+ }),
354
+ get selected() {
355
+ return store.active
356
+ },
357
+ all() {
358
+ return store.themes
359
+ },
360
+ syntax,
361
+ subtleSyntax,
362
+ mode() {
363
+ return store.mode
364
+ },
365
+ setMode(mode: "dark" | "light") {
366
+ setStore("mode", mode)
367
+ kv.set("theme_mode", mode)
368
+ },
369
+ set(theme: string) {
370
+ setStore("active", theme)
371
+ kv.set("theme", theme)
372
+ },
373
+ get ready() {
374
+ return store.ready
375
+ },
376
+ }
377
+ },
378
+ })
379
+
380
+ const CUSTOM_THEME_GLOB = new Bun.Glob("themes/*.json")
381
+ async function getCustomThemes() {
382
+ const directories = [
383
+ Global.Path.config,
384
+ ...(await Array.fromAsync(
385
+ Filesystem.up({
386
+ targets: [".opencode"],
387
+ start: process.cwd(),
388
+ }),
389
+ )),
390
+ ]
391
+
392
+ const result: Record<string, ThemeJson> = {}
393
+ for (const dir of directories) {
394
+ for await (const item of CUSTOM_THEME_GLOB.scan({
395
+ absolute: true,
396
+ followSymlinks: true,
397
+ dot: true,
398
+ cwd: dir,
399
+ })) {
400
+ const name = path.basename(item, ".json")
401
+ result[name] = await Bun.file(item).json()
402
+ }
403
+ }
404
+ return result
405
+ }
406
+
407
+ function generateSystem(colors: TerminalColors, mode: "dark" | "light"): ThemeJson {
408
+ const bg = RGBA.fromHex(colors.defaultBackground ?? colors.palette[0]!)
409
+ const fg = RGBA.fromHex(colors.defaultForeground ?? colors.palette[7]!)
410
+ const isDark = mode == "dark"
411
+
412
+ const col = (i: number) => {
413
+ const value = colors.palette[i]
414
+ if (value) return RGBA.fromHex(value)
415
+ return ansiToRgba(i)
416
+ }
417
+
418
+ const tint = (base: RGBA, overlay: RGBA, alpha: number) => {
419
+ const r = base.r + (overlay.r - base.r) * alpha
420
+ const g = base.g + (overlay.g - base.g) * alpha
421
+ const b = base.b + (overlay.b - base.b) * alpha
422
+ return RGBA.fromInts(Math.round(r * 255), Math.round(g * 255), Math.round(b * 255))
423
+ }
424
+
425
+ // Generate gray scale based on terminal background
426
+ const grays = generateGrayScale(bg, isDark)
427
+ const textMuted = generateMutedTextColor(bg, isDark)
428
+
429
+ // ANSI color references
430
+ const ansiColors = {
431
+ black: col(0),
432
+ red: col(1),
433
+ green: col(2),
434
+ yellow: col(3),
435
+ blue: col(4),
436
+ magenta: col(5),
437
+ cyan: col(6),
438
+ white: col(7),
439
+ redBright: col(9),
440
+ greenBright: col(10),
441
+ }
442
+
443
+ const diffAlpha = isDark ? 0.22 : 0.14
444
+ const diffAddedBg = tint(bg, ansiColors.green, diffAlpha)
445
+ const diffRemovedBg = tint(bg, ansiColors.red, diffAlpha)
446
+ const diffAddedLineNumberBg = tint(grays[3], ansiColors.green, diffAlpha)
447
+ const diffRemovedLineNumberBg = tint(grays[3], ansiColors.red, diffAlpha)
448
+
449
+ return {
450
+ theme: {
451
+ // Primary colors using ANSI
452
+ primary: ansiColors.cyan,
453
+ secondary: ansiColors.magenta,
454
+ accent: ansiColors.cyan,
455
+
456
+ // Status colors using ANSI
457
+ error: ansiColors.red,
458
+ warning: ansiColors.yellow,
459
+ success: ansiColors.green,
460
+ info: ansiColors.cyan,
461
+
462
+ // Text colors
463
+ text: fg,
464
+ textMuted,
465
+ selectedListItemText: bg,
466
+
467
+ // Background colors
468
+ background: bg,
469
+ backgroundPanel: grays[2],
470
+ backgroundElement: grays[3],
471
+ backgroundMenu: grays[3],
472
+
473
+ // Border colors
474
+ borderSubtle: grays[6],
475
+ border: grays[7],
476
+ borderActive: grays[8],
477
+
478
+ // Diff colors
479
+ diffAdded: ansiColors.green,
480
+ diffRemoved: ansiColors.red,
481
+ diffContext: grays[7],
482
+ diffHunkHeader: grays[7],
483
+ diffHighlightAdded: ansiColors.greenBright,
484
+ diffHighlightRemoved: ansiColors.redBright,
485
+ diffAddedBg,
486
+ diffRemovedBg,
487
+ diffContextBg: grays[1],
488
+ diffLineNumber: grays[6],
489
+ diffAddedLineNumberBg,
490
+ diffRemovedLineNumberBg,
491
+
492
+ // Markdown colors
493
+ markdownText: fg,
494
+ markdownHeading: fg,
495
+ markdownLink: ansiColors.blue,
496
+ markdownLinkText: ansiColors.cyan,
497
+ markdownCode: ansiColors.green,
498
+ markdownBlockQuote: ansiColors.yellow,
499
+ markdownEmph: ansiColors.yellow,
500
+ markdownStrong: fg,
501
+ markdownHorizontalRule: grays[7],
502
+ markdownListItem: ansiColors.blue,
503
+ markdownListEnumeration: ansiColors.cyan,
504
+ markdownImage: ansiColors.blue,
505
+ markdownImageText: ansiColors.cyan,
506
+ markdownCodeBlock: fg,
507
+
508
+ // Syntax colors
509
+ syntaxComment: textMuted,
510
+ syntaxKeyword: ansiColors.magenta,
511
+ syntaxFunction: ansiColors.blue,
512
+ syntaxVariable: fg,
513
+ syntaxString: ansiColors.green,
514
+ syntaxNumber: ansiColors.yellow,
515
+ syntaxType: ansiColors.cyan,
516
+ syntaxOperator: ansiColors.cyan,
517
+ syntaxPunctuation: fg,
518
+ },
519
+ }
520
+ }
521
+
522
+ function generateGrayScale(bg: RGBA, isDark: boolean): Record<number, RGBA> {
523
+ const grays: Record<number, RGBA> = {}
524
+
525
+ // RGBA stores floats in range 0-1, convert to 0-255
526
+ const bgR = bg.r * 255
527
+ const bgG = bg.g * 255
528
+ const bgB = bg.b * 255
529
+
530
+ const luminance = 0.299 * bgR + 0.587 * bgG + 0.114 * bgB
531
+
532
+ for (let i = 1; i <= 12; i++) {
533
+ const factor = i / 12.0
534
+
535
+ let grayValue: number
536
+ let newR: number
537
+ let newG: number
538
+ let newB: number
539
+
540
+ if (isDark) {
541
+ if (luminance < 10) {
542
+ grayValue = Math.floor(factor * 0.4 * 255)
543
+ newR = grayValue
544
+ newG = grayValue
545
+ newB = grayValue
546
+ } else {
547
+ const newLum = luminance + (255 - luminance) * factor * 0.4
548
+
549
+ const ratio = newLum / luminance
550
+ newR = Math.min(bgR * ratio, 255)
551
+ newG = Math.min(bgG * ratio, 255)
552
+ newB = Math.min(bgB * ratio, 255)
553
+ }
554
+ } else {
555
+ if (luminance > 245) {
556
+ grayValue = Math.floor(255 - factor * 0.4 * 255)
557
+ newR = grayValue
558
+ newG = grayValue
559
+ newB = grayValue
560
+ } else {
561
+ const newLum = luminance * (1 - factor * 0.4)
562
+
563
+ const ratio = newLum / luminance
564
+ newR = Math.max(bgR * ratio, 0)
565
+ newG = Math.max(bgG * ratio, 0)
566
+ newB = Math.max(bgB * ratio, 0)
567
+ }
568
+ }
569
+
570
+ grays[i] = RGBA.fromInts(Math.floor(newR), Math.floor(newG), Math.floor(newB))
571
+ }
572
+
573
+ return grays
574
+ }
575
+
576
+ function generateMutedTextColor(bg: RGBA, isDark: boolean): RGBA {
577
+ // RGBA stores floats in range 0-1, convert to 0-255
578
+ const bgR = bg.r * 255
579
+ const bgG = bg.g * 255
580
+ const bgB = bg.b * 255
581
+
582
+ const bgLum = 0.299 * bgR + 0.587 * bgG + 0.114 * bgB
583
+
584
+ let grayValue: number
585
+
586
+ if (isDark) {
587
+ if (bgLum < 10) {
588
+ // Very dark/black background
589
+ grayValue = 180 // #b4b4b4
590
+ } else {
591
+ // Scale up for lighter dark backgrounds
592
+ grayValue = Math.min(Math.floor(160 + bgLum * 0.3), 200)
593
+ }
594
+ } else {
595
+ if (bgLum > 245) {
596
+ // Very light/white background
597
+ grayValue = 75 // #4b4b4b
598
+ } else {
599
+ // Scale down for darker light backgrounds
600
+ grayValue = Math.max(Math.floor(100 - (255 - bgLum) * 0.2), 60)
601
+ }
602
+ }
603
+
604
+ return RGBA.fromInts(grayValue, grayValue, grayValue)
605
+ }
606
+
607
+ function generateSyntax(theme: Theme) {
608
+ return SyntaxStyle.fromTheme(getSyntaxRules(theme))
609
+ }
610
+
611
+ function generateSubtleSyntax(theme: Theme) {
612
+ const rules = getSyntaxRules(theme)
613
+ return SyntaxStyle.fromTheme(
614
+ rules.map((rule) => {
615
+ if (rule.style.foreground) {
616
+ const fg = rule.style.foreground
617
+ return {
618
+ ...rule,
619
+ style: {
620
+ ...rule.style,
621
+ foreground: RGBA.fromInts(
622
+ Math.round(fg.r * 255),
623
+ Math.round(fg.g * 255),
624
+ Math.round(fg.b * 255),
625
+ Math.round(theme.thinkingOpacity * 255),
626
+ ),
627
+ },
628
+ }
629
+ }
630
+ return rule
631
+ }),
632
+ )
633
+ }
634
+
635
+ function getSyntaxRules(theme: Theme) {
636
+ return [
637
+ {
638
+ scope: ["default"],
639
+ style: {
640
+ foreground: theme.text,
641
+ },
642
+ },
643
+ {
644
+ scope: ["prompt"],
645
+ style: {
646
+ foreground: theme.accent,
647
+ },
648
+ },
649
+ {
650
+ scope: ["extmark.file"],
651
+ style: {
652
+ foreground: theme.warning,
653
+ bold: true,
654
+ },
655
+ },
656
+ {
657
+ scope: ["extmark.agent"],
658
+ style: {
659
+ foreground: theme.secondary,
660
+ bold: true,
661
+ },
662
+ },
663
+ {
664
+ scope: ["extmark.paste"],
665
+ style: {
666
+ foreground: theme.background,
667
+ background: theme.warning,
668
+ bold: true,
669
+ },
670
+ },
671
+ {
672
+ scope: ["comment"],
673
+ style: {
674
+ foreground: theme.syntaxComment,
675
+ italic: true,
676
+ },
677
+ },
678
+ {
679
+ scope: ["comment.documentation"],
680
+ style: {
681
+ foreground: theme.syntaxComment,
682
+ italic: true,
683
+ },
684
+ },
685
+ {
686
+ scope: ["string", "symbol"],
687
+ style: {
688
+ foreground: theme.syntaxString,
689
+ },
690
+ },
691
+ {
692
+ scope: ["number", "boolean"],
693
+ style: {
694
+ foreground: theme.syntaxNumber,
695
+ },
696
+ },
697
+ {
698
+ scope: ["character.special"],
699
+ style: {
700
+ foreground: theme.syntaxString,
701
+ },
702
+ },
703
+ {
704
+ scope: ["keyword.return", "keyword.conditional", "keyword.repeat", "keyword.coroutine"],
705
+ style: {
706
+ foreground: theme.syntaxKeyword,
707
+ italic: true,
708
+ },
709
+ },
710
+ {
711
+ scope: ["keyword.type"],
712
+ style: {
713
+ foreground: theme.syntaxType,
714
+ bold: true,
715
+ italic: true,
716
+ },
717
+ },
718
+ {
719
+ scope: ["keyword.function", "function.method"],
720
+ style: {
721
+ foreground: theme.syntaxFunction,
722
+ },
723
+ },
724
+ {
725
+ scope: ["keyword"],
726
+ style: {
727
+ foreground: theme.syntaxKeyword,
728
+ italic: true,
729
+ },
730
+ },
731
+ {
732
+ scope: ["keyword.import"],
733
+ style: {
734
+ foreground: theme.syntaxKeyword,
735
+ },
736
+ },
737
+ {
738
+ scope: ["operator", "keyword.operator", "punctuation.delimiter"],
739
+ style: {
740
+ foreground: theme.syntaxOperator,
741
+ },
742
+ },
743
+ {
744
+ scope: ["keyword.conditional.ternary"],
745
+ style: {
746
+ foreground: theme.syntaxOperator,
747
+ },
748
+ },
749
+ {
750
+ scope: ["variable", "variable.parameter", "function.method.call", "function.call"],
751
+ style: {
752
+ foreground: theme.syntaxVariable,
753
+ },
754
+ },
755
+ {
756
+ scope: ["variable.member", "function", "constructor"],
757
+ style: {
758
+ foreground: theme.syntaxFunction,
759
+ },
760
+ },
761
+ {
762
+ scope: ["type", "module"],
763
+ style: {
764
+ foreground: theme.syntaxType,
765
+ },
766
+ },
767
+ {
768
+ scope: ["constant"],
769
+ style: {
770
+ foreground: theme.syntaxNumber,
771
+ },
772
+ },
773
+ {
774
+ scope: ["property"],
775
+ style: {
776
+ foreground: theme.syntaxVariable,
777
+ },
778
+ },
779
+ {
780
+ scope: ["class"],
781
+ style: {
782
+ foreground: theme.syntaxType,
783
+ },
784
+ },
785
+ {
786
+ scope: ["parameter"],
787
+ style: {
788
+ foreground: theme.syntaxVariable,
789
+ },
790
+ },
791
+ {
792
+ scope: ["punctuation", "punctuation.bracket"],
793
+ style: {
794
+ foreground: theme.syntaxPunctuation,
795
+ },
796
+ },
797
+ {
798
+ scope: ["variable.builtin", "type.builtin", "function.builtin", "module.builtin", "constant.builtin"],
799
+ style: {
800
+ foreground: theme.error,
801
+ },
802
+ },
803
+ {
804
+ scope: ["variable.super"],
805
+ style: {
806
+ foreground: theme.error,
807
+ },
808
+ },
809
+ {
810
+ scope: ["string.escape", "string.regexp"],
811
+ style: {
812
+ foreground: theme.syntaxKeyword,
813
+ },
814
+ },
815
+ {
816
+ scope: ["keyword.directive"],
817
+ style: {
818
+ foreground: theme.syntaxKeyword,
819
+ italic: true,
820
+ },
821
+ },
822
+ {
823
+ scope: ["punctuation.special"],
824
+ style: {
825
+ foreground: theme.syntaxOperator,
826
+ },
827
+ },
828
+ {
829
+ scope: ["keyword.modifier"],
830
+ style: {
831
+ foreground: theme.syntaxKeyword,
832
+ italic: true,
833
+ },
834
+ },
835
+ {
836
+ scope: ["keyword.exception"],
837
+ style: {
838
+ foreground: theme.syntaxKeyword,
839
+ italic: true,
840
+ },
841
+ },
842
+ // Markdown specific styles
843
+ {
844
+ scope: ["markup.heading"],
845
+ style: {
846
+ foreground: theme.markdownHeading,
847
+ bold: true,
848
+ },
849
+ },
850
+ {
851
+ scope: ["markup.heading.1"],
852
+ style: {
853
+ foreground: theme.markdownHeading,
854
+ bold: true,
855
+ },
856
+ },
857
+ {
858
+ scope: ["markup.heading.2"],
859
+ style: {
860
+ foreground: theme.markdownHeading,
861
+ bold: true,
862
+ },
863
+ },
864
+ {
865
+ scope: ["markup.heading.3"],
866
+ style: {
867
+ foreground: theme.markdownHeading,
868
+ bold: true,
869
+ },
870
+ },
871
+ {
872
+ scope: ["markup.heading.4"],
873
+ style: {
874
+ foreground: theme.markdownHeading,
875
+ bold: true,
876
+ },
877
+ },
878
+ {
879
+ scope: ["markup.heading.5"],
880
+ style: {
881
+ foreground: theme.markdownHeading,
882
+ bold: true,
883
+ },
884
+ },
885
+ {
886
+ scope: ["markup.heading.6"],
887
+ style: {
888
+ foreground: theme.markdownHeading,
889
+ bold: true,
890
+ },
891
+ },
892
+ {
893
+ scope: ["markup.bold", "markup.strong"],
894
+ style: {
895
+ foreground: theme.markdownStrong,
896
+ bold: true,
897
+ },
898
+ },
899
+ {
900
+ scope: ["markup.italic"],
901
+ style: {
902
+ foreground: theme.markdownEmph,
903
+ italic: true,
904
+ },
905
+ },
906
+ {
907
+ scope: ["markup.list"],
908
+ style: {
909
+ foreground: theme.markdownListItem,
910
+ },
911
+ },
912
+ {
913
+ scope: ["markup.quote"],
914
+ style: {
915
+ foreground: theme.markdownBlockQuote,
916
+ italic: true,
917
+ },
918
+ },
919
+ {
920
+ scope: ["markup.raw", "markup.raw.block"],
921
+ style: {
922
+ foreground: theme.markdownCode,
923
+ },
924
+ },
925
+ {
926
+ scope: ["markup.raw.inline"],
927
+ style: {
928
+ foreground: theme.markdownCode,
929
+ background: theme.background,
930
+ },
931
+ },
932
+ {
933
+ scope: ["markup.link"],
934
+ style: {
935
+ foreground: theme.markdownLink,
936
+ underline: true,
937
+ },
938
+ },
939
+ {
940
+ scope: ["markup.link.label"],
941
+ style: {
942
+ foreground: theme.markdownLinkText,
943
+ underline: true,
944
+ },
945
+ },
946
+ {
947
+ scope: ["markup.link.url"],
948
+ style: {
949
+ foreground: theme.markdownLink,
950
+ underline: true,
951
+ },
952
+ },
953
+ {
954
+ scope: ["label"],
955
+ style: {
956
+ foreground: theme.markdownLinkText,
957
+ },
958
+ },
959
+ {
960
+ scope: ["spell", "nospell"],
961
+ style: {
962
+ foreground: theme.text,
963
+ },
964
+ },
965
+ {
966
+ scope: ["conceal"],
967
+ style: {
968
+ foreground: theme.textMuted,
969
+ },
970
+ },
971
+ // Additional common highlight groups
972
+ {
973
+ scope: ["string.special", "string.special.url"],
974
+ style: {
975
+ foreground: theme.markdownLink,
976
+ underline: true,
977
+ },
978
+ },
979
+ {
980
+ scope: ["character"],
981
+ style: {
982
+ foreground: theme.syntaxString,
983
+ },
984
+ },
985
+ {
986
+ scope: ["float"],
987
+ style: {
988
+ foreground: theme.syntaxNumber,
989
+ },
990
+ },
991
+ {
992
+ scope: ["comment.error"],
993
+ style: {
994
+ foreground: theme.error,
995
+ italic: true,
996
+ bold: true,
997
+ },
998
+ },
999
+ {
1000
+ scope: ["comment.warning"],
1001
+ style: {
1002
+ foreground: theme.warning,
1003
+ italic: true,
1004
+ bold: true,
1005
+ },
1006
+ },
1007
+ {
1008
+ scope: ["comment.todo", "comment.note"],
1009
+ style: {
1010
+ foreground: theme.info,
1011
+ italic: true,
1012
+ bold: true,
1013
+ },
1014
+ },
1015
+ {
1016
+ scope: ["namespace"],
1017
+ style: {
1018
+ foreground: theme.syntaxType,
1019
+ },
1020
+ },
1021
+ {
1022
+ scope: ["field"],
1023
+ style: {
1024
+ foreground: theme.syntaxVariable,
1025
+ },
1026
+ },
1027
+ {
1028
+ scope: ["type.definition"],
1029
+ style: {
1030
+ foreground: theme.syntaxType,
1031
+ bold: true,
1032
+ },
1033
+ },
1034
+ {
1035
+ scope: ["keyword.export"],
1036
+ style: {
1037
+ foreground: theme.syntaxKeyword,
1038
+ },
1039
+ },
1040
+ {
1041
+ scope: ["attribute", "annotation"],
1042
+ style: {
1043
+ foreground: theme.warning,
1044
+ },
1045
+ },
1046
+ {
1047
+ scope: ["tag"],
1048
+ style: {
1049
+ foreground: theme.error,
1050
+ },
1051
+ },
1052
+ {
1053
+ scope: ["tag.attribute"],
1054
+ style: {
1055
+ foreground: theme.syntaxKeyword,
1056
+ },
1057
+ },
1058
+ {
1059
+ scope: ["tag.delimiter"],
1060
+ style: {
1061
+ foreground: theme.syntaxOperator,
1062
+ },
1063
+ },
1064
+ {
1065
+ scope: ["markup.strikethrough"],
1066
+ style: {
1067
+ foreground: theme.textMuted,
1068
+ },
1069
+ },
1070
+ {
1071
+ scope: ["markup.underline"],
1072
+ style: {
1073
+ foreground: theme.text,
1074
+ underline: true,
1075
+ },
1076
+ },
1077
+ {
1078
+ scope: ["markup.list.checked"],
1079
+ style: {
1080
+ foreground: theme.success,
1081
+ },
1082
+ },
1083
+ {
1084
+ scope: ["markup.list.unchecked"],
1085
+ style: {
1086
+ foreground: theme.textMuted,
1087
+ },
1088
+ },
1089
+ {
1090
+ scope: ["diff.plus"],
1091
+ style: {
1092
+ foreground: theme.diffAdded,
1093
+ background: theme.diffAddedBg,
1094
+ },
1095
+ },
1096
+ {
1097
+ scope: ["diff.minus"],
1098
+ style: {
1099
+ foreground: theme.diffRemoved,
1100
+ background: theme.diffRemovedBg,
1101
+ },
1102
+ },
1103
+ {
1104
+ scope: ["diff.delta"],
1105
+ style: {
1106
+ foreground: theme.diffContext,
1107
+ background: theme.diffContextBg,
1108
+ },
1109
+ },
1110
+ {
1111
+ scope: ["error"],
1112
+ style: {
1113
+ foreground: theme.error,
1114
+ bold: true,
1115
+ },
1116
+ },
1117
+ {
1118
+ scope: ["warning"],
1119
+ style: {
1120
+ foreground: theme.warning,
1121
+ bold: true,
1122
+ },
1123
+ },
1124
+ {
1125
+ scope: ["info"],
1126
+ style: {
1127
+ foreground: theme.info,
1128
+ },
1129
+ },
1130
+ {
1131
+ scope: ["debug"],
1132
+ style: {
1133
+ foreground: theme.textMuted,
1134
+ },
1135
+ },
1136
+ ]
1137
+ }