@rsktash/beads-ui 0.1.1 → 0.1.3

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 (304) hide show
  1. package/dist/assets/abap-DsBKuouk.js +1 -0
  2. package/dist/assets/actionscript-3-D_z4Izcz.js +1 -0
  3. package/dist/assets/ada-727ZlQH0.js +1 -0
  4. package/dist/assets/andromeeda-C3khCPGq.js +1 -0
  5. package/dist/assets/angular-html-LfdN0zeE.js +1 -0
  6. package/dist/assets/angular-ts-CKsD7JZE.js +1 -0
  7. package/dist/assets/apache-Dn00JSTd.js +1 -0
  8. package/dist/assets/apex-COJ4H7py.js +1 -0
  9. package/dist/assets/apl-BBq3IX1j.js +1 -0
  10. package/dist/assets/applescript-Bu5BbsvL.js +1 -0
  11. package/dist/assets/ara-7O62HKoU.js +1 -0
  12. package/dist/assets/asciidoc-BPT9niGB.js +1 -0
  13. package/dist/assets/asm-Dhn9LcZ4.js +1 -0
  14. package/dist/assets/astro-CqkE3fuf.js +1 -0
  15. package/dist/assets/aurora-x-D-2ljcwZ.js +1 -0
  16. package/dist/assets/awk-eg146-Ew.js +1 -0
  17. package/dist/assets/ayu-dark-Cv9koXgw.js +1 -0
  18. package/dist/assets/ballerina-Du268qiB.js +1 -0
  19. package/dist/assets/bat-fje9CFhw.js +1 -0
  20. package/dist/assets/beancount-BwXTMy5W.js +1 -0
  21. package/dist/assets/berry-3xVqZejG.js +1 -0
  22. package/dist/assets/bibtex-xW4inM5L.js +1 -0
  23. package/dist/assets/bicep-DHo0CJ0O.js +1 -0
  24. package/dist/assets/blade-a8OxSdnT.js +1 -0
  25. package/dist/assets/bsl-Dgyn0ogV.js +1 -0
  26. package/dist/assets/c-C3t2pwGQ.js +1 -0
  27. package/dist/assets/cadence-DNquZEk8.js +1 -0
  28. package/dist/assets/cairo--RitsXJZ.js +1 -0
  29. package/dist/assets/catppuccin-frappe-CD_QflpE.js +1 -0
  30. package/dist/assets/catppuccin-latte-DRW-0cLl.js +1 -0
  31. package/dist/assets/catppuccin-macchiato-C-_shW-Y.js +1 -0
  32. package/dist/assets/catppuccin-mocha-LGGdnPYs.js +1 -0
  33. package/dist/assets/clarity-BHOwM8T6.js +1 -0
  34. package/dist/assets/clojure-DxSadP1t.js +1 -0
  35. package/dist/assets/cmake-DbXoA79R.js +1 -0
  36. package/dist/assets/cobol-PTqiYgYu.js +1 -0
  37. package/dist/assets/codeowners-Bp6g37R7.js +1 -0
  38. package/dist/assets/codeql-sacFqUAJ.js +1 -0
  39. package/dist/assets/coffee-dyiR41kL.js +1 -0
  40. package/dist/assets/common-lisp-C7gG9l05.js +1 -0
  41. package/dist/assets/coq-Dsg_Bt_b.js +1 -0
  42. package/dist/assets/cpp-BksuvNSY.js +1 -0
  43. package/dist/assets/crystal-DtDmRg-F.js +1 -0
  44. package/dist/assets/csharp-D9R-vmeu.js +1 -0
  45. package/dist/assets/css-BPhBrDlE.js +1 -0
  46. package/dist/assets/csv-B0qRVHPH.js +1 -0
  47. package/dist/assets/cue-DtFQj3wx.js +1 -0
  48. package/dist/assets/cypher-m2LEI-9-.js +1 -0
  49. package/dist/assets/d-BoXegm-a.js +1 -0
  50. package/dist/assets/dark-plus-C3mMm8J8.js +1 -0
  51. package/dist/assets/dart-B9wLZaAG.js +1 -0
  52. package/dist/assets/dax-ClGRhx96.js +1 -0
  53. package/dist/assets/desktop-DEIpsLCJ.js +1 -0
  54. package/dist/assets/diff-BgYniUM_.js +1 -0
  55. package/dist/assets/docker-COcR7UxN.js +1 -0
  56. package/dist/assets/dotenv-BjQB5zDj.js +1 -0
  57. package/dist/assets/dracula-BzJJZx-M.js +1 -0
  58. package/dist/assets/dracula-soft-BXkSAIEj.js +1 -0
  59. package/dist/assets/dream-maker-C-nORZOA.js +1 -0
  60. package/dist/assets/edge-D5gP-w-T.js +1 -0
  61. package/dist/assets/elixir-CLiX3zqd.js +1 -0
  62. package/dist/assets/elm-CmHSxxaM.js +1 -0
  63. package/dist/assets/emacs-lisp-BX77sIaO.js +1 -0
  64. package/dist/assets/erb-BYTLMnw6.js +1 -0
  65. package/dist/assets/erlang-B-DoSBHF.js +1 -0
  66. package/dist/assets/everforest-dark-BgDCqdQA.js +1 -0
  67. package/dist/assets/everforest-light-C8M2exoo.js +1 -0
  68. package/dist/assets/fennel-bCA53EVm.js +1 -0
  69. package/dist/assets/fish-w-ucz2PV.js +1 -0
  70. package/dist/assets/fluent-Dayu4EKP.js +1 -0
  71. package/dist/assets/fortran-fixed-form-TqA4NnZg.js +1 -0
  72. package/dist/assets/fortran-free-form-DKXYxT9g.js +1 -0
  73. package/dist/assets/fsharp-XplgxFYe.js +1 -0
  74. package/dist/assets/gdresource-BHYsBjWJ.js +1 -0
  75. package/dist/assets/gdscript-DfxzS6Rs.js +1 -0
  76. package/dist/assets/gdshader-SKMF96pI.js +1 -0
  77. package/dist/assets/genie-ajMbGru0.js +1 -0
  78. package/dist/assets/gherkin--30QC5Em.js +1 -0
  79. package/dist/assets/git-commit-i4q6IMui.js +1 -0
  80. package/dist/assets/git-rebase-B-v9cOL2.js +1 -0
  81. package/dist/assets/github-dark-DHJKELXO.js +1 -0
  82. package/dist/assets/github-dark-default-Cuk6v7N8.js +1 -0
  83. package/dist/assets/github-dark-dimmed-DH5Ifo-i.js +1 -0
  84. package/dist/assets/github-dark-high-contrast-E3gJ1_iC.js +1 -0
  85. package/dist/assets/github-light-DAi9KRSo.js +1 -0
  86. package/dist/assets/github-light-default-D7oLnXFd.js +1 -0
  87. package/dist/assets/github-light-high-contrast-BfjtVDDH.js +1 -0
  88. package/dist/assets/gleam-B430Bg39.js +1 -0
  89. package/dist/assets/glimmer-js-D-cwc0-E.js +1 -0
  90. package/dist/assets/glimmer-ts-pgjy16dm.js +1 -0
  91. package/dist/assets/glsl-DBO2IWDn.js +1 -0
  92. package/dist/assets/gnuplot-CM8KxXT1.js +1 -0
  93. package/dist/assets/go-B1SYOhNW.js +1 -0
  94. package/dist/assets/graphql-cDcHW_If.js +1 -0
  95. package/dist/assets/groovy-DkBy-JyN.js +1 -0
  96. package/dist/assets/hack-D1yCygmZ.js +1 -0
  97. package/dist/assets/haml-B2EZWmdv.js +1 -0
  98. package/dist/assets/handlebars-BQGss363.js +1 -0
  99. package/dist/assets/haskell-BILxekzW.js +1 -0
  100. package/dist/assets/haxe-C5wWYbrZ.js +1 -0
  101. package/dist/assets/hcl-HzYwdGDm.js +1 -0
  102. package/dist/assets/hjson-T-Tgc4AT.js +1 -0
  103. package/dist/assets/hlsl-ifBTmRxC.js +1 -0
  104. package/dist/assets/houston-DnULxvSX.js +1 -0
  105. package/dist/assets/html-C2L_23MC.js +1 -0
  106. package/dist/assets/html-derivative-CSfWNPLT.js +1 -0
  107. package/dist/assets/http-FRrOvY1W.js +1 -0
  108. package/dist/assets/hxml-TIA70rKU.js +1 -0
  109. package/dist/assets/hy-BMj5Y0dO.js +1 -0
  110. package/dist/assets/imba-bv_oIlVt.js +1 -0
  111. package/dist/assets/index-BSNzF5KT.js +121 -0
  112. package/dist/assets/index-Ux6DCth7.css +1 -0
  113. package/dist/assets/ini-BjABl1g7.js +1 -0
  114. package/dist/assets/java-xI-RfyKK.js +1 -0
  115. package/dist/assets/javascript-ySlJ1b_l.js +1 -0
  116. package/dist/assets/jinja-DGy0s7-h.js +1 -0
  117. package/dist/assets/jison-BqZprYcd.js +1 -0
  118. package/dist/assets/json-BQoSv7ci.js +1 -0
  119. package/dist/assets/json5-w8dY5SsB.js +1 -0
  120. package/dist/assets/jsonc-TU54ms6u.js +1 -0
  121. package/dist/assets/jsonl-DREVFZK8.js +1 -0
  122. package/dist/assets/jsonnet-BfivnA6A.js +1 -0
  123. package/dist/assets/jssm-P4WzXJd0.js +1 -0
  124. package/dist/assets/jsx-BAng5TT0.js +1 -0
  125. package/dist/assets/julia-BBuGR-5E.js +1 -0
  126. package/dist/assets/kanagawa-dragon-CkXjmgJE.js +1 -0
  127. package/dist/assets/kanagawa-lotus-CfQXZHmo.js +1 -0
  128. package/dist/assets/kanagawa-wave-DWedfzmr.js +1 -0
  129. package/dist/assets/kotlin-B5lbUyaz.js +1 -0
  130. package/dist/assets/kusto-mebxcVVE.js +1 -0
  131. package/dist/assets/laserwave-DUszq2jm.js +1 -0
  132. package/dist/assets/latex-C-cWTeAZ.js +1 -0
  133. package/dist/assets/lean-XBlWyCtg.js +1 -0
  134. package/dist/assets/less-BfCpw3nA.js +1 -0
  135. package/dist/assets/light-plus-B7mTdjB0.js +1 -0
  136. package/dist/assets/liquid-D3W5UaiH.js +1 -0
  137. package/dist/assets/log-Cc5clBb7.js +1 -0
  138. package/dist/assets/logo-IuBKFhSY.js +1 -0
  139. package/dist/assets/lua-CvWAzNxB.js +1 -0
  140. package/dist/assets/luau-Du5NY7AG.js +1 -0
  141. package/dist/assets/make-Bvotw-X0.js +1 -0
  142. package/dist/assets/markdown-UIAJJxZW.js +1 -0
  143. package/dist/assets/marko-z0MBrx5-.js +1 -0
  144. package/dist/assets/material-theme-D5KoaKCx.js +1 -0
  145. package/dist/assets/material-theme-darker-BfHTSMKl.js +1 -0
  146. package/dist/assets/material-theme-lighter-B0m2ddpp.js +1 -0
  147. package/dist/assets/material-theme-ocean-CyktbL80.js +1 -0
  148. package/dist/assets/material-theme-palenight-Csfq5Kiy.js +1 -0
  149. package/dist/assets/matlab-D9-PGadD.js +1 -0
  150. package/dist/assets/mdc-DB_EDNY_.js +1 -0
  151. package/dist/assets/mdx-sdHcTMYB.js +1 -0
  152. package/dist/assets/mermaid-Ci6OQyBP.js +1 -0
  153. package/dist/assets/min-dark-CafNBF8u.js +1 -0
  154. package/dist/assets/min-light-CTRr51gU.js +1 -0
  155. package/dist/assets/mipsasm-BC5c_5Pe.js +1 -0
  156. package/dist/assets/mojo-Tz6hzZYG.js +1 -0
  157. package/dist/assets/monokai-D4h5O-jR.js +1 -0
  158. package/dist/assets/move-DB_GagMm.js +1 -0
  159. package/dist/assets/narrat-DLbgOhZU.js +1 -0
  160. package/dist/assets/nextflow-B0XVJmRM.js +1 -0
  161. package/dist/assets/nginx-D_VnBJ67.js +1 -0
  162. package/dist/assets/night-owl-C39BiMTA.js +1 -0
  163. package/dist/assets/nim-ZlGxZxc3.js +1 -0
  164. package/dist/assets/nix-shcSOmrb.js +1 -0
  165. package/dist/assets/nord-Ddv68eIx.js +1 -0
  166. package/dist/assets/nushell-D4Tzg5kh.js +1 -0
  167. package/dist/assets/objective-c-Deuh7S70.js +1 -0
  168. package/dist/assets/objective-cpp-BUEGK8hf.js +1 -0
  169. package/dist/assets/ocaml-BNioltXt.js +1 -0
  170. package/dist/assets/one-dark-pro-GBQ2dnAY.js +1 -0
  171. package/dist/assets/one-light-PoHY5YXO.js +1 -0
  172. package/dist/assets/pascal-JqZropPD.js +1 -0
  173. package/dist/assets/perl-CHQXSrWU.js +1 -0
  174. package/dist/assets/php-B5ebYQev.js +1 -0
  175. package/dist/assets/plastic-3e1v2bzS.js +1 -0
  176. package/dist/assets/plsql-LKU2TuZ1.js +1 -0
  177. package/dist/assets/po-BFLt1xDp.js +1 -0
  178. package/dist/assets/poimandres-CS3Unz2-.js +1 -0
  179. package/dist/assets/polar-DKykz6zU.js +1 -0
  180. package/dist/assets/postcss-B3ZDOciz.js +1 -0
  181. package/dist/assets/powerquery-CSHBycmS.js +1 -0
  182. package/dist/assets/powershell-BIEUsx6d.js +1 -0
  183. package/dist/assets/prisma-B48N-Iqd.js +1 -0
  184. package/dist/assets/prolog-BY-TUvya.js +1 -0
  185. package/dist/assets/proto-zocC4JxJ.js +1 -0
  186. package/dist/assets/pug-CM9l7STV.js +1 -0
  187. package/dist/assets/puppet-Cza_XSSt.js +1 -0
  188. package/dist/assets/purescript-Bg-kzb6g.js +1 -0
  189. package/dist/assets/python-DhUJRlN_.js +1 -0
  190. package/dist/assets/qml-D8XfuvdV.js +1 -0
  191. package/dist/assets/qmldir-C8lEn-DE.js +1 -0
  192. package/dist/assets/qss-DhMKtDLN.js +1 -0
  193. package/dist/assets/r-CwjWoCRV.js +1 -0
  194. package/dist/assets/racket-CzouJOBO.js +1 -0
  195. package/dist/assets/raku-B1bQXN8T.js +1 -0
  196. package/dist/assets/razor-CNLDkMZG.js +1 -0
  197. package/dist/assets/red-bN70gL4F.js +1 -0
  198. package/dist/assets/reg-5LuOXUq_.js +1 -0
  199. package/dist/assets/regexp-DWJ3fJO_.js +1 -0
  200. package/dist/assets/rel-DJlmqQ1C.js +1 -0
  201. package/dist/assets/riscv-QhoSD0DR.js +1 -0
  202. package/dist/assets/rose-pine-CmCqftbK.js +1 -0
  203. package/dist/assets/rose-pine-dawn-Ds-gbosJ.js +1 -0
  204. package/dist/assets/rose-pine-moon-CjDtw9vr.js +1 -0
  205. package/dist/assets/rst-4NLicBqY.js +1 -0
  206. package/dist/assets/ruby-DeZ3UC14.js +1 -0
  207. package/dist/assets/rust-Be6lgOlo.js +1 -0
  208. package/dist/assets/sas-BmTFh92c.js +1 -0
  209. package/dist/assets/sass-BJ4Li9vH.js +1 -0
  210. package/dist/assets/scala-DQVVAn-B.js +1 -0
  211. package/dist/assets/scheme-BJGe-b2p.js +1 -0
  212. package/dist/assets/scss-C31hgJw-.js +1 -0
  213. package/dist/assets/sdbl-BLhTXw86.js +1 -0
  214. package/dist/assets/shaderlab-B7qAK45m.js +1 -0
  215. package/dist/assets/shellscript-atvbtKCR.js +1 -0
  216. package/dist/assets/shellsession-C_rIy8kc.js +1 -0
  217. package/dist/assets/slack-dark-BthQWCQV.js +1 -0
  218. package/dist/assets/slack-ochin-DqwNpetd.js +1 -0
  219. package/dist/assets/smalltalk-DkLiglaE.js +1 -0
  220. package/dist/assets/snazzy-light-Bw305WKR.js +1 -0
  221. package/dist/assets/solarized-dark-DXbdFlpD.js +1 -0
  222. package/dist/assets/solarized-light-L9t79GZl.js +1 -0
  223. package/dist/assets/solidity-C1w2a3ep.js +1 -0
  224. package/dist/assets/soy-C-lX7w71.js +1 -0
  225. package/dist/assets/sparql-bYkjHRlG.js +1 -0
  226. package/dist/assets/splunk-Cf8iN4DR.js +1 -0
  227. package/dist/assets/sql-COK4E0Yg.js +1 -0
  228. package/dist/assets/ssh-config-BknIz3MU.js +1 -0
  229. package/dist/assets/stata-DorPZHa4.js +1 -0
  230. package/dist/assets/stylus-BeQkCIfX.js +1 -0
  231. package/dist/assets/svelte-MSaWC3Je.js +1 -0
  232. package/dist/assets/swift-BSxZ-RaX.js +1 -0
  233. package/dist/assets/synthwave-84-CbfX1IO0.js +1 -0
  234. package/dist/assets/system-verilog-C7L56vO4.js +1 -0
  235. package/dist/assets/systemd-CUnW07Te.js +1 -0
  236. package/dist/assets/talonscript-C1XDQQGZ.js +1 -0
  237. package/dist/assets/tasl-CQjiPCtT.js +1 -0
  238. package/dist/assets/tcl-DQ1-QYvQ.js +1 -0
  239. package/dist/assets/templ-dwX3ZSMB.js +1 -0
  240. package/dist/assets/terraform-BbSNqyBO.js +1 -0
  241. package/dist/assets/tex-rYs2v40G.js +1 -0
  242. package/dist/assets/tokyo-night-DBQeEorK.js +1 -0
  243. package/dist/assets/toml-CB2ApiWb.js +1 -0
  244. package/dist/assets/ts-tags-CipyTH0X.js +1 -0
  245. package/dist/assets/tsv-B_m7g4N7.js +1 -0
  246. package/dist/assets/tsx-B6W0miNI.js +1 -0
  247. package/dist/assets/turtle-BMR_PYu6.js +1 -0
  248. package/dist/assets/twig-NC5TFiHP.js +1 -0
  249. package/dist/assets/typescript-Dj6nwHGl.js +1 -0
  250. package/dist/assets/typespec-BpWG_bgh.js +1 -0
  251. package/dist/assets/typst-BVUVsWT6.js +1 -0
  252. package/dist/assets/v-CAQ2eGtk.js +1 -0
  253. package/dist/assets/vala-BFOHcciG.js +1 -0
  254. package/dist/assets/vb-CdO5JTpU.js +1 -0
  255. package/dist/assets/verilog-CJaU5se_.js +1 -0
  256. package/dist/assets/vesper-BEBZ7ncR.js +1 -0
  257. package/dist/assets/vhdl-DYoNaHQp.js +1 -0
  258. package/dist/assets/viml-m4uW47V2.js +1 -0
  259. package/dist/assets/vitesse-black-Bkuqu6BP.js +1 -0
  260. package/dist/assets/vitesse-dark-D0r3Knsf.js +1 -0
  261. package/dist/assets/vitesse-light-CVO1_9PV.js +1 -0
  262. package/dist/assets/vue-BuYVFjOK.js +1 -0
  263. package/dist/assets/vue-html-xdeiXROB.js +1 -0
  264. package/dist/assets/vyper-nyqBNV6O.js +1 -0
  265. package/dist/assets/wasm-C6j12Q_x.js +1 -0
  266. package/dist/assets/wasm-CG6Dc4jp.js +1 -0
  267. package/dist/assets/wenyan-7A4Fjokl.js +1 -0
  268. package/dist/assets/wgsl-CB0Krxn9.js +1 -0
  269. package/dist/assets/wikitext-DCE3LsBG.js +1 -0
  270. package/dist/assets/wolfram-C3FkfJm5.js +1 -0
  271. package/dist/assets/xml-e3z08dGr.js +1 -0
  272. package/dist/assets/xsl-Dd0NUgwM.js +1 -0
  273. package/dist/assets/yaml-CVw76BM1.js +1 -0
  274. package/dist/assets/zenscript-HnGAYVZD.js +1 -0
  275. package/dist/assets/zig-BVz_zdnA.js +1 -0
  276. package/{client → dist}/index.html +2 -1
  277. package/package.json +9 -2
  278. package/server/dolt-pool.js +69 -13
  279. package/.github/workflows/publish.yml +0 -28
  280. package/client/postcss.config.js +0 -11
  281. package/client/src/App.tsx +0 -35
  282. package/client/src/components/IssueCard.tsx +0 -73
  283. package/client/src/components/Layout.tsx +0 -175
  284. package/client/src/components/Markdown.tsx +0 -77
  285. package/client/src/components/PriorityBadge.tsx +0 -26
  286. package/client/src/components/SearchDialog.tsx +0 -137
  287. package/client/src/components/SectionEditor.tsx +0 -212
  288. package/client/src/components/StatusBadge.tsx +0 -64
  289. package/client/src/components/TypeBadge.tsx +0 -26
  290. package/client/src/hooks/use-mutation.ts +0 -55
  291. package/client/src/hooks/use-search.ts +0 -19
  292. package/client/src/hooks/use-subscription.ts +0 -187
  293. package/client/src/index.css +0 -133
  294. package/client/src/lib/avatar.ts +0 -17
  295. package/client/src/lib/types.ts +0 -115
  296. package/client/src/lib/ws-client.ts +0 -214
  297. package/client/src/lib/ws-context.tsx +0 -28
  298. package/client/src/main.tsx +0 -10
  299. package/client/src/views/Board.tsx +0 -200
  300. package/client/src/views/Detail.tsx +0 -398
  301. package/client/src/views/List.tsx +0 -461
  302. package/client/tailwind.config.ts +0 -68
  303. package/client/tsconfig.json +0 -16
  304. package/client/vite.config.ts +0 -20
@@ -1,77 +0,0 @@
1
- import { useEffect, useState } from "react";
2
- import { marked } from "marked";
3
- import DOMPurify from "dompurify";
4
- import { codeToHtml } from "shiki";
5
-
6
- export function Markdown({ content }: { content: string }) {
7
- const [html, setHtml] = useState("");
8
-
9
- useEffect(() => {
10
- if (!content) {
11
- setHtml("");
12
- return;
13
- }
14
-
15
- let cancelled = false;
16
-
17
- async function render() {
18
- const tokens = marked.lexer(content);
19
-
20
- // Collect code blocks for syntax highlighting
21
- const codeBlocks: Array<{ code: string; lang: string }> = [];
22
- marked.walkTokens(tokens, (token) => {
23
- if (token.type === "code") {
24
- codeBlocks.push({
25
- code: (token as { text: string }).text,
26
- lang: (token as { lang?: string }).lang || "text",
27
- });
28
- }
29
- });
30
-
31
- // Highlight all code blocks in parallel
32
- const highlighted = await Promise.all(
33
- codeBlocks.map(async (block) => {
34
- try {
35
- return await codeToHtml(block.code, {
36
- lang: block.lang,
37
- theme: "github-light",
38
- });
39
- } catch {
40
- return `<pre><code>${DOMPurify.sanitize(block.code)}</code></pre>`;
41
- }
42
- }),
43
- );
44
-
45
- // Render markdown and replace code blocks with highlighted versions
46
- let rendered = marked.parser(tokens) as string;
47
- let i = 0;
48
- rendered = rendered.replace(
49
- /<pre><code[^>]*>[\s\S]*?<\/code><\/pre>/g,
50
- () => highlighted[i++] || "",
51
- );
52
-
53
- const sanitized = DOMPurify.sanitize(rendered, {
54
- ADD_TAGS: ["span"],
55
- ADD_ATTR: ["style", "class"],
56
- });
57
-
58
- if (!cancelled) setHtml(sanitized);
59
- }
60
-
61
- render();
62
- return () => {
63
- cancelled = true;
64
- };
65
- }, [content]);
66
-
67
- return (
68
- <div
69
- className="prose prose-stone prose-sm max-w-none
70
- prose-pre:bg-stone-50 prose-pre:border prose-pre:border-stone-200 prose-pre:rounded
71
- prose-code:text-sm prose-code:font-mono
72
- prose-headings:font-semibold prose-headings:text-stone-900
73
- prose-a:text-blue-600 prose-a:no-underline hover:prose-a:underline"
74
- dangerouslySetInnerHTML={{ __html: html }}
75
- />
76
- );
77
- }
@@ -1,26 +0,0 @@
1
- const PRIORITY_CONFIG: Record<number, { bg: string; text: string }> = {
2
- 0: { bg: "rgba(239,68,68,0.12)", text: "#DC2626" },
3
- 1: { bg: "rgba(249,115,22,0.12)", text: "#EA580C" },
4
- 2: { bg: "rgba(59,130,246,0.12)", text: "#2563EB" },
5
- 3: { bg: "rgba(34,197,94,0.12)", text: "#16A34A" },
6
- 4: { bg: "rgba(156,163,175,0.1)", text: "#9CA3AF" },
7
- };
8
-
9
- export function PriorityBadge({ priority }: { priority: number }) {
10
- const config = PRIORITY_CONFIG[priority] || PRIORITY_CONFIG[4];
11
- return (
12
- <span
13
- className="inline-flex items-center justify-center font-mono font-bold"
14
- style={{
15
- fontSize: "11px",
16
- lineHeight: 1,
17
- padding: "3px 8px",
18
- borderRadius: "var(--radius-sm)",
19
- backgroundColor: config.bg,
20
- color: config.text,
21
- }}
22
- >
23
- P{priority}
24
- </span>
25
- );
26
- }
@@ -1,137 +0,0 @@
1
- import { useState, useEffect, useRef } from "react";
2
- import { useSubscription } from "../hooks/use-subscription";
3
- import { useSearch } from "../hooks/use-search";
4
- import { StatusBadge } from "./StatusBadge";
5
- import { PriorityBadge } from "./PriorityBadge";
6
-
7
- export function SearchDialog() {
8
- const [open, setOpen] = useState(false);
9
- const { issues } = useSubscription("all-issues");
10
- const { query, setQuery, results } = useSearch(issues);
11
- const [selectedIndex, setSelectedIndex] = useState(0);
12
- const inputRef = useRef<HTMLInputElement>(null);
13
-
14
- // Cmd+K to open, Escape to close
15
- useEffect(() => {
16
- const handler = (e: KeyboardEvent) => {
17
- const tag = (e.target as HTMLElement)?.tagName;
18
- if (e.key === "k" && (e.metaKey || e.ctrlKey) && tag !== "TEXTAREA") {
19
- e.preventDefault();
20
- setOpen(true);
21
- }
22
- if (e.key === "Escape" && open) {
23
- e.preventDefault();
24
- setOpen(false);
25
- setQuery("");
26
- setSelectedIndex(0);
27
- }
28
- };
29
- window.addEventListener("keydown", handler);
30
- return () => window.removeEventListener("keydown", handler);
31
- }, [open, setQuery]);
32
-
33
- // Focus input when opened
34
- useEffect(() => {
35
- if (open) setTimeout(() => inputRef.current?.focus(), 0);
36
- }, [open]);
37
-
38
- // Reset selection when results change
39
- useEffect(() => setSelectedIndex(0), [results]);
40
-
41
- const handleKeyDown = (e: React.KeyboardEvent) => {
42
- const shown = results.slice(0, 20);
43
- if (e.key === "ArrowDown") {
44
- e.preventDefault();
45
- setSelectedIndex((i) => Math.min(i + 1, shown.length - 1));
46
- }
47
- if (e.key === "ArrowUp") {
48
- e.preventDefault();
49
- setSelectedIndex((i) => Math.max(i - 1, 0));
50
- }
51
- if (e.key === "Enter" && shown[selectedIndex]) {
52
- window.location.hash = `#/detail/${shown[selectedIndex].id}`;
53
- setOpen(false);
54
- setQuery("");
55
- }
56
- };
57
-
58
- if (!open) return null;
59
-
60
- const shown = results.slice(0, 20);
61
-
62
- return (
63
- <div
64
- className="fixed inset-0 z-50 flex items-start justify-center pt-[20vh]"
65
- onClick={() => {
66
- setOpen(false);
67
- setQuery("");
68
- }}
69
- >
70
- <div className="absolute inset-0" style={{ background: "var(--bg-overlay)" }} />
71
- <div
72
- className="relative w-full max-w-lg overflow-hidden"
73
- style={{
74
- background: "var(--bg-elevated)",
75
- borderRadius: "var(--radius-lg)",
76
- border: "1px solid var(--border-subtle)",
77
- boxShadow: "0 25px 50px -12px rgba(0,0,0,0.15)",
78
- }}
79
- onClick={(e) => e.stopPropagation()}
80
- >
81
- <input
82
- ref={inputRef}
83
- type="text"
84
- value={query}
85
- onChange={(e) => setQuery(e.target.value)}
86
- onKeyDown={handleKeyDown}
87
- placeholder="Search issues..."
88
- className="w-full px-4 py-3 text-sm outline-none"
89
- style={{
90
- borderBottom: "1px solid var(--border-subtle)",
91
- background: "transparent",
92
- color: "var(--text-primary)",
93
- }}
94
- />
95
- <div className="max-h-80 overflow-y-auto">
96
- {shown.length === 0 && query && (
97
- <div className="px-4 py-6 text-sm text-center" style={{ color: "var(--text-tertiary)" }}>
98
- No results
99
- </div>
100
- )}
101
- {shown.map((issue, i) => (
102
- <button
103
- key={issue.id}
104
- onClick={() => {
105
- window.location.hash = `#/detail/${issue.id}`;
106
- setOpen(false);
107
- setQuery("");
108
- }}
109
- className="w-full flex items-center gap-3 px-4 py-2.5 text-sm text-left transition-colors"
110
- style={{
111
- background: i === selectedIndex ? "var(--bg-hover)" : "transparent",
112
- }}
113
- onMouseEnter={(e) => { e.currentTarget.style.background = "var(--bg-hover)"; }}
114
- onMouseLeave={(e) => { e.currentTarget.style.background = i === selectedIndex ? "var(--bg-hover)" : "transparent"; }}
115
- >
116
- <span className="font-mono text-xs w-28 shrink-0" style={{ color: "var(--text-tertiary)" }}>
117
- {issue.id}
118
- </span>
119
- <StatusBadge status={issue.status} />
120
- <span className="flex-1 truncate" style={{ color: "var(--text-primary)" }}>{issue.title}</span>
121
- <PriorityBadge priority={issue.priority} />
122
- </button>
123
- ))}
124
- </div>
125
- <div
126
- className="px-4 py-2 text-xs"
127
- style={{
128
- color: "var(--text-tertiary)",
129
- borderTop: "1px solid var(--border-subtle)",
130
- }}
131
- >
132
- &uarr;&darr; navigate &middot; &crarr; open &middot; esc close
133
- </div>
134
- </div>
135
- </div>
136
- );
137
- }
@@ -1,212 +0,0 @@
1
- import { useState, useCallback, useEffect } from "react";
2
- import { Markdown } from "./Markdown";
3
-
4
- interface SectionEditorProps {
5
- label: string;
6
- value: string;
7
- placeholder?: string;
8
- onSave: (value: string) => Promise<unknown>;
9
- }
10
-
11
- export function SectionEditor({
12
- label,
13
- value,
14
- placeholder,
15
- onSave,
16
- }: SectionEditorProps) {
17
- const [editing, setEditing] = useState(false);
18
- const [draft, setDraft] = useState(value);
19
- const [tab, setTab] = useState<"edit" | "preview">("edit");
20
- const [saving, setSaving] = useState(false);
21
-
22
- // Sync draft with external value when not editing
23
- useEffect(() => {
24
- if (!editing) setDraft(value);
25
- }, [value, editing]);
26
-
27
- const handleSave = useCallback(async () => {
28
- setSaving(true);
29
- try {
30
- await onSave(draft);
31
- setEditing(false);
32
- setTab("edit");
33
- } finally {
34
- setSaving(false);
35
- }
36
- }, [draft, onSave]);
37
-
38
- const handleCancel = useCallback(() => {
39
- setDraft(value);
40
- setEditing(false);
41
- setTab("edit");
42
- }, [value]);
43
-
44
- const handleEditorKeyDown = useCallback(
45
- (e: React.KeyboardEvent) => {
46
- if (e.key === "s" && (e.metaKey || e.ctrlKey)) {
47
- e.preventDefault();
48
- handleSave();
49
- }
50
- if (e.key === "Escape") {
51
- e.preventDefault();
52
- handleCancel();
53
- }
54
- },
55
- [handleSave, handleCancel],
56
- );
57
-
58
- if (!editing) {
59
- return (
60
- <div
61
- className="group rounded-lg p-4"
62
- style={{
63
- background: "var(--bg-elevated)",
64
- border: "1px solid var(--border-subtle)",
65
- boxShadow: "var(--shadow-card)",
66
- }}
67
- >
68
- <div className="flex items-center justify-between mb-2">
69
- <h3
70
- className="font-semibold uppercase tracking-wider"
71
- style={{ fontSize: "11px", color: "var(--text-tertiary)" }}
72
- >
73
- {label}
74
- </h3>
75
- <button
76
- onClick={() => setEditing(true)}
77
- className="opacity-0 group-hover:opacity-100 transition-opacity"
78
- style={{ color: "var(--text-tertiary)" }}
79
- onMouseEnter={(e) => { e.currentTarget.style.color = "var(--accent)"; }}
80
- onMouseLeave={(e) => { e.currentTarget.style.color = "var(--text-tertiary)"; }}
81
- title={`Edit ${label}`}
82
- >
83
- <svg
84
- className="w-4 h-4"
85
- fill="none"
86
- stroke="currentColor"
87
- viewBox="0 0 24 24"
88
- >
89
- <path
90
- strokeLinecap="round"
91
- strokeLinejoin="round"
92
- strokeWidth={2}
93
- d="M15.232 5.232l3.536 3.536m-2.036-5.036a2.5 2.5 0 113.536 3.536L6.5 21.036H3v-3.572L16.732 3.732z"
94
- />
95
- </svg>
96
- </button>
97
- </div>
98
- {value ? (
99
- <Markdown content={value} />
100
- ) : (
101
- <p className="text-sm italic" style={{ color: "var(--text-tertiary)" }}>
102
- {placeholder || `Add ${label.toLowerCase()}...`}
103
- </p>
104
- )}
105
- </div>
106
- );
107
- }
108
-
109
- return (
110
- <div
111
- className="rounded-lg p-4"
112
- style={{
113
- background: "var(--bg-elevated)",
114
- border: "1px solid var(--accent)",
115
- boxShadow: "var(--shadow-card)",
116
- }}
117
- onKeyDown={handleEditorKeyDown}
118
- >
119
- <div className="flex items-center justify-between mb-2">
120
- <h3
121
- className="font-semibold uppercase tracking-wider"
122
- style={{ fontSize: "11px", color: "var(--text-tertiary)" }}
123
- >
124
- {label}
125
- </h3>
126
- <div
127
- className="flex overflow-hidden text-xs rounded-md"
128
- style={{ border: "1px solid var(--border-default)" }}
129
- >
130
- <button
131
- onClick={() => setTab("edit")}
132
- className="px-2.5 py-1 font-medium"
133
- style={{
134
- background: tab === "edit" ? "var(--bg-hover)" : "var(--bg-elevated)",
135
- color: tab === "edit" ? "var(--text-primary)" : "var(--text-secondary)",
136
- }}
137
- >
138
- Edit
139
- </button>
140
- <button
141
- onClick={() => setTab("preview")}
142
- className="px-2.5 py-1 font-medium"
143
- style={{
144
- background: tab === "preview" ? "var(--bg-hover)" : "var(--bg-elevated)",
145
- color: tab === "preview" ? "var(--text-primary)" : "var(--text-secondary)",
146
- borderLeft: "1px solid var(--border-default)",
147
- }}
148
- >
149
- Preview
150
- </button>
151
- </div>
152
- </div>
153
- {tab === "edit" ? (
154
- <textarea
155
- value={draft}
156
- onChange={(e) => setDraft(e.target.value)}
157
- rows={10}
158
- className="w-full p-3 text-sm font-mono rounded-md resize-y outline-none"
159
- style={{
160
- border: "1px solid var(--border-default)",
161
- background: "var(--bg-base)",
162
- color: "var(--text-primary)",
163
- }}
164
- onFocus={(e) => { e.currentTarget.style.borderColor = "var(--accent)"; }}
165
- onBlur={(e) => { e.currentTarget.style.borderColor = "var(--border-default)"; }}
166
- autoFocus
167
- />
168
- ) : (
169
- <div
170
- className="p-3 rounded-md min-h-[160px]"
171
- style={{
172
- border: "1px solid var(--border-default)",
173
- background: "var(--bg-base)",
174
- }}
175
- >
176
- <Markdown content={draft} />
177
- </div>
178
- )}
179
- <div className="flex items-center justify-end gap-2 mt-3">
180
- <span className="text-xs mr-auto" style={{ color: "var(--text-tertiary)" }}>
181
- Cmd+S to save, Esc to cancel
182
- </span>
183
- <button
184
- onClick={handleCancel}
185
- className="px-3 py-1.5 text-sm rounded-md transition-colors"
186
- style={{
187
- color: "var(--text-secondary)",
188
- border: "1px solid var(--border-default)",
189
- background: "var(--bg-elevated)",
190
- }}
191
- onMouseEnter={(e) => { e.currentTarget.style.background = "var(--bg-hover)"; }}
192
- onMouseLeave={(e) => { e.currentTarget.style.background = "var(--bg-elevated)"; }}
193
- >
194
- Cancel
195
- </button>
196
- <button
197
- onClick={handleSave}
198
- disabled={saving}
199
- className="px-3 py-1.5 text-sm rounded-md font-medium disabled:opacity-50 transition-colors"
200
- style={{
201
- background: "var(--accent)",
202
- color: "white",
203
- }}
204
- onMouseEnter={(e) => { e.currentTarget.style.opacity = "0.9"; }}
205
- onMouseLeave={(e) => { e.currentTarget.style.opacity = "1"; }}
206
- >
207
- {saving ? "Saving..." : "Save"}
208
- </button>
209
- </div>
210
- </div>
211
- );
212
- }
@@ -1,64 +0,0 @@
1
- const STATUS_CONFIG: Record<string, { dot: string; bg: string; text: string; border: string; label: string }> = {
2
- open: {
3
- dot: "var(--status-open)",
4
- bg: "rgba(59,130,246,0.08)",
5
- text: "#2563EB",
6
- border: "rgba(59,130,246,0.2)",
7
- label: "Open",
8
- },
9
- in_progress: {
10
- dot: "var(--status-in-progress)",
11
- bg: "rgba(245,158,11,0.08)",
12
- text: "#D97706",
13
- border: "rgba(245,158,11,0.2)",
14
- label: "In Progress",
15
- },
16
- blocked: {
17
- dot: "var(--status-blocked)",
18
- bg: "rgba(239,68,68,0.08)",
19
- text: "#DC2626",
20
- border: "rgba(239,68,68,0.2)",
21
- label: "Blocked",
22
- },
23
- closed: {
24
- dot: "var(--status-closed)",
25
- bg: "rgba(34,197,94,0.08)",
26
- text: "#16A34A",
27
- border: "rgba(34,197,94,0.2)",
28
- label: "Closed",
29
- },
30
- };
31
-
32
- const DEFAULT_CONFIG = {
33
- dot: "#9CA3AF",
34
- bg: "rgba(156,163,175,0.08)",
35
- text: "#6B7280",
36
- border: "rgba(156,163,175,0.2)",
37
- label: "",
38
- };
39
-
40
- export function StatusBadge({ status }: { status: string }) {
41
- const config = STATUS_CONFIG[status] || DEFAULT_CONFIG;
42
- return (
43
- <span
44
- className="inline-flex items-center gap-1.5 px-2 py-0.5 font-semibold"
45
- style={{
46
- fontSize: "11px",
47
- borderRadius: "20px",
48
- backgroundColor: config.bg,
49
- color: config.text,
50
- border: `1px solid ${config.border}`,
51
- }}
52
- >
53
- <span
54
- className="rounded-full shrink-0"
55
- style={{
56
- width: "6px",
57
- height: "6px",
58
- backgroundColor: config.dot,
59
- }}
60
- />
61
- {config.label || status}
62
- </span>
63
- );
64
- }
@@ -1,26 +0,0 @@
1
- const TYPE_CONFIG: Record<string, { bg: string; text: string }> = {
2
- epic: { bg: "rgba(124,58,237,0.1)", text: "#7C3AED" },
3
- feature: { bg: "rgba(99,102,241,0.1)", text: "#6366F1" },
4
- bug: { bg: "rgba(239,68,68,0.1)", text: "#DC2626" },
5
- task: { bg: "rgba(22,163,74,0.1)", text: "#16A34A" },
6
- chore: { bg: "rgba(120,113,108,0.1)", text: "#78716C" },
7
- };
8
-
9
- const DEFAULT = { bg: "rgba(120,113,108,0.1)", text: "#78716C" };
10
-
11
- export function TypeBadge({ type }: { type: string }) {
12
- const config = TYPE_CONFIG[type] || DEFAULT;
13
- return (
14
- <span
15
- className="inline-flex items-center px-2 py-0.5 font-semibold capitalize"
16
- style={{
17
- fontSize: "11px",
18
- borderRadius: "var(--radius-sm)",
19
- backgroundColor: config.bg,
20
- color: config.text,
21
- }}
22
- >
23
- {type}
24
- </span>
25
- );
26
- }
@@ -1,55 +0,0 @@
1
- import { useCallback } from "react";
2
- import { useWs } from "../lib/ws-context";
3
- import type {
4
- EditTextPayload,
5
- CreateIssuePayload,
6
- } from "../lib/types";
7
-
8
- export function useMutation() {
9
- const ws = useWs();
10
-
11
- return {
12
- updateStatus: useCallback(
13
- (id: string, status: "open" | "in_progress" | "closed") =>
14
- ws.updateStatus({ id, status }),
15
- [ws],
16
- ),
17
- editText: useCallback(
18
- (payload: EditTextPayload) => ws.editText(payload),
19
- [ws],
20
- ),
21
- createIssue: useCallback(
22
- (payload: CreateIssuePayload) => ws.createIssue(payload),
23
- [ws],
24
- ),
25
- updatePriority: useCallback(
26
- (id: string, priority: number) => ws.updatePriority(id, priority),
27
- [ws],
28
- ),
29
- updateAssignee: useCallback(
30
- (id: string, assignee: string) => ws.updateAssignee(id, assignee),
31
- [ws],
32
- ),
33
- addLabel: useCallback(
34
- (id: string, label: string) => ws.addLabel(id, label),
35
- [ws],
36
- ),
37
- removeLabel: useCallback(
38
- (id: string, label: string) => ws.removeLabel(id, label),
39
- [ws],
40
- ),
41
- deleteIssue: useCallback((id: string) => ws.deleteIssue(id), [ws]),
42
- addComment: useCallback(
43
- (id: string, text: string) => ws.addComment(id, text),
44
- [ws],
45
- ),
46
- addDep: useCallback(
47
- (a: string, b: string) => ws.addDep(a, b),
48
- [ws],
49
- ),
50
- removeDep: useCallback(
51
- (a: string, b: string) => ws.removeDep(a, b),
52
- [ws],
53
- ),
54
- };
55
- }
@@ -1,19 +0,0 @@
1
- import { useState, useMemo } from "react";
2
- import type { Issue } from "../lib/types";
3
-
4
- export function useSearch(issues: Issue[]) {
5
- const [query, setQuery] = useState("");
6
-
7
- const results = useMemo(() => {
8
- if (!query.trim()) return issues;
9
- const q = query.toLowerCase();
10
- return issues.filter(
11
- (i) =>
12
- i.id.toLowerCase().includes(q) ||
13
- i.title.toLowerCase().includes(q) ||
14
- (i.description || "").toLowerCase().includes(q),
15
- );
16
- }, [issues, query]);
17
-
18
- return { query, setQuery, results };
19
- }