@fatdoge/wtree 0.2.1 → 0.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (319) hide show
  1. package/api/routes/worktrees.ts +50 -1
  2. package/dist/assets/abap-BdImnpbu.js +1 -0
  3. package/dist/assets/actionscript-3-CoDkCxhg.js +1 -0
  4. package/dist/assets/ada-bCR0ucgS.js +1 -0
  5. package/dist/assets/andromeeda-C4gqWexZ.js +1 -0
  6. package/dist/assets/angular-html-CU67Zn6k.js +1 -0
  7. package/dist/assets/angular-ts-BwZT4LLn.js +1 -0
  8. package/dist/assets/apache-Pmp26Uib.js +1 -0
  9. package/dist/assets/apex-D8_7TLub.js +1 -0
  10. package/dist/assets/apl-dKokRX4l.js +1 -0
  11. package/dist/assets/applescript-Co6uUVPk.js +1 -0
  12. package/dist/assets/ara-BRHolxvo.js +1 -0
  13. package/dist/assets/asciidoc-Ve4PFQV2.js +1 -0
  14. package/dist/assets/asm-D_Q5rh1f.js +1 -0
  15. package/dist/assets/astro-CbQHKStN.js +1 -0
  16. package/dist/assets/aurora-x-D-2ljcwZ.js +1 -0
  17. package/dist/assets/awk-DMzUqQB5.js +1 -0
  18. package/dist/assets/ayu-dark-DYE7WIF3.js +1 -0
  19. package/dist/assets/ayu-light-BA47KaF1.js +1 -0
  20. package/dist/assets/ayu-mirage-32ctXXKs.js +1 -0
  21. package/dist/assets/ballerina-BFfxhgS-.js +1 -0
  22. package/dist/assets/bat-BkioyH1T.js +1 -0
  23. package/dist/assets/beancount-k_qm7-4y.js +1 -0
  24. package/dist/assets/berry-uYugtg8r.js +1 -0
  25. package/dist/assets/bibtex-CHM0blh-.js +1 -0
  26. package/dist/assets/bicep-Bmn6On1c.js +1 -0
  27. package/dist/assets/bird2-DPOp833l.js +1 -0
  28. package/dist/assets/blade-D4QpJJKB.js +1 -0
  29. package/dist/assets/bsl-BO_Y6i37.js +1 -0
  30. package/dist/assets/c-BIGW1oBm.js +1 -0
  31. package/dist/assets/c3-eo99z4R2.js +1 -0
  32. package/dist/assets/cadence-Bv_4Rxtq.js +1 -0
  33. package/dist/assets/cairo-KRGpt6FW.js +1 -0
  34. package/dist/assets/catppuccin-frappe-DFWUc33u.js +1 -0
  35. package/dist/assets/catppuccin-latte-C9dUb6Cb.js +1 -0
  36. package/dist/assets/catppuccin-macchiato-DQyhUUbL.js +1 -0
  37. package/dist/assets/catppuccin-mocha-D87Tk5Gz.js +1 -0
  38. package/dist/assets/clarity-D53aC0YG.js +1 -0
  39. package/dist/assets/clojure-P80f7IUj.js +1 -0
  40. package/dist/assets/cmake-D1j8_8rp.js +1 -0
  41. package/dist/assets/cobol-nwyudZeR.js +1 -0
  42. package/dist/assets/codeowners-Bp6g37R7.js +1 -0
  43. package/dist/assets/codeql-DsOJ9woJ.js +1 -0
  44. package/dist/assets/coffee-Ch7k5sss.js +1 -0
  45. package/dist/assets/common-lisp-Cg-RD9OK.js +1 -0
  46. package/dist/assets/coq-DkFqJrB1.js +1 -0
  47. package/dist/assets/cpp-CofmeUqb.js +1 -0
  48. package/dist/assets/crystal-tKQVLTB8.js +1 -0
  49. package/dist/assets/csharp-COcwbKMJ.js +1 -0
  50. package/dist/assets/css-DPfMkruS.js +1 -0
  51. package/dist/assets/csv-fuZLfV_i.js +1 -0
  52. package/dist/assets/cue-D82EKSYY.js +1 -0
  53. package/dist/assets/cypher-COkxafJQ.js +1 -0
  54. package/dist/assets/d-85-TOEBH.js +1 -0
  55. package/dist/assets/dark-plus-C3mMm8J8.js +1 -0
  56. package/dist/assets/dart-CF10PKvl.js +1 -0
  57. package/dist/assets/dax-CEL-wOlO.js +1 -0
  58. package/dist/assets/desktop-BmXAJ9_W.js +1 -0
  59. package/dist/assets/diff-D97Zzqfu.js +1 -0
  60. package/dist/assets/docker-BcOcwvcX.js +1 -0
  61. package/dist/assets/dotenv-Da5cRb03.js +1 -0
  62. package/dist/assets/dracula-BzJJZx-M.js +1 -0
  63. package/dist/assets/dracula-soft-BXkSAIEj.js +1 -0
  64. package/dist/assets/dream-maker-BtqSS_iP.js +1 -0
  65. package/dist/assets/edge-BkV0erSs.js +1 -0
  66. package/dist/assets/elixir-CDX3lj18.js +1 -0
  67. package/dist/assets/elm-DbKCFpqz.js +1 -0
  68. package/dist/assets/emacs-lisp-C9XAeP06.js +1 -0
  69. package/dist/assets/erb-B12qg9BL.js +1 -0
  70. package/dist/assets/erlang-DsQrWhSR.js +1 -0
  71. package/dist/assets/everforest-dark-BgDCqdQA.js +1 -0
  72. package/dist/assets/everforest-light-C8M2exoo.js +1 -0
  73. package/dist/assets/fennel-BYunw83y.js +1 -0
  74. package/dist/assets/fish-BvzEVeQv.js +1 -0
  75. package/dist/assets/fluent-C4IJs8-o.js +1 -0
  76. package/dist/assets/fortran-fixed-form-CkoXwp7k.js +1 -0
  77. package/dist/assets/fortran-free-form-BxgE0vQu.js +1 -0
  78. package/dist/assets/fsharp-CXgrBDvD.js +1 -0
  79. package/dist/assets/gdresource-BOOCDP_w.js +1 -0
  80. package/dist/assets/gdscript-C5YyOfLZ.js +1 -0
  81. package/dist/assets/gdshader-DkwncUOv.js +1 -0
  82. package/dist/assets/genie-D0YGMca9.js +1 -0
  83. package/dist/assets/gherkin-DyxjwDmM.js +1 -0
  84. package/dist/assets/git-commit-F4YmCXRG.js +1 -0
  85. package/dist/assets/git-rebase-r7XF79zn.js +1 -0
  86. package/dist/assets/github-dark-DHJKELXO.js +1 -0
  87. package/dist/assets/github-dark-default-Cuk6v7N8.js +1 -0
  88. package/dist/assets/github-dark-dimmed-DH5Ifo-i.js +1 -0
  89. package/dist/assets/github-dark-high-contrast-E3gJ1_iC.js +1 -0
  90. package/dist/assets/github-light-DAi9KRSo.js +1 -0
  91. package/dist/assets/github-light-default-D7oLnXFd.js +1 -0
  92. package/dist/assets/github-light-high-contrast-BfjtVDDH.js +1 -0
  93. package/dist/assets/gleam-BspZqrRM.js +1 -0
  94. package/dist/assets/glimmer-js-Rg0-pVw9.js +1 -0
  95. package/dist/assets/glimmer-ts-U6CK756n.js +1 -0
  96. package/dist/assets/glsl-DplSGwfg.js +1 -0
  97. package/dist/assets/gn-n2N0HUVH.js +1 -0
  98. package/dist/assets/gnuplot-DdkO51Og.js +1 -0
  99. package/dist/assets/go-CxLEBnE3.js +1 -0
  100. package/dist/assets/graphql-ChdNCCLP.js +1 -0
  101. package/dist/assets/groovy-gcz8RCvz.js +1 -0
  102. package/dist/assets/gruvbox-dark-hard-CFHQjOhq.js +1 -0
  103. package/dist/assets/gruvbox-dark-medium-GsRaNv29.js +1 -0
  104. package/dist/assets/gruvbox-dark-soft-CVdnzihN.js +1 -0
  105. package/dist/assets/gruvbox-light-hard-CH1njM8p.js +1 -0
  106. package/dist/assets/gruvbox-light-medium-DRw_LuNl.js +1 -0
  107. package/dist/assets/gruvbox-light-soft-hJgmCMqR.js +1 -0
  108. package/dist/assets/hack-CaT9iCJl.js +1 -0
  109. package/dist/assets/haml-B8DHNrY2.js +1 -0
  110. package/dist/assets/handlebars-BL8al0AC.js +1 -0
  111. package/dist/assets/haskell-Df6bDoY_.js +1 -0
  112. package/dist/assets/haxe-CzTSHFRz.js +1 -0
  113. package/dist/assets/hcl-BWvSN4gD.js +1 -0
  114. package/dist/assets/hjson-D5-asLiD.js +1 -0
  115. package/dist/assets/hlsl-D3lLCCz7.js +1 -0
  116. package/dist/assets/horizon-BUw7H-hv.js +1 -0
  117. package/dist/assets/horizon-bright-Cn-bp-IR.js +1 -0
  118. package/dist/assets/houston-DnULxvSX.js +1 -0
  119. package/dist/assets/html-GMplVEZG.js +1 -0
  120. package/dist/assets/html-derivative-BFtXZ54Q.js +1 -0
  121. package/dist/assets/http-jrhK8wxY.js +1 -0
  122. package/dist/assets/hurl-irOxFIW8.js +1 -0
  123. package/dist/assets/hxml-Bvhsp5Yf.js +1 -0
  124. package/dist/assets/hy-DFXneXwc.js +1 -0
  125. package/dist/assets/imba-DGztddWO.js +1 -0
  126. package/dist/assets/index-bLnnvz_q.css +1 -0
  127. package/dist/assets/index-sP_n0D3A.js +2246 -0
  128. package/dist/assets/ini-BEwlwnbL.js +1 -0
  129. package/dist/assets/java-CylS5w8V.js +1 -0
  130. package/dist/assets/javascript-wDzz0qaB.js +1 -0
  131. package/dist/assets/jinja-4LBKfQ-Z.js +1 -0
  132. package/dist/assets/jison-wvAkD_A8.js +1 -0
  133. package/dist/assets/json-Cp-IABpG.js +1 -0
  134. package/dist/assets/json5-C9tS-k6U.js +1 -0
  135. package/dist/assets/jsonc-Des-eS-w.js +1 -0
  136. package/dist/assets/jsonl-DcaNXYhu.js +1 -0
  137. package/dist/assets/jsonnet-DFQXde-d.js +1 -0
  138. package/dist/assets/jssm-C2t-YnRu.js +1 -0
  139. package/dist/assets/jsx-g9-lgVsj.js +1 -0
  140. package/dist/assets/julia-CxzCAyBv.js +1 -0
  141. package/dist/assets/just-Cw27pwNe.js +1 -0
  142. package/dist/assets/kanagawa-dragon-CkXjmgJE.js +1 -0
  143. package/dist/assets/kanagawa-lotus-CfQXZHmo.js +1 -0
  144. package/dist/assets/kanagawa-wave-DWedfzmr.js +1 -0
  145. package/dist/assets/kdl-DV7GczEv.js +1 -0
  146. package/dist/assets/kotlin-BdnUsdx6.js +1 -0
  147. package/dist/assets/kusto-DZf3V79B.js +1 -0
  148. package/dist/assets/laserwave-DUszq2jm.js +1 -0
  149. package/dist/assets/latex-CWtU0Tv5.js +1 -0
  150. package/dist/assets/lean-BZvkOJ9d.js +1 -0
  151. package/dist/assets/less-B1dDrJ26.js +1 -0
  152. package/dist/assets/light-plus-B7mTdjB0.js +1 -0
  153. package/dist/assets/liquid-DYVedYrR.js +1 -0
  154. package/dist/assets/llvm-DjAJT7YJ.js +1 -0
  155. package/dist/assets/log-2UxHyX5q.js +1 -0
  156. package/dist/assets/logo-BtOb2qkB.js +1 -0
  157. package/dist/assets/lua-BaeVxFsk.js +1 -0
  158. package/dist/assets/luau-C-HG3fhB.js +1 -0
  159. package/dist/assets/make-CHLpvVh8.js +1 -0
  160. package/dist/assets/markdown-Cvjx9yec.js +1 -0
  161. package/dist/assets/marko-CnJfTvn9.js +1 -0
  162. package/dist/assets/material-theme-D5KoaKCx.js +1 -0
  163. package/dist/assets/material-theme-darker-BfHTSMKl.js +1 -0
  164. package/dist/assets/material-theme-lighter-B0m2ddpp.js +1 -0
  165. package/dist/assets/material-theme-ocean-CyktbL80.js +1 -0
  166. package/dist/assets/material-theme-palenight-Csfq5Kiy.js +1 -0
  167. package/dist/assets/matlab-D7o27uSR.js +1 -0
  168. package/dist/assets/mdc-BMNejdWA.js +1 -0
  169. package/dist/assets/mdx-Cmh6b_Ma.js +1 -0
  170. package/dist/assets/mermaid-mWjccvbQ.js +1 -0
  171. package/dist/assets/min-dark-CafNBF8u.js +1 -0
  172. package/dist/assets/min-light-CTRr51gU.js +1 -0
  173. package/dist/assets/mipsasm-CKIfxQSi.js +1 -0
  174. package/dist/assets/mojo-rZm6bMo-.js +1 -0
  175. package/dist/assets/monokai-D4h5O-jR.js +1 -0
  176. package/dist/assets/moonbit-_H4v1dQx.js +1 -0
  177. package/dist/assets/move-IF9eRakj.js +1 -0
  178. package/dist/assets/narrat-DRg8JJMk.js +1 -0
  179. package/dist/assets/nextflow-Zz6hmt5N.js +1 -0
  180. package/dist/assets/nextflow-groovy-BeH2EWoN.js +1 -0
  181. package/dist/assets/nginx-BpAMiNFr.js +1 -0
  182. package/dist/assets/night-owl-C39BiMTA.js +1 -0
  183. package/dist/assets/night-owl-light-CMTm3GFP.js +1 -0
  184. package/dist/assets/nim-CVrawwO9.js +1 -0
  185. package/dist/assets/nix-CwoSXNpI.js +1 -0
  186. package/dist/assets/nord-Ddv68eIx.js +1 -0
  187. package/dist/assets/nushell-Cz2AlsmD.js +1 -0
  188. package/dist/assets/objective-c-DXmwc3jG.js +1 -0
  189. package/dist/assets/objective-cpp-CLxacb5B.js +1 -0
  190. package/dist/assets/ocaml-C0hk2d4L.js +1 -0
  191. package/dist/assets/odin-BBf5iR-q.js +1 -0
  192. package/dist/assets/one-dark-pro-DVMEJ2y_.js +1 -0
  193. package/dist/assets/one-light-C3Wv6jpd.js +1 -0
  194. package/dist/assets/openscad-C4EeE6gA.js +1 -0
  195. package/dist/assets/pascal-D93ZcfNL.js +1 -0
  196. package/dist/assets/perl-C0TMdlhV.js +1 -0
  197. package/dist/assets/php-Dhbhpdrm.js +1 -0
  198. package/dist/assets/pierre-dark-DF2SEV7i.js +1 -0
  199. package/dist/assets/pierre-light-DOlZxES8.js +1 -0
  200. package/dist/assets/pkl-u5AG7uiY.js +1 -0
  201. package/dist/assets/plastic-3e1v2bzS.js +1 -0
  202. package/dist/assets/plsql-ChMvpjG-.js +1 -0
  203. package/dist/assets/po-BTJTHyun.js +1 -0
  204. package/dist/assets/poimandres-CS3Unz2-.js +1 -0
  205. package/dist/assets/polar-C0HS_06l.js +1 -0
  206. package/dist/assets/postcss-CXtECtnM.js +1 -0
  207. package/dist/assets/powerquery-CEu0bR-o.js +1 -0
  208. package/dist/assets/powershell-Dpen1YoG.js +1 -0
  209. package/dist/assets/prisma-Dd19v3D-.js +1 -0
  210. package/dist/assets/prolog-CbFg5uaA.js +1 -0
  211. package/dist/assets/proto-C7zT0LnQ.js +1 -0
  212. package/dist/assets/pug-CGlum2m_.js +1 -0
  213. package/dist/assets/puppet-BMWR74SV.js +1 -0
  214. package/dist/assets/purescript-CklMAg4u.js +1 -0
  215. package/dist/assets/python-B6aJPvgy.js +1 -0
  216. package/dist/assets/qml-3beO22l8.js +1 -0
  217. package/dist/assets/qmldir-C8lEn-DE.js +1 -0
  218. package/dist/assets/qss-IeuSbFQv.js +1 -0
  219. package/dist/assets/r-Dspwwk_N.js +1 -0
  220. package/dist/assets/racket-BqYA7rlc.js +1 -0
  221. package/dist/assets/raku-DXvB9xmW.js +1 -0
  222. package/dist/assets/razor-Uh8Bk_45.js +1 -0
  223. package/dist/assets/red-bN70gL4F.js +1 -0
  224. package/dist/assets/reg-C-SQnVFl.js +1 -0
  225. package/dist/assets/regexp-CDVJQ6XC.js +1 -0
  226. package/dist/assets/rel-C3B-1QV4.js +1 -0
  227. package/dist/assets/riscv-BM1_JUlF.js +1 -0
  228. package/dist/assets/ron-D8l8udqQ.js +1 -0
  229. package/dist/assets/rose-pine-dawn-DHQR4-dF.js +1 -0
  230. package/dist/assets/rose-pine-moon-D4_iv3hh.js +1 -0
  231. package/dist/assets/rose-pine-qdsjHGoJ.js +1 -0
  232. package/dist/assets/rosmsg-BJDFO7_C.js +1 -0
  233. package/dist/assets/rst-BrH8l1NY.js +1 -0
  234. package/dist/assets/ruby-Dw2BHqvy.js +1 -0
  235. package/dist/assets/rust-B1yitclQ.js +1 -0
  236. package/dist/assets/sas-cz2c8ADy.js +1 -0
  237. package/dist/assets/sass-Cj5Yp3dK.js +1 -0
  238. package/dist/assets/scala-C151Ov-r.js +1 -0
  239. package/dist/assets/scheme-C98Dy4si.js +1 -0
  240. package/dist/assets/scss-OYdSNvt2.js +1 -0
  241. package/dist/assets/sdbl-DVxCFoDh.js +1 -0
  242. package/dist/assets/shaderlab-Dg9Lc6iA.js +1 -0
  243. package/dist/assets/shellscript-Yzrsuije.js +1 -0
  244. package/dist/assets/shellsession-BADoaaVG.js +1 -0
  245. package/dist/assets/slack-dark-BthQWCQV.js +1 -0
  246. package/dist/assets/slack-ochin-DqwNpetd.js +1 -0
  247. package/dist/assets/smalltalk-BERRCDM3.js +1 -0
  248. package/dist/assets/snazzy-light-Bw305WKR.js +1 -0
  249. package/dist/assets/solarized-dark-DXbdFlpD.js +1 -0
  250. package/dist/assets/solarized-light-L9t79GZl.js +1 -0
  251. package/dist/assets/solidity-rGO070M0.js +1 -0
  252. package/dist/assets/soy-Brmx7dQM.js +1 -0
  253. package/dist/assets/sparql-rVzFXLq3.js +1 -0
  254. package/dist/assets/splunk-BtCnVYZw.js +1 -0
  255. package/dist/assets/sql-BLtJtn59.js +1 -0
  256. package/dist/assets/ssh-config-_ykCGR6B.js +1 -0
  257. package/dist/assets/stata-BH5u7GGu.js +1 -0
  258. package/dist/assets/stylus-BEDo0Tqx.js +1 -0
  259. package/dist/assets/surrealql-Bq5Q-fJD.js +1 -0
  260. package/dist/assets/svelte-C_ipcX3V.js +1 -0
  261. package/dist/assets/swift-D82vCrfD.js +1 -0
  262. package/dist/assets/synthwave-84-CbfX1IO0.js +1 -0
  263. package/dist/assets/system-verilog-CnnmHF94.js +1 -0
  264. package/dist/assets/systemd-4A_iFExJ.js +1 -0
  265. package/dist/assets/talonscript-CkByrt1z.js +1 -0
  266. package/dist/assets/tasl-QIJgUcNo.js +1 -0
  267. package/dist/assets/tcl-dwOrl1Do.js +1 -0
  268. package/dist/assets/templ-P3uqSqPl.js +1 -0
  269. package/dist/assets/terraform-BETggiCN.js +1 -0
  270. package/dist/assets/tex-idrVyKtj.js +1 -0
  271. package/dist/assets/tokyo-night-hegEt444.js +1 -0
  272. package/dist/assets/toml-vGWfd6FD.js +1 -0
  273. package/dist/assets/ts-tags-zn1MmPIZ.js +1 -0
  274. package/dist/assets/tsv-B_m7g4N7.js +1 -0
  275. package/dist/assets/tsx-COt5Ahok.js +1 -0
  276. package/dist/assets/turtle-BsS91CYL.js +1 -0
  277. package/dist/assets/twig-DNn4PbVi.js +1 -0
  278. package/dist/assets/typescript-BPQ3VLAy.js +1 -0
  279. package/dist/assets/typespec-BGHnOYBU.js +1 -0
  280. package/dist/assets/typst-DHCkPAjA.js +1 -0
  281. package/dist/assets/v-BcVCzyr7.js +1 -0
  282. package/dist/assets/vala-CsfeWuGM.js +1 -0
  283. package/dist/assets/vb-D17OF-Vu.js +1 -0
  284. package/dist/assets/verilog-BQ8w6xss.js +1 -0
  285. package/dist/assets/vesper-DU1UobuO.js +1 -0
  286. package/dist/assets/vhdl-CeAyd5Ju.js +1 -0
  287. package/dist/assets/viml-CJc9bBzg.js +1 -0
  288. package/dist/assets/vitesse-black-Bkuqu6BP.js +1 -0
  289. package/dist/assets/vitesse-dark-D0r3Knsf.js +1 -0
  290. package/dist/assets/vitesse-light-CVO1_9PV.js +1 -0
  291. package/dist/assets/vue-DN_0RTcg.js +1 -0
  292. package/dist/assets/vue-html-AaS7Mt5G.js +1 -0
  293. package/dist/assets/vue-vine-CQOfvN7w.js +1 -0
  294. package/dist/assets/vyper-CDx5xZoG.js +1 -0
  295. package/dist/assets/wasm-CG6Dc4jp.js +1 -0
  296. package/dist/assets/wasm-MzD3tlZU.js +1 -0
  297. package/dist/assets/wenyan-BV7otONQ.js +1 -0
  298. package/dist/assets/wgsl-Dx-B1_4e.js +1 -0
  299. package/dist/assets/wikitext-BhOHFoWU.js +1 -0
  300. package/dist/assets/wit-5i3qLPDT.js +1 -0
  301. package/dist/assets/wolfram-lXgVvXCa.js +1 -0
  302. package/dist/assets/xml-sdJ4AIDG.js +1 -0
  303. package/dist/assets/xsl-CtQFsRM5.js +1 -0
  304. package/dist/assets/yaml-Buea-lGh.js +1 -0
  305. package/dist/assets/zenscript-DVFEvuxE.js +1 -0
  306. package/dist/assets/zig-VOosw3JB.js +1 -0
  307. package/dist/index.html +2 -2
  308. package/dist-node/api/routes/worktrees.js +40 -1
  309. package/package.json +3 -1
  310. package/shared/wtui-types.ts +20 -0
  311. package/skills/wtree/SKILL.md +21 -2
  312. package/src/components/DiffPreviewModal.tsx +381 -0
  313. package/src/components/Modal.tsx +11 -2
  314. package/src/i18n/locales/en.json +17 -0
  315. package/src/i18n/locales/zh.json +17 -0
  316. package/src/pages/Worktrees.tsx +55 -1
  317. package/src/stores/worktreeStore.ts +7 -1
  318. package/dist/assets/index-C78PMV-C.js +0 -179
  319. package/dist/assets/index-CR9jga1C.css +0 -1
@@ -0,0 +1 @@
1
+ import e from"./java-CylS5w8V.js";const n=Object.freeze(JSON.parse(`{"displayName":"XML","name":"xml","patterns":[{"begin":"(<\\\\?)\\\\s*([-0-9A-Z_a-z]+)","captures":{"1":{"name":"punctuation.definition.tag.xml"},"2":{"name":"entity.name.tag.xml"}},"end":"(\\\\?>)","name":"meta.tag.preprocessor.xml","patterns":[{"match":" ([-A-Za-z]+)","name":"entity.other.attribute-name.xml"},{"include":"#doublequotedString"},{"include":"#singlequotedString"}]},{"begin":"(<!)(DOCTYPE)\\\\s+([:A-Z_a-z][-.0-:A-Z_a-z]*)","captures":{"1":{"name":"punctuation.definition.tag.xml"},"2":{"name":"keyword.other.doctype.xml"},"3":{"name":"variable.language.documentroot.xml"}},"end":"\\\\s*(>)","name":"meta.tag.sgml.doctype.xml","patterns":[{"include":"#internalSubset"}]},{"include":"#comments"},{"begin":"(<)((?:([-0-9A-Z_a-z]+)(:))?([-0-:A-Z_a-z]+))(?=(\\\\s[^>]*)?></\\\\2>)","beginCaptures":{"1":{"name":"punctuation.definition.tag.xml"},"2":{"name":"entity.name.tag.xml"},"3":{"name":"entity.name.tag.namespace.xml"},"4":{"name":"punctuation.separator.namespace.xml"},"5":{"name":"entity.name.tag.localname.xml"}},"end":"(>)(</)((?:([-0-9A-Z_a-z]+)(:))?([-0-:A-Z_a-z]+))(>)","endCaptures":{"1":{"name":"punctuation.definition.tag.xml"},"2":{"name":"punctuation.definition.tag.xml"},"3":{"name":"entity.name.tag.xml"},"4":{"name":"entity.name.tag.namespace.xml"},"5":{"name":"punctuation.separator.namespace.xml"},"6":{"name":"entity.name.tag.localname.xml"},"7":{"name":"punctuation.definition.tag.xml"}},"name":"meta.tag.no-content.xml","patterns":[{"include":"#tagStuff"}]},{"begin":"(</?)(?:([-.\\\\w]+)((:)))?([-.:\\\\w]+)","captures":{"1":{"name":"punctuation.definition.tag.xml"},"2":{"name":"entity.name.tag.namespace.xml"},"3":{"name":"entity.name.tag.xml"},"4":{"name":"punctuation.separator.namespace.xml"},"5":{"name":"entity.name.tag.localname.xml"}},"end":"(/?>)","name":"meta.tag.xml","patterns":[{"include":"#tagStuff"}]},{"include":"#entity"},{"include":"#bare-ampersand"},{"begin":"<%@","beginCaptures":{"0":{"name":"punctuation.section.embedded.begin.xml"}},"end":"%>","endCaptures":{"0":{"name":"punctuation.section.embedded.end.xml"}},"name":"source.java-props.embedded.xml","patterns":[{"match":"page|include|taglib","name":"keyword.other.page-props.xml"}]},{"begin":"<%[!=]?(?!--)","beginCaptures":{"0":{"name":"punctuation.section.embedded.begin.xml"}},"end":"(?!--)%>","endCaptures":{"0":{"name":"punctuation.section.embedded.end.xml"}},"name":"source.java.embedded.xml","patterns":[{"include":"source.java"}]},{"begin":"<!\\\\[CDATA\\\\[","beginCaptures":{"0":{"name":"punctuation.definition.string.begin.xml"}},"end":"]]>","endCaptures":{"0":{"name":"punctuation.definition.string.end.xml"}},"name":"string.unquoted.cdata.xml"}],"repository":{"EntityDecl":{"begin":"(<!)(ENTITY)\\\\s+(%\\\\s+)?([:A-Z_a-z][-.0-:A-Z_a-z]*)(\\\\s+(?:SYSTEM|PUBLIC)\\\\s+)?","captures":{"1":{"name":"punctuation.definition.tag.xml"},"2":{"name":"keyword.other.entity.xml"},"3":{"name":"punctuation.definition.entity.xml"},"4":{"name":"variable.language.entity.xml"},"5":{"name":"keyword.other.entitytype.xml"}},"end":"(>)","patterns":[{"include":"#doublequotedString"},{"include":"#singlequotedString"}]},"bare-ampersand":{"match":"&","name":"invalid.illegal.bad-ampersand.xml"},"comments":{"patterns":[{"begin":"<%--","captures":{"0":{"name":"punctuation.definition.comment.xml"},"end":"--%>","name":"comment.block.xml"}},{"begin":"<!--","captures":{"0":{"name":"punctuation.definition.comment.xml"}},"end":"-->","name":"comment.block.xml","patterns":[{"begin":"--(?!>)","captures":{"0":{"name":"invalid.illegal.bad-comments-or-CDATA.xml"}}}]}]},"doublequotedString":{"begin":"\\"","beginCaptures":{"0":{"name":"punctuation.definition.string.begin.xml"}},"end":"\\"","endCaptures":{"0":{"name":"punctuation.definition.string.end.xml"}},"name":"string.quoted.double.xml","patterns":[{"include":"#entity"},{"include":"#bare-ampersand"}]},"entity":{"captures":{"1":{"name":"punctuation.definition.constant.xml"},"3":{"name":"punctuation.definition.constant.xml"}},"match":"(&)([:A-Z_a-z][-.0-:A-Z_a-z]*|#[0-9]+|#x\\\\h+)(;)","name":"constant.character.entity.xml"},"internalSubset":{"begin":"(\\\\[)","captures":{"1":{"name":"punctuation.definition.constant.xml"}},"end":"(])","name":"meta.internalsubset.xml","patterns":[{"include":"#EntityDecl"},{"include":"#parameterEntity"},{"include":"#comments"}]},"parameterEntity":{"captures":{"1":{"name":"punctuation.definition.constant.xml"},"3":{"name":"punctuation.definition.constant.xml"}},"match":"(%)([:A-Z_a-z][-.0-:A-Z_a-z]*)(;)","name":"constant.character.parameter-entity.xml"},"singlequotedString":{"begin":"'","beginCaptures":{"0":{"name":"punctuation.definition.string.begin.xml"}},"end":"'","endCaptures":{"0":{"name":"punctuation.definition.string.end.xml"}},"name":"string.quoted.single.xml","patterns":[{"include":"#entity"},{"include":"#bare-ampersand"}]},"tagStuff":{"patterns":[{"captures":{"1":{"name":"entity.other.attribute-name.namespace.xml"},"2":{"name":"entity.other.attribute-name.xml"},"3":{"name":"punctuation.separator.namespace.xml"},"4":{"name":"entity.other.attribute-name.localname.xml"}},"match":"(?:^|\\\\s+)(?:([-.\\\\w]+)((:)))?([-.:\\\\w]+)\\\\s*="},{"include":"#doublequotedString"},{"include":"#singlequotedString"}]}},"scopeName":"text.xml","embeddedLangs":["java"]}`)),a=[...e,n];export{a as default};
@@ -0,0 +1 @@
1
+ import e from"./xml-sdJ4AIDG.js";import"./java-CylS5w8V.js";const n=Object.freeze(JSON.parse(`{"displayName":"XSL","name":"xsl","patterns":[{"begin":"(<)(xsl)((:))(template)","captures":{"1":{"name":"punctuation.definition.tag.xml"},"2":{"name":"entity.name.tag.namespace.xml"},"3":{"name":"entity.name.tag.xml"},"4":{"name":"punctuation.separator.namespace.xml"},"5":{"name":"entity.name.tag.localname.xml"}},"end":"(>)","name":"meta.tag.xml.template","patterns":[{"captures":{"1":{"name":"entity.other.attribute-name.namespace.xml"},"2":{"name":"entity.other.attribute-name.xml"},"3":{"name":"punctuation.separator.namespace.xml"},"4":{"name":"entity.other.attribute-name.localname.xml"}},"match":" (?:([-0-9A-Z_a-z]+)((:)))?([-A-Za-z]+)"},{"include":"#doublequotedString"},{"include":"#singlequotedString"}]},{"include":"text.xml"}],"repository":{"doublequotedString":{"begin":"\\"","beginCaptures":{"0":{"name":"punctuation.definition.string.begin.xml"}},"end":"\\"","endCaptures":{"0":{"name":"punctuation.definition.string.end.xml"}},"name":"string.quoted.double.xml"},"singlequotedString":{"begin":"'","beginCaptures":{"0":{"name":"punctuation.definition.string.begin.xml"}},"end":"'","endCaptures":{"0":{"name":"punctuation.definition.string.end.xml"}},"name":"string.quoted.single.xml"}},"scopeName":"text.xml.xsl","embeddedLangs":["xml"]}`)),m=[...e,n];export{m as default};
@@ -0,0 +1 @@
1
+ const e=Object.freeze(JSON.parse(`{"displayName":"YAML","fileTypes":["yaml","yml","rviz","reek","clang-format","yaml-tmlanguage","syntax","sublime-syntax"],"firstLineMatch":"^%YAML( ?1.\\\\d+)?","name":"yaml","patterns":[{"include":"#comment"},{"include":"#property"},{"include":"#directive"},{"match":"^---","name":"entity.other.document.begin.yaml"},{"match":"^\\\\.{3}","name":"entity.other.document.end.yaml"},{"include":"#node"}],"repository":{"block-collection":{"patterns":[{"include":"#block-sequence"},{"include":"#block-mapping"}]},"block-mapping":{"patterns":[{"include":"#block-pair"}]},"block-node":{"patterns":[{"include":"#prototype"},{"include":"#block-scalar"},{"include":"#block-collection"},{"include":"#flow-scalar-plain-out"},{"include":"#flow-node"}]},"block-pair":{"patterns":[{"begin":"\\\\?","beginCaptures":{"1":{"name":"punctuation.definition.key-value.begin.yaml"}},"end":"(?=\\\\?)|^ *(:)|(:)","endCaptures":{"1":{"name":"punctuation.separator.key-value.mapping.yaml"},"2":{"name":"invalid.illegal.expected-newline.yaml"}},"name":"meta.block-mapping.yaml","patterns":[{"include":"#block-node"}]},{"begin":"(?=(?:[^-\\\\]!\\"#%\\\\&'*,:>?@\\\\[\`{|}\\\\s]|[-:?]\\\\S)([^:\\\\s]|:\\\\S|\\\\s+(?![#\\\\s]))*\\\\s*:(\\\\s|$))","end":"(?=\\\\s*$|\\\\s+#|\\\\s*:(\\\\s|$))","patterns":[{"include":"#flow-scalar-plain-out-implicit-type"},{"begin":"[^-\\\\]!\\"#%\\\\&'*,:>?@\\\\[\`{|}\\\\s]|[-:?]\\\\S","beginCaptures":{"0":{"name":"entity.name.tag.yaml"}},"contentName":"entity.name.tag.yaml","end":"(?=\\\\s*$|\\\\s+#|\\\\s*:(\\\\s|$))","name":"string.unquoted.plain.out.yaml"}]},{"match":":(?=\\\\s|$)","name":"punctuation.separator.key-value.mapping.yaml"}]},"block-scalar":{"begin":"(?:(\\\\|)|(>))([1-9])?([-+])?(.*\\\\n?)","beginCaptures":{"1":{"name":"keyword.control.flow.block-scalar.literal.yaml"},"2":{"name":"keyword.control.flow.block-scalar.folded.yaml"},"3":{"name":"constant.numeric.indentation-indicator.yaml"},"4":{"name":"storage.modifier.chomping-indicator.yaml"},"5":{"patterns":[{"include":"#comment"},{"match":".+","name":"invalid.illegal.expected-comment-or-newline.yaml"}]}},"end":"^(?=\\\\S)|(?!\\\\G)","patterns":[{"begin":"^( +)(?! )","end":"^(?!\\\\1|\\\\s*$)","name":"string.unquoted.block.yaml"}]},"block-sequence":{"match":"(-)(?!\\\\S)","name":"punctuation.definition.block.sequence.item.yaml"},"comment":{"begin":"(?:^([\\\\t ]*)|[\\\\t ]+)(?=#\\\\p{print}*$)","beginCaptures":{"1":{"name":"punctuation.whitespace.comment.leading.yaml"}},"end":"(?!\\\\G)","patterns":[{"begin":"#","beginCaptures":{"0":{"name":"punctuation.definition.comment.yaml"}},"end":"\\\\n","name":"comment.line.number-sign.yaml"}]},"directive":{"begin":"^%","beginCaptures":{"0":{"name":"punctuation.definition.directive.begin.yaml"}},"end":"(?=$|[\\\\t ]+($|#))","name":"meta.directive.yaml","patterns":[{"captures":{"1":{"name":"keyword.other.directive.yaml.yaml"},"2":{"name":"constant.numeric.yaml-version.yaml"}},"match":"\\\\G(YAML)[\\\\t ]+(\\\\d+\\\\.\\\\d+)"},{"captures":{"1":{"name":"keyword.other.directive.tag.yaml"},"2":{"name":"storage.type.tag-handle.yaml"},"3":{"name":"support.type.tag-prefix.yaml"}},"match":"\\\\G(TAG)(?:[\\\\t ]+(!(?:[-0-9A-Za-z]*!)?)(?:[\\\\t ]+(!(?:%\\\\h{2}|[]!#$\\\\&-;=?-\\\\[_a-z~])*|(?![]!,\\\\[{}])(?:%\\\\h{2}|[]!#$\\\\&-;=?-\\\\[_a-z~])+))?)?"},{"captures":{"1":{"name":"support.other.directive.reserved.yaml"},"2":{"name":"string.unquoted.directive-name.yaml"},"3":{"name":"string.unquoted.directive-parameter.yaml"}},"match":"\\\\G(\\\\w+)(?:[\\\\t ]+(\\\\w+)(?:[\\\\t ]+(\\\\w+))?)?"},{"match":"\\\\S+","name":"invalid.illegal.unrecognized.yaml"}]},"flow-alias":{"captures":{"1":{"name":"keyword.control.flow.alias.yaml"},"2":{"name":"punctuation.definition.alias.yaml"},"3":{"name":"variable.other.alias.yaml"},"4":{"name":"invalid.illegal.character.anchor.yaml"}},"match":"((\\\\*))([^],/\\\\[{}\\\\s]+)([^],}\\\\s]\\\\S*)?"},"flow-collection":{"patterns":[{"include":"#flow-sequence"},{"include":"#flow-mapping"}]},"flow-mapping":{"begin":"\\\\{","beginCaptures":{"0":{"name":"punctuation.definition.mapping.begin.yaml"}},"end":"}","endCaptures":{"0":{"name":"punctuation.definition.mapping.end.yaml"}},"name":"meta.flow-mapping.yaml","patterns":[{"include":"#prototype"},{"match":",","name":"punctuation.separator.mapping.yaml"},{"include":"#flow-pair"}]},"flow-node":{"patterns":[{"include":"#prototype"},{"include":"#flow-alias"},{"include":"#flow-collection"},{"include":"#flow-scalar"}]},"flow-pair":{"patterns":[{"begin":"\\\\?","beginCaptures":{"0":{"name":"punctuation.definition.key-value.begin.yaml"}},"end":"(?=[],}])","name":"meta.flow-pair.explicit.yaml","patterns":[{"include":"#prototype"},{"include":"#flow-pair"},{"include":"#flow-node"},{"begin":":(?=\\\\s|$|[],\\\\[{}])","beginCaptures":{"0":{"name":"punctuation.separator.key-value.mapping.yaml"}},"end":"(?=[],}])","patterns":[{"include":"#flow-value"}]}]},{"begin":"(?=(?:[^-\\\\]!\\"#%\\\\&'*,:>?@\\\\[\`{|}\\\\s]|[-:?][^],\\\\[{}\\\\s])([^],:\\\\[{}\\\\s]|:[^],\\\\[{}\\\\s]|\\\\s+(?![#\\\\s]))*\\\\s*:(\\\\s|$))","end":"(?=\\\\s*$|\\\\s+#|\\\\s*:(\\\\s|$)|\\\\s*:[],\\\\[{}]|\\\\s*[],\\\\[{}])","name":"meta.flow-pair.key.yaml","patterns":[{"include":"#flow-scalar-plain-in-implicit-type"},{"begin":"[^-\\\\]!\\"#%\\\\&'*,:>?@\\\\[\`{|}\\\\s]|[-:?][^],\\\\[{}\\\\s]","beginCaptures":{"0":{"name":"entity.name.tag.yaml"}},"contentName":"entity.name.tag.yaml","end":"(?=\\\\s*$|\\\\s+#|\\\\s*:(\\\\s|$)|\\\\s*:[],\\\\[{}]|\\\\s*[],\\\\[{}])","name":"string.unquoted.plain.in.yaml"}]},{"include":"#flow-node"},{"begin":":(?=\\\\s|$|[],\\\\[{}])","captures":{"0":{"name":"punctuation.separator.key-value.mapping.yaml"}},"end":"(?=[],}])","name":"meta.flow-pair.yaml","patterns":[{"include":"#flow-value"}]}]},"flow-scalar":{"patterns":[{"include":"#flow-scalar-double-quoted"},{"include":"#flow-scalar-single-quoted"},{"include":"#flow-scalar-plain-in"}]},"flow-scalar-double-quoted":{"begin":"\\"","beginCaptures":{"0":{"name":"punctuation.definition.string.begin.yaml"}},"end":"\\"","endCaptures":{"0":{"name":"punctuation.definition.string.end.yaml"}},"name":"string.quoted.double.yaml","patterns":[{"match":"\\\\\\\\([ \\"/0LN\\\\\\\\_abefnprtv]|x\\\\d\\\\d|u\\\\d{4}|U\\\\d{8})","name":"constant.character.escape.yaml"},{"match":"\\\\\\\\\\\\n","name":"constant.character.escape.double-quoted.newline.yaml"}]},"flow-scalar-plain-in":{"patterns":[{"include":"#flow-scalar-plain-in-implicit-type"},{"begin":"[^-\\\\]!\\"#%\\\\&'*,:>?@\\\\[\`{|}\\\\s]|[-:?][^],\\\\[{}\\\\s]","end":"(?=\\\\s*$|\\\\s+#|\\\\s*:(\\\\s|$)|\\\\s*:[],\\\\[{}]|\\\\s*[],\\\\[{}])","name":"string.unquoted.plain.in.yaml"}]},"flow-scalar-plain-in-implicit-type":{"patterns":[{"captures":{"1":{"name":"constant.language.null.yaml"},"2":{"name":"constant.language.boolean.yaml"},"3":{"name":"constant.numeric.integer.yaml"},"4":{"name":"constant.numeric.float.yaml"},"5":{"name":"constant.other.timestamp.yaml"},"6":{"name":"constant.language.value.yaml"},"7":{"name":"constant.language.merge.yaml"}},"match":"(?:(null|Null|NULL|~)|([Yy]|yes|Yes|YES|[Nn]|no|No|NO|true|True|TRUE|false|False|FALSE|on|On|ON|off|Off|OFF)|([-+]?0b[01_]+|[-+]?0[0-7_]+|[-+]?(?:0|[1-9][0-9_]*)|[-+]?0x[_\\\\h]+|[-+]?[1-9][0-9_]*(?::[0-5]?[0-9])+)|([-+]?(?:[0-9][0-9_]*)?\\\\.[.0-9]*(?:[Ee][-+][0-9]+)?|[-+]?[0-9][0-9_]*(?::[0-5]?[0-9])+\\\\.[0-9_]*|[-+]?\\\\.(?:inf|Inf|INF)|\\\\.(?:nan|NaN|NAN))|(\\\\d{4}-\\\\d{2}-\\\\d{2}|\\\\d{4}-\\\\d{1,2}-\\\\d{1,2}(?:[Tt]|[\\\\t ]+)\\\\d{1,2}:\\\\d{2}:\\\\d{2}(?:\\\\.\\\\d*)?(?:[\\\\t ]*Z|[-+]\\\\d{1,2}(?::\\\\d{1,2})?)?)|(=)|(<<))(?=\\\\s*$|\\\\s+#|\\\\s*:(\\\\s|$)|\\\\s*:[],\\\\[{}]|\\\\s*[],\\\\[{}])"}]},"flow-scalar-plain-out":{"patterns":[{"include":"#flow-scalar-plain-out-implicit-type"},{"begin":"[^-\\\\]!\\"#%\\\\&'*,:>?@\\\\[\`{|}\\\\s]|[-:?]\\\\S","end":"(?=\\\\s*$|\\\\s+#|\\\\s*:(\\\\s|$))","name":"string.unquoted.plain.out.yaml"}]},"flow-scalar-plain-out-implicit-type":{"patterns":[{"captures":{"1":{"name":"constant.language.null.yaml"},"2":{"name":"constant.language.boolean.yaml"},"3":{"name":"constant.numeric.integer.yaml"},"4":{"name":"constant.numeric.float.yaml"},"5":{"name":"constant.other.timestamp.yaml"},"6":{"name":"constant.language.value.yaml"},"7":{"name":"constant.language.merge.yaml"}},"match":"(?:(null|Null|NULL|~)|([Yy]|yes|Yes|YES|[Nn]|no|No|NO|true|True|TRUE|false|False|FALSE|on|On|ON|off|Off|OFF)|([-+]?0b[01_]+|[-+]?0[0-7_]+|[-+]?(?:0|[1-9][0-9_]*)|[-+]?0x[_\\\\h]+|[-+]?[1-9][0-9_]*(?::[0-5]?[0-9])+)|([-+]?(?:[0-9][0-9_]*)?\\\\.[.0-9]*(?:[Ee][-+][0-9]+)?|[-+]?[0-9][0-9_]*(?::[0-5]?[0-9])+\\\\.[0-9_]*|[-+]?\\\\.(?:inf|Inf|INF)|\\\\.(?:nan|NaN|NAN))|(\\\\d{4}-\\\\d{2}-\\\\d{2}|\\\\d{4}-\\\\d{1,2}-\\\\d{1,2}(?:[Tt]|[\\\\t ]+)\\\\d{1,2}:\\\\d{2}:\\\\d{2}(?:\\\\.\\\\d*)?(?:[\\\\t ]*Z|[-+]\\\\d{1,2}(?::\\\\d{1,2})?)?)|(=)|(<<))(?=\\\\s*$|\\\\s+#|\\\\s*:(\\\\s|$))"}]},"flow-scalar-single-quoted":{"begin":"'","beginCaptures":{"0":{"name":"punctuation.definition.string.begin.yaml"}},"end":"'(?!')","endCaptures":{"0":{"name":"punctuation.definition.string.end.yaml"}},"name":"string.quoted.single.yaml","patterns":[{"match":"''","name":"constant.character.escape.single-quoted.yaml"}]},"flow-sequence":{"begin":"\\\\[","beginCaptures":{"0":{"name":"punctuation.definition.sequence.begin.yaml"}},"end":"]","endCaptures":{"0":{"name":"punctuation.definition.sequence.end.yaml"}},"name":"meta.flow-sequence.yaml","patterns":[{"include":"#prototype"},{"match":",","name":"punctuation.separator.sequence.yaml"},{"include":"#flow-pair"},{"include":"#flow-node"}]},"flow-value":{"patterns":[{"begin":"\\\\G(?![],}])","end":"(?=[],}])","name":"meta.flow-pair.value.yaml","patterns":[{"include":"#flow-node"}]}]},"node":{"patterns":[{"include":"#block-node"}]},"property":{"begin":"(?=[!\\\\&])","end":"(?!\\\\G)","name":"meta.property.yaml","patterns":[{"captures":{"1":{"name":"keyword.control.property.anchor.yaml"},"2":{"name":"punctuation.definition.anchor.yaml"},"3":{"name":"entity.name.type.anchor.yaml"},"4":{"name":"invalid.illegal.character.anchor.yaml"}},"match":"\\\\G((&))([^],/\\\\[{}\\\\s]+)(\\\\S+)?"},{"match":"\\\\G!(?:<(?:%\\\\h{2}|[]!#$\\\\&-;=?-\\\\[_a-z~])+>|(?:[-0-9A-Za-z]*!)?(?:%\\\\h{2}|[#$\\\\&-+\\\\--;=?-Z_a-z~])+|)(?=[\\\\t ]|$)","name":"storage.type.tag-handle.yaml"},{"match":"\\\\S+","name":"invalid.illegal.tag-handle.yaml"}]},"prototype":{"patterns":[{"include":"#comment"},{"include":"#property"}]}},"scopeName":"source.yaml","aliases":["yml"]}`)),n=[e];export{n as default};
@@ -0,0 +1 @@
1
+ const e=Object.freeze(JSON.parse(`{"displayName":"ZenScript","fileTypes":["zs"],"name":"zenscript","patterns":[{"match":"\\\\b((0([Xx])\\\\h*)|(([0-9]+\\\\.?[0-9]*)|(\\\\.[0-9]+))(([Ee])([-+])?[0-9]+)?)([DFLUdflu]|UL|ul)?\\\\b","name":"constant.numeric.zenscript"},{"match":"\\\\b-?(0[BOXbox])(0|[1-9A-Fa-f][_\\\\h]*)[A-Z_a-z]*\\\\b","name":"constant.numeric.zenscript"},{"include":"#code"},{"match":"\\\\b((?:[a-z]\\\\w*\\\\.)*[A-Z]+\\\\w*)(?=\\\\[)","name":"storage.type.object.array.zenscript"}],"repository":{"brackets":{"patterns":[{"captures":{"1":{"name":"keyword.control.zenscript"},"2":{"name":"keyword.other.zenscript"},"3":{"name":"keyword.control.zenscript"},"4":{"name":"variable.other.zenscript"},"5":{"name":"keyword.control.zenscript"},"6":{"name":"constant.numeric.zenscript"},"7":{"name":"keyword.control.zenscript"}},"match":"(<)\\\\b(.*?)(:(.*?(:(\\\\*|\\\\d+)?)?)?)(>)","name":"keyword.other.zenscript"}]},"class":{"captures":{"1":{"name":"storage.type.zenscript"},"2":{"name":"entity.name.type.class.zenscript"}},"match":"(zenClass)\\\\s+(\\\\w+)","name":"meta.class.zenscript"},"code":{"patterns":[{"include":"#class"},{"include":"#functions"},{"include":"#dots"},{"include":"#quotes"},{"include":"#brackets"},{"include":"#comments"},{"include":"#var"},{"include":"#keywords"},{"include":"#constants"},{"include":"#operators"}]},"comments":{"patterns":[{"match":"//[^\\\\n]*","name":"comment.line.double=slash"},{"begin":"/\\\\*","beginCaptures":{"0":{"name":"comment.block"}},"end":"\\\\*/","endCaptures":{"0":{"name":"comment.block"}},"name":"comment.block"}]},"dots":{"captures":{"1":{"name":"storage.type.zenscript"},"2":{"name":"keyword.control.zenscript"},"5":{"name":"keyword.control.zenscript"}},"match":"\\\\b(\\\\w+)(\\\\.)(\\\\w+)((\\\\.)(\\\\w+))*","name":"plain.text.zenscript"},"functions":{"captures":{"0":{"name":"storage.type.function.zenscript"},"1":{"name":"entity.name.function.zenscript"}},"match":"function\\\\s+([$A-Z_a-z][$\\\\w]*)\\\\s*(?=\\\\()","name":"meta.function.zenscript"},"keywords":{"patterns":[{"match":"\\\\b(instanceof|get|implements|set|import|function|override|const|if|else|do|while|for|throw|panic|lock|try|catch|finally|return|break|continue|switch|case|default|in|is|as|match|throws|super|new)\\\\b","name":"keyword.control.zenscript"},{"match":"\\\\b(zenClass|zenConstructor|alias|class|interface|enum|struct|expand|variant|set|void|bool|byte|sbyte|short|ushort|int|uint|long|ulong|usize|float|double|char|string)\\\\b","name":"storage.type.zenscript"},{"match":"\\\\b(variant|abstract|final|private|public|export|internal|static|protected|implicit|virtual|extern|immutable)\\\\b","name":"storage.modifier.zenscript"},{"match":"\\\\b(Native|Precondition)\\\\b","name":"entity.other.attribute-name"},{"match":"\\\\b(null|true|false)\\\\b","name":"constant.language"}]},"operators":{"patterns":[{"match":"\\\\b(\\\\.\\\\.??|\\\\.\\\\.\\\\.|[+,]|\\\\+=|\\\\+\\\\+|-=??|--|~=??|\\\\*=??|/=??|%=??|\\\\|=??|\\\\|\\\\||&=??|&&|\\\\^=??|\\\\?\\\\.??|\\\\?\\\\?|<=??|<<=??|>=??|>>=??|>>>=??|=>?|===??|!=??|!==|[$\`])\\\\b","name":"keyword.control"},{"match":"\\\\b([:;])\\\\b","name":"keyword.control"}]},"quotes":{"patterns":[{"begin":"\\"","beginCaptures":{"0":{"name":"punctuation.definition.string.begin.zenscript"}},"end":"\\"","endCaptures":{"0":{"name":"punctuation.definition.string.end.zenscript"}},"name":"string.quoted.double.zenscript","patterns":[{"match":"\\\\\\\\.","name":"constant.character.escape.zenscript"}]},{"begin":"'","beginCaptures":{"0":{"name":"punctuation.definition.string.begin.zenscript"}},"end":"'","endCaptures":{"0":{"name":"punctuation.definition.string.end.zenscript"}},"name":"string.quoted.single.zenscript","patterns":[{"match":"\\\\\\\\.","name":"constant.character.escape.zenscript"}]}]},"var":{"match":"\\\\b(va[lr])\\\\b","name":"storage.type"}},"scopeName":"source.zenscript"}`)),t=[e];export{t as default};
@@ -0,0 +1 @@
1
+ const e=Object.freeze(JSON.parse(`{"displayName":"Zig","fileTypes":["zig","zon"],"name":"zig","patterns":[{"include":"#comments"},{"include":"#strings"},{"include":"#keywords"},{"include":"#operators"},{"include":"#punctuation"},{"include":"#numbers"},{"include":"#support"},{"include":"#variables"}],"repository":{"commentContents":{"patterns":[{"match":"\\\\b(TODO|FIXME|XXX|NOTE)\\\\b:?","name":"keyword.todo.zig"}]},"comments":{"patterns":[{"begin":"//[!/](?=[^/])","end":"$","name":"comment.line.documentation.zig","patterns":[{"include":"#commentContents"}]},{"begin":"//","end":"$","name":"comment.line.double-slash.zig","patterns":[{"include":"#commentContents"}]}]},"keywords":{"patterns":[{"match":"\\\\binline\\\\b(?!\\\\s*\\\\bfn\\\\b)","name":"keyword.control.repeat.zig"},{"match":"\\\\b(while|for)\\\\b","name":"keyword.control.repeat.zig"},{"match":"\\\\b(extern|packed|export|pub|noalias|inline|comptime|volatile|align|linksection|threadlocal|allowzero|noinline|callconv)\\\\b","name":"keyword.storage.zig"},{"match":"\\\\b(struct|enum|union|opaque)\\\\b","name":"keyword.structure.zig"},{"match":"\\\\b(asm|unreachable)\\\\b","name":"keyword.statement.zig"},{"match":"\\\\b(break|return|continue|defer|errdefer)\\\\b","name":"keyword.control.flow.zig"},{"match":"\\\\b(resume|suspend|nosuspend)\\\\b","name":"keyword.control.async.zig"},{"match":"\\\\b(try|catch)\\\\b","name":"keyword.control.trycatch.zig"},{"match":"\\\\b(if|else|switch|orelse)\\\\b","name":"keyword.control.conditional.zig"},{"match":"\\\\b(null|undefined)\\\\b","name":"keyword.constant.default.zig"},{"match":"\\\\b(true|false)\\\\b","name":"keyword.constant.bool.zig"},{"match":"\\\\b(test|and|or)\\\\b","name":"keyword.default.zig"},{"match":"\\\\b(bool|void|noreturn|type|error|anyerror|anyframe|anytype|anyopaque)\\\\b","name":"keyword.type.zig"},{"match":"\\\\b(f16|f32|f64|f80|f128|u\\\\d+|i\\\\d+|isize|usize|comptime_int|comptime_float)\\\\b","name":"keyword.type.integer.zig"},{"match":"\\\\b(c_(?:char|short|ushort|int|uint|long|ulong|longlong|ulonglong|longdouble))\\\\b","name":"keyword.type.c.zig"}]},"numbers":{"patterns":[{"match":"\\\\b0x\\\\h[_\\\\h]*(\\\\.\\\\h[_\\\\h]*)?([Pp][-+]?[_\\\\h]+)?\\\\b","name":"constant.numeric.hexfloat.zig"},{"match":"\\\\b[0-9][0-9_]*(\\\\.[0-9][0-9_]*)?([Ee][-+]?[0-9_]+)?\\\\b","name":"constant.numeric.float.zig"},{"match":"\\\\b[0-9][0-9_]*\\\\b","name":"constant.numeric.decimal.zig"},{"match":"\\\\b0x[_\\\\h]+\\\\b","name":"constant.numeric.hexadecimal.zig"},{"match":"\\\\b0o[0-7_]+\\\\b","name":"constant.numeric.octal.zig"},{"match":"\\\\b0b[01_]+\\\\b","name":"constant.numeric.binary.zig"},{"match":"\\\\b[0-9](([EPep][-+])|[0-9A-Z_a-z])*(\\\\.(([EPep][-+])|[0-9A-Z_a-z])*)?([EPep][-+])?[0-9A-Z_a-z]*\\\\b","name":"constant.numeric.invalid.zig"}]},"operators":{"patterns":[{"match":"(?<=\\\\[)\\\\*c(?=])","name":"keyword.operator.c-pointer.zig"},{"match":"\\\\b((and|or))\\\\b|(==|!=|<=|>=|[<>])","name":"keyword.operator.comparison.zig"},{"match":"(-%?|\\\\+%?|\\\\*%?|[%/])=?","name":"keyword.operator.arithmetic.zig"},{"match":"(<<%?|>>|[!\\\\&^|~])=?","name":"keyword.operator.bitwise.zig"},{"match":"(==|\\\\+\\\\+|\\\\*\\\\*|->)","name":"keyword.operator.special.zig"},{"match":"=","name":"keyword.operator.assignment.zig"},{"match":"\\\\?","name":"keyword.operator.question.zig"}]},"punctuation":{"patterns":[{"match":"\\\\.","name":"punctuation.accessor.zig"},{"match":",","name":"punctuation.comma.zig"},{"match":":","name":"punctuation.separator.key-value.zig"},{"match":";","name":"punctuation.terminator.statement.zig"}]},"stringcontent":{"patterns":[{"match":"\\\\\\\\([\\"'\\\\\\\\nrt]|(x\\\\h{2})|(u\\\\{\\\\h+}))","name":"constant.character.escape.zig"},{"match":"\\\\\\\\.","name":"invalid.illegal.unrecognized-string-escape.zig"}]},"strings":{"patterns":[{"begin":"\\"","end":"\\"","name":"string.quoted.double.zig","patterns":[{"include":"#stringcontent"}]},{"begin":"\\\\\\\\\\\\\\\\","end":"$","name":"string.multiline.zig"},{"match":"'([^'\\\\\\\\]|\\\\\\\\(x\\\\h{2}|[012][0-7]{0,2}|3[0-6][0-7]?|37[0-7]?|[4-7][0-7]?|.))'","name":"string.quoted.single.zig"}]},"support":{"patterns":[{"match":"@[A-Z_a-z][0-9A-Z_a-z]*","name":"support.function.builtin.zig"}]},"variables":{"patterns":[{"name":"meta.function.declaration.zig","patterns":[{"captures":{"1":{"name":"storage.type.function.zig"},"2":{"name":"entity.name.type.zig"}},"match":"\\\\b(fn)\\\\s+([A-Z][0-9A-Za-z]*)\\\\b"},{"captures":{"1":{"name":"storage.type.function.zig"},"2":{"name":"entity.name.function.zig"}},"match":"\\\\b(fn)\\\\s+([A-Z_a-z][0-9A-Z_a-z]*)\\\\b"},{"begin":"\\\\b(fn)\\\\s+@\\"","beginCaptures":{"1":{"name":"storage.type.function.zig"}},"end":"\\"","name":"entity.name.function.string.zig","patterns":[{"include":"#stringcontent"}]},{"match":"\\\\b(const|var|fn)\\\\b","name":"keyword.default.zig"}]},{"name":"meta.function.call.zig","patterns":[{"match":"([A-Z][0-9A-Za-z]*)(?=\\\\s*\\\\()","name":"entity.name.type.zig"},{"match":"([A-Z_a-z][0-9A-Z_a-z]*)(?=\\\\s*\\\\()","name":"entity.name.function.zig"}]},{"name":"meta.variable.zig","patterns":[{"match":"\\\\b[A-Z_a-z][0-9A-Z_a-z]*\\\\b","name":"variable.zig"},{"begin":"@\\"","end":"\\"","name":"variable.string.zig","patterns":[{"include":"#stringcontent"}]}]}]}},"scopeName":"source.zig"}`)),n=[e];export{n as default};
package/dist/index.html CHANGED
@@ -5,8 +5,8 @@
5
5
  <link rel="icon" type="image/svg+xml" href="/favicon.svg" />
6
6
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
7
7
  <title>TreeLab - Git Worktree Manager</title>
8
- <script type="module" crossorigin src="/assets/index-C78PMV-C.js"></script>
9
- <link rel="stylesheet" crossorigin href="/assets/index-CR9jga1C.css">
8
+ <script type="module" crossorigin src="/assets/index-sP_n0D3A.js"></script>
9
+ <link rel="stylesheet" crossorigin href="/assets/index-bLnnvz_q.css">
10
10
  </head>
11
11
  <body>
12
12
  <div id="root"></div>
@@ -1,6 +1,6 @@
1
1
  import express from 'express';
2
2
  import path from 'node:path';
3
- import { getGitDirAbsolute } from '../core/git.js';
3
+ import { getGitDirAbsolute, git } from '../core/git.js';
4
4
  import { listWorktrees } from '../core/worktree.js';
5
5
  import { gitOrThrow } from '../core/git.js';
6
6
  import { pathFromId } from '../core/id.js';
@@ -159,6 +159,45 @@ export function createWorktreeRouter(getRepoRoot) {
159
159
  });
160
160
  }
161
161
  });
162
+ router.get('/worktrees/:id/staged', (req, res) => {
163
+ try {
164
+ const wtPath = pathFromId(req.params.id);
165
+ const parseNameStatus = (stdout) => stdout
166
+ .split('\n')
167
+ .map((line) => line.trim())
168
+ .filter(Boolean)
169
+ .map((line) => {
170
+ const tab = line.indexOf('\t');
171
+ if (tab === -1)
172
+ return { status: '?', path: line };
173
+ return { status: line.slice(0, tab).trim(), path: line.slice(tab + 1).trim() };
174
+ });
175
+ const stagedStatus = git(wtPath, ['diff', '--cached', '--name-status']);
176
+ const stagedFiles = stagedStatus.ok ? parseNameStatus(stagedStatus.stdout) : [];
177
+ const stagedDiffResult = git(wtPath, ['diff', '--cached']);
178
+ const stagedDiff = stagedDiffResult.ok ? stagedDiffResult.stdout : '';
179
+ const unstagedStatus = git(wtPath, ['diff', '--name-status']);
180
+ const unstagedFiles = unstagedStatus.ok ? parseNameStatus(unstagedStatus.stdout) : [];
181
+ const unstagedDiffResult = git(wtPath, ['diff']);
182
+ const unstagedDiff = unstagedDiffResult.ok ? unstagedDiffResult.stdout : '';
183
+ const graphResult = git(wtPath, ['log', '--graph', '--abbrev-commit', '--format=%h %s (%an, %ar)', '-20']);
184
+ const commitGraph = graphResult.ok ? graphResult.stdout : '';
185
+ res.json({
186
+ ok: true,
187
+ data: {
188
+ staged: { files: stagedFiles, diff: stagedDiff },
189
+ unstaged: { files: unstagedFiles, diff: unstagedDiff },
190
+ commitGraph,
191
+ },
192
+ });
193
+ }
194
+ catch (e) {
195
+ res.status(500).json({
196
+ ok: false,
197
+ error: { code: 'DIFF_FAILED', message: errMsg(e) || 'Failed to get diff' },
198
+ });
199
+ }
200
+ });
162
201
  router.post('/worktrees/prune', (req, res) => {
163
202
  try {
164
203
  const root = getRepoRoot();
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@fatdoge/wtree",
3
3
  "private": false,
4
- "version": "0.2.1",
4
+ "version": "0.3.0",
5
5
  "description": "CLI + UI tool for managing git worktrees",
6
6
  "keywords": [
7
7
  "git",
@@ -52,6 +52,8 @@
52
52
  "test": "vitest run"
53
53
  },
54
54
  "dependencies": {
55
+ "@pierre/diffs": "^1.1.21",
56
+ "@pierre/trees": "1.0.0-beta.3",
55
57
  "@vitejs/plugin-react": "^4.4.1",
56
58
  "autoprefixer": "^10.4.21",
57
59
  "chalk": "^5.6.0",
@@ -25,6 +25,26 @@ export type WtuiConfig = {
25
25
  editorCommand?: string
26
26
  }
27
27
 
28
+ export type FileChange = {
29
+ status: string
30
+ path: string
31
+ }
32
+
33
+ export type CommitInfo = {
34
+ hash: string
35
+ shortHash: string
36
+ message: string
37
+ author: string
38
+ date: string
39
+ parents: string[]
40
+ }
41
+
42
+ export type WorktreeDiffInfo = {
43
+ staged: { files: FileChange[]; diff: string }
44
+ unstaged: { files: FileChange[]; diff: string }
45
+ commitGraph: string
46
+ }
47
+
28
48
  export type ApiError = {
29
49
  code: string
30
50
  message: string
@@ -9,9 +9,28 @@ user-invocable: true
9
9
 
10
10
  You manage git worktrees using the `wtree` CLI tool. Always use non-interactive flags so commands complete without user input.
11
11
 
12
- ## Prerequisites
12
+ ## Environment Setup (MUST do first)
13
13
 
14
- The `wtree` CLI must be installed globally (`npm install -g @fatdoge/wtree`) or run via `npx @fatdoge/wtree`. If working inside the wtree project itself, use `node dist-node/api/cli/wtree.js`.
14
+ Before running any wtree command, you MUST check if wtree is installed locally. Run this check once at the start:
15
+
16
+ ```bash
17
+ wtree --version 2>/dev/null
18
+ ```
19
+
20
+ - **If it outputs a version number** (e.g., `0.2.1`): wtree is installed, proceed with commands directly using `wtree`.
21
+ - **If it fails or outputs nothing**: wtree is NOT installed. Install it globally first:
22
+
23
+ ```bash
24
+ npm install -g @fatdoge/wtree --registry=https://registry.npmjs.org/
25
+ ```
26
+
27
+ Then verify the installation:
28
+
29
+ ```bash
30
+ wtree --version
31
+ ```
32
+
33
+ **Do NOT use `npx` to run wtree** — it is significantly slower. Always ensure wtree is installed globally first.
15
34
 
16
35
  ## Important Rules
17
36
 
@@ -0,0 +1,381 @@
1
+ import { PatchDiff } from '@pierre/diffs/react'
2
+ import { FileTree, useFileTree } from '@pierre/trees/react'
3
+ import { useTranslation } from 'react-i18next'
4
+ import { useMemo, useState, useCallback, useEffect, useRef } from 'react'
5
+ import { Copy, GitCommitHorizontal, ChevronDown, ChevronRight } from 'lucide-react'
6
+ import { toast } from 'sonner'
7
+ import Modal from './Modal'
8
+ import Button from './Button'
9
+ import { useThemeStore } from '../stores/themeStore'
10
+ import type { FileChange, WorktreeDiffInfo } from '../../shared/wtui-types'
11
+
12
+ type Props = {
13
+ open: boolean
14
+ onClose: () => void
15
+ worktreePath: string
16
+ diffInfo: WorktreeDiffInfo | null
17
+ loading?: boolean
18
+ }
19
+
20
+ function buildPatchMap(patch: string): Map<string, string> {
21
+ const map = new Map<string, string>()
22
+ if (!patch) return map
23
+ const lines = patch.split('\n')
24
+ let current: string[] = []
25
+ let currentPath = ''
26
+ const flush = () => {
27
+ if (currentPath && current.length > 0) map.set(currentPath, current.join('\n'))
28
+ current = []
29
+ currentPath = ''
30
+ }
31
+ for (const line of lines) {
32
+ if (line.startsWith('diff --git ')) {
33
+ flush()
34
+ const match = line.match(/^diff --git a\/.+ b\/(.+)$/)
35
+ currentPath = match?.[1] ?? ''
36
+ }
37
+ current.push(line)
38
+ }
39
+ flush()
40
+ return map
41
+ }
42
+
43
+ const GIT_STATUS_MAP: Record<string, 'added' | 'modified' | 'deleted' | 'renamed' | 'untracked'> = {
44
+ A: 'added', M: 'modified', D: 'deleted', R: 'renamed', C: 'added', U: 'untracked', '?': 'untracked',
45
+ }
46
+
47
+ const TREE_STYLES_LIGHT = {
48
+ colorScheme: 'light' as const,
49
+ '--trees-theme-sidebar-bg': '#ffffff',
50
+ '--trees-theme-sidebar-fg': '#1e293b',
51
+ '--trees-theme-sidebar-border': '#e2e8f0',
52
+ '--trees-theme-list-hover-bg': '#f8fafc',
53
+ '--trees-theme-list-active-selection-bg': '#eff6ff',
54
+ '--trees-theme-list-active-selection-fg': '#1e293b',
55
+ '--trees-theme-git-added-fg': '#16a34a',
56
+ '--trees-theme-git-modified-fg': '#2563eb',
57
+ '--trees-theme-git-deleted-fg': '#dc2626',
58
+ }
59
+
60
+ const TREE_STYLES_DARK = {
61
+ colorScheme: 'dark' as const,
62
+ '--trees-theme-sidebar-bg': '#0f172a',
63
+ '--trees-theme-sidebar-fg': '#e2e8f0',
64
+ '--trees-theme-sidebar-border': '#1e293b',
65
+ '--trees-theme-list-hover-bg': 'rgba(15,23,42,0.4)',
66
+ '--trees-theme-list-active-selection-bg': 'rgba(30,58,138,0.3)',
67
+ '--trees-theme-list-active-selection-fg': '#e2e8f0',
68
+ '--trees-theme-git-added-fg': '#4ade80',
69
+ '--trees-theme-git-modified-fg': '#60a5fa',
70
+ '--trees-theme-git-deleted-fg': '#f87171',
71
+ }
72
+
73
+ type SelectedFile = { path: string; group: 'staged' | 'unstaged' }
74
+
75
+ /* ── Collapsible section wrapper ── */
76
+
77
+ function CollapsibleSection({
78
+ title,
79
+ titleColor,
80
+ icon,
81
+ count,
82
+ defaultOpen = true,
83
+ children,
84
+ }: {
85
+ title: string
86
+ titleColor: string
87
+ icon?: React.ReactNode
88
+ count?: number
89
+ defaultOpen?: boolean
90
+ children: React.ReactNode
91
+ }) {
92
+ const [open, setOpen] = useState(defaultOpen)
93
+ return (
94
+ <div>
95
+ <button
96
+ onClick={() => setOpen(!open)}
97
+ className={`w-full flex items-center gap-1 px-2 py-1 text-[11px] font-semibold uppercase tracking-wide ${titleColor} hover:bg-slate-50 dark:hover:bg-slate-900/40 shrink-0`}
98
+ >
99
+ {open ? <ChevronDown className="h-3 w-3" /> : <ChevronRight className="h-3 w-3" />}
100
+ {icon}
101
+ {title}
102
+ {count != null && <span className="ml-auto font-normal text-slate-400 dark:text-slate-500 normal-case">{count}</span>}
103
+ </button>
104
+ {open && children}
105
+ </div>
106
+ )
107
+ }
108
+
109
+ /* ── File tree section ── */
110
+
111
+ function SidebarTree({
112
+ files,
113
+ isDark,
114
+ onSelect,
115
+ group,
116
+ }: {
117
+ files: FileChange[]
118
+ isDark: boolean
119
+ onSelect: (file: SelectedFile) => void
120
+ group: 'staged' | 'unstaged'
121
+ }) {
122
+ const paths = useMemo(() => files.map((f) => f.path), [files])
123
+ const gitStatus = useMemo(
124
+ () => files.map((f) => ({
125
+ path: f.path,
126
+ status: GIT_STATUS_MAP[f.status[0]?.toUpperCase()] ?? ('modified' as const),
127
+ })),
128
+ [files],
129
+ )
130
+
131
+ const { model } = useFileTree({
132
+ paths,
133
+ gitStatus,
134
+ initialExpansion: 'open',
135
+ flattenEmptyDirectories: true,
136
+ density: 'compact',
137
+ icons: { set: 'standard', colored: true },
138
+ initialVisibleRowCount: paths.length + 10,
139
+ onSelectionChange: useCallback(
140
+ (selected: string[]) => {
141
+ const filePath = selected[0]
142
+ if (filePath && files.some((f) => f.path === filePath)) {
143
+ onSelect({ path: filePath, group })
144
+ }
145
+ },
146
+ [files, onSelect, group],
147
+ ),
148
+ })
149
+
150
+ // compact density itemHeight=24. Count files + unique parent dirs for row estimate.
151
+ const dirCount = new Set(files.map((f) => f.path.substring(0, f.path.lastIndexOf('/'))).filter(Boolean)).size
152
+ const treeHeight = Math.max((files.length + dirCount) * 24, 48)
153
+
154
+ return (
155
+ <FileTree
156
+ model={model}
157
+ style={{
158
+ height: `${treeHeight}px`,
159
+ ...(isDark ? TREE_STYLES_DARK : TREE_STYLES_LIGHT),
160
+ }}
161
+ />
162
+ )
163
+ }
164
+
165
+ /* ── Commit graph (ASCII from git log --graph) ── */
166
+
167
+ function CommitGraphView({ graphText }: { graphText: string }) {
168
+ if (!graphText) return null
169
+ const lines = graphText.split('\n').filter((l) => l.length > 0)
170
+ return (
171
+ <pre className="font-mono text-[11px] leading-5 px-1 py-1 text-slate-700 dark:text-slate-300 whitespace-pre overflow-x-auto">
172
+ {lines.map((line, i) => {
173
+ // Split line into graph prefix (*/|/\ chars) and commit text
174
+ const match = line.match(/^([* |/\\]+?)(\s[a-f0-9]{7,}.*)$/i)
175
+ if (match) {
176
+ return (
177
+ <div key={i}>
178
+ <span className="text-blue-400 dark:text-blue-500">{match[1]}</span>
179
+ <span>{match[2]}</span>
180
+ </div>
181
+ )
182
+ }
183
+ // Pure graph lines (like |\ or |/)
184
+ return (
185
+ <div key={i} className="text-blue-400 dark:text-blue-500">{line}</div>
186
+ )
187
+ })}
188
+ </pre>
189
+ )
190
+ }
191
+
192
+ /* ── Draggable vertical divider ── */
193
+
194
+ function useDragResize(initialWidth: number, minWidth: number, maxWidth: number) {
195
+ const [width, setWidth] = useState(initialWidth)
196
+ const dragging = useRef(false)
197
+ const startX = useRef(0)
198
+ const startW = useRef(0)
199
+
200
+ const onMouseDown = useCallback((e: React.MouseEvent) => {
201
+ e.preventDefault()
202
+ dragging.current = true
203
+ startX.current = e.clientX
204
+ startW.current = width
205
+
206
+ const onMouseMove = (ev: MouseEvent) => {
207
+ if (!dragging.current) return
208
+ const delta = ev.clientX - startX.current
209
+ setWidth(Math.max(minWidth, Math.min(maxWidth, startW.current + delta)))
210
+ }
211
+ const onMouseUp = () => {
212
+ dragging.current = false
213
+ document.removeEventListener('mousemove', onMouseMove)
214
+ document.removeEventListener('mouseup', onMouseUp)
215
+ document.body.style.cursor = ''
216
+ document.body.style.userSelect = ''
217
+ }
218
+ document.body.style.cursor = 'col-resize'
219
+ document.body.style.userSelect = 'none'
220
+ document.addEventListener('mousemove', onMouseMove)
221
+ document.addEventListener('mouseup', onMouseUp)
222
+ }, [width, minWidth, maxWidth])
223
+
224
+ return { width, onMouseDown }
225
+ }
226
+
227
+ /* ── Main component ── */
228
+
229
+ export default function DiffPreviewModal({ open, onClose, worktreePath, diffInfo, loading }: Props) {
230
+ const { t } = useTranslation()
231
+ const theme = useThemeStore((s) => s.theme)
232
+ const [selected, setSelected] = useState<SelectedFile | null>(null)
233
+ const { width: sidebarWidth, onMouseDown: onDividerMouseDown } = useDragResize(260, 160, 640)
234
+
235
+ const resolvedThemeType = theme === 'system' ? 'system' : theme === 'dark' ? 'dark' : 'light'
236
+ const isDark =
237
+ theme === 'dark' ||
238
+ (theme === 'system' && typeof window !== 'undefined' && window.matchMedia('(prefers-color-scheme: dark)').matches)
239
+
240
+ const staged = diffInfo?.staged
241
+ const unstaged = diffInfo?.unstaged
242
+ const commitGraph = diffInfo?.commitGraph ?? ''
243
+ const allDiff = [staged?.diff, unstaged?.diff].filter(Boolean).join('\n')
244
+ const hasChanges = (staged?.files.length ?? 0) > 0 || (unstaged?.files.length ?? 0) > 0
245
+
246
+ const stagedPatchMap = useMemo(() => buildPatchMap(staged?.diff ?? ''), [staged?.diff])
247
+ const unstagedPatchMap = useMemo(() => buildPatchMap(unstaged?.diff ?? ''), [unstaged?.diff])
248
+
249
+ useEffect(() => {
250
+ if (!diffInfo) { setSelected(null); return }
251
+ const first = diffInfo.staged.files[0] ?? diffInfo.unstaged.files[0]
252
+ if (first) {
253
+ setSelected({ path: first.path, group: diffInfo.staged.files[0] ? 'staged' : 'unstaged' })
254
+ } else {
255
+ setSelected(null)
256
+ }
257
+ }, [diffInfo])
258
+
259
+ const selectedPatch = useMemo(() => {
260
+ if (!selected) return null
261
+ const map = selected.group === 'staged' ? stagedPatchMap : unstagedPatchMap
262
+ return map.get(selected.path) ?? null
263
+ }, [selected, stagedPatchMap, unstagedPatchMap])
264
+
265
+ const handleCopy = async () => {
266
+ try {
267
+ await navigator.clipboard.writeText(selectedPatch || allDiff)
268
+ toast.success(t('diff.toast.copied'))
269
+ } catch {
270
+ toast.error(t('diff.toast.copyFailed'))
271
+ }
272
+ }
273
+
274
+ return (
275
+ <Modal
276
+ title={t('diff.title')}
277
+ open={open}
278
+ onClose={onClose}
279
+ size="full"
280
+ footer={
281
+ <>
282
+ <Button variant="secondary" size="sm" onClick={handleCopy} disabled={!allDiff}>
283
+ <Copy className="h-4 w-4" />
284
+ {t('diff.copyDiff')}
285
+ </Button>
286
+ <Button variant="secondary" size="sm" onClick={onClose}>
287
+ {t('diff.close')}
288
+ </Button>
289
+ </>
290
+ }
291
+ >
292
+ <div className="max-h-[75vh]">
293
+ <div className="text-xs font-mono text-slate-500 dark:text-slate-400 truncate mb-2">{worktreePath}</div>
294
+
295
+ {loading ? (
296
+ <div className="space-y-2">
297
+ <div className="h-5 animate-pulse rounded bg-slate-100 dark:bg-slate-900" />
298
+ <div className="h-5 animate-pulse rounded bg-slate-100 dark:bg-slate-900 w-3/4" />
299
+ </div>
300
+ ) : !hasChanges && !commitGraph ? (
301
+ <div className="text-sm text-slate-500 dark:text-slate-400">{t('diff.noChanges')}</div>
302
+ ) : (
303
+ <div className="flex rounded-lg border border-slate-200 dark:border-slate-800 overflow-hidden" style={{ height: 'calc(75vh - 40px)' }}>
304
+ {/* Left sidebar */}
305
+ <div className="shrink-0 flex flex-col bg-white dark:bg-slate-950" style={{ width: `${sidebarWidth}px` }}>
306
+ {/* File trees + commits — scrollable as a whole */}
307
+ <div className="flex-1 overflow-y-auto min-h-0">
308
+ {staged && staged.files.length > 0 && (
309
+ <CollapsibleSection
310
+ title={t('diff.stagedFiles')}
311
+ titleColor="text-emerald-600 dark:text-emerald-400"
312
+ count={staged.files.length}
313
+ >
314
+ <SidebarTree files={staged.files} isDark={isDark} onSelect={setSelected} group="staged" />
315
+ </CollapsibleSection>
316
+ )}
317
+
318
+ {unstaged && unstaged.files.length > 0 && (
319
+ <>
320
+ {staged && staged.files.length > 0 && <div className="border-t border-slate-200 dark:border-slate-800" />}
321
+ <CollapsibleSection
322
+ title={t('diff.unstagedFiles')}
323
+ titleColor="text-amber-600 dark:text-amber-400"
324
+ count={unstaged.files.length}
325
+ >
326
+ <SidebarTree files={unstaged.files} isDark={isDark} onSelect={setSelected} group="unstaged" />
327
+ </CollapsibleSection>
328
+ </>
329
+ )}
330
+ </div>
331
+
332
+ {/* Commits — pinned to bottom */}
333
+ {commitGraph && (
334
+ <div className="shrink-0 border-t border-slate-200 dark:border-slate-800">
335
+ <CollapsibleSection
336
+ title={t('diff.commits')}
337
+ titleColor="text-slate-500 dark:text-slate-400"
338
+ icon={<GitCommitHorizontal className="h-3 w-3" />}
339
+ defaultOpen={!hasChanges}
340
+ >
341
+ <div className="max-h-64 overflow-auto">
342
+ <CommitGraphView graphText={commitGraph} />
343
+ </div>
344
+ </CollapsibleSection>
345
+ </div>
346
+ )}
347
+ </div>
348
+
349
+ {/* Draggable divider */}
350
+ <div
351
+ className="w-1 shrink-0 cursor-col-resize bg-slate-200 dark:bg-slate-800 hover:bg-blue-400 dark:hover:bg-blue-500 transition-colors"
352
+ onMouseDown={onDividerMouseDown}
353
+ />
354
+
355
+ {/* Right panel — diff viewer */}
356
+ <div className="flex-1 min-w-0 overflow-auto bg-white dark:bg-slate-950">
357
+ {selectedPatch ? (
358
+ <PatchDiff
359
+ patch={selectedPatch}
360
+ disableWorkerPool
361
+ options={{
362
+ theme: { dark: 'github-dark', light: 'github-light' },
363
+ themeType: resolvedThemeType,
364
+ }}
365
+ />
366
+ ) : selected ? (
367
+ <div className="flex items-center justify-center h-full text-sm text-slate-400 dark:text-slate-500">
368
+ {t('diff.noDiffContent')}
369
+ </div>
370
+ ) : (
371
+ <div className="flex items-center justify-center h-full text-sm text-slate-400 dark:text-slate-500">
372
+ {t('diff.selectFile')}
373
+ </div>
374
+ )}
375
+ </div>
376
+ </div>
377
+ )}
378
+ </div>
379
+ </Modal>
380
+ )
381
+ }
@@ -7,14 +7,23 @@ interface Props {
7
7
  title: string
8
8
  children: ReactNode
9
9
  footer?: ReactNode
10
+ size?: 'md' | 'lg' | 'xl' | '2xl' | 'full'
10
11
  }
11
12
 
12
- export default function Modal({ open, onClose, title, children, footer }: Props) {
13
+ const SIZE_CLS: Record<string, string> = {
14
+ md: 'max-w-md',
15
+ lg: 'max-w-2xl',
16
+ xl: 'max-w-4xl',
17
+ '2xl': 'max-w-6xl',
18
+ full: 'max-w-[calc(100vw-2rem)]',
19
+ }
20
+
21
+ export default function Modal({ open, onClose, title, children, footer, size = 'md' }: Props) {
13
22
  if (!open) return null
14
23
 
15
24
  return (
16
25
  <div className="fixed inset-0 z-50 flex items-center justify-center bg-slate-900/50 p-4 backdrop-blur-sm dark:bg-black/50">
17
- <div className="w-full max-w-md rounded-xl border border-slate-200 bg-white shadow-xl dark:border-slate-800 dark:bg-slate-950">
26
+ <div className={`w-full ${SIZE_CLS[size]} rounded-xl border border-slate-200 bg-white shadow-xl dark:border-slate-800 dark:bg-slate-950`}>
18
27
  <div className="flex items-center justify-between border-b border-slate-100 px-4 py-3 dark:border-slate-800">
19
28
  <div className="font-semibold text-slate-900 dark:text-slate-100">{title}</div>
20
29
  <button onClick={onClose} className="text-slate-400 hover:text-slate-600 dark:hover:text-slate-200">