@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,461 +0,0 @@
1
- import { useState, useMemo, useEffect, useRef } from "react";
2
- import { useSubscription } from "../hooks/use-subscription";
3
- import { StatusBadge } from "../components/StatusBadge";
4
- import { PriorityBadge } from "../components/PriorityBadge";
5
- import { TypeBadge } from "../components/TypeBadge";
6
- import { getInitials, getAvatarColor } from "../lib/avatar";
7
- import type { Issue } from "../lib/types";
8
-
9
- const PAGE_SIZE = 20;
10
-
11
- const SearchIcon = () => (
12
- <svg width="14" height="14" viewBox="0 0 16 16" fill="none" stroke="currentColor" strokeWidth="1.5" strokeLinecap="round">
13
- <circle cx="7" cy="7" r="5" />
14
- <line x1="11" y1="11" x2="14" y2="14" />
15
- </svg>
16
- );
17
-
18
- function SkeletonRow() {
19
- return (
20
- <div
21
- className="flex items-center gap-3 px-4 py-3"
22
- style={{ borderBottom: "1px solid var(--border-subtle)" }}
23
- >
24
- <div className="w-36 h-3 rounded skeleton-shimmer" />
25
- <div className="w-20 h-5 rounded-full skeleton-shimmer" />
26
- <div className="w-8 h-5 rounded skeleton-shimmer" />
27
- <div className="flex-1 h-3 rounded skeleton-shimmer" />
28
- <div className="w-14 h-5 rounded skeleton-shimmer" />
29
- <div className="w-24 h-5 rounded skeleton-shimmer" />
30
- <div className="w-16 h-3 rounded skeleton-shimmer" />
31
- </div>
32
- );
33
- }
34
-
35
- function Spinner() {
36
- return (
37
- <svg
38
- width="20"
39
- height="20"
40
- viewBox="0 0 20 20"
41
- fill="none"
42
- style={{ animation: "spin 0.8s linear infinite" }}
43
- >
44
- <circle cx="10" cy="10" r="8" stroke="var(--border-default)" strokeWidth="2" />
45
- <path
46
- d="M10 2a8 8 0 0 1 8 8"
47
- stroke="var(--accent)"
48
- strokeWidth="2"
49
- strokeLinecap="round"
50
- />
51
- </svg>
52
- );
53
- }
54
-
55
- function Toolbar({
56
- search,
57
- setSearch,
58
- statusFilter,
59
- setStatusFilter,
60
- typeFilter,
61
- setTypeFilter,
62
- }: {
63
- search: string;
64
- setSearch: (v: string) => void;
65
- statusFilter: string;
66
- setStatusFilter: (v: string) => void;
67
- typeFilter: string;
68
- setTypeFilter: (v: string) => void;
69
- }) {
70
- return (
71
- <div
72
- className="flex items-center gap-3 mb-4 px-4 py-3 rounded-lg"
73
- style={{
74
- background: "var(--bg-elevated)",
75
- border: "1px solid var(--border-subtle)",
76
- boxShadow: "var(--shadow-sm)",
77
- }}
78
- >
79
- <div className="relative flex-1 max-w-xs">
80
- <span className="absolute left-3 top-1/2 -translate-y-1/2" style={{ color: "var(--text-tertiary)" }}>
81
- <SearchIcon />
82
- </span>
83
- <input
84
- type="text"
85
- value={search}
86
- onChange={(e) => setSearch(e.target.value)}
87
- placeholder="Search issues..."
88
- className="w-full pl-9 pr-3 py-1.5 text-sm rounded-md outline-none"
89
- style={{
90
- border: "1px solid var(--border-default)",
91
- background: "var(--bg-base)",
92
- color: "var(--text-primary)",
93
- }}
94
- onFocus={(e) => { e.currentTarget.style.borderColor = "var(--accent)"; }}
95
- onBlur={(e) => { e.currentTarget.style.borderColor = "var(--border-default)"; }}
96
- />
97
- </div>
98
- <select
99
- value={statusFilter}
100
- onChange={(e) => setStatusFilter(e.target.value)}
101
- className="px-3 py-1.5 text-sm rounded-md"
102
- style={{
103
- border: "1px solid var(--border-default)",
104
- background: "var(--bg-base)",
105
- color: "var(--text-primary)",
106
- }}
107
- >
108
- <option value="all">All statuses</option>
109
- <option value="open">Open</option>
110
- <option value="in_progress">In Progress</option>
111
- <option value="blocked">Blocked</option>
112
- <option value="closed">Closed</option>
113
- </select>
114
- <select
115
- value={typeFilter}
116
- onChange={(e) => setTypeFilter(e.target.value)}
117
- className="px-3 py-1.5 text-sm rounded-md"
118
- style={{
119
- border: "1px solid var(--border-default)",
120
- background: "var(--bg-base)",
121
- color: "var(--text-primary)",
122
- }}
123
- >
124
- <option value="all">All types</option>
125
- <option value="epic">Epic</option>
126
- <option value="feature">Feature</option>
127
- <option value="task">Task</option>
128
- <option value="bug">Bug</option>
129
- <option value="chore">Chore</option>
130
- </select>
131
- </div>
132
- );
133
- }
134
-
135
- function Pagination({
136
- page,
137
- totalPages,
138
- total,
139
- pageSize,
140
- onPageChange,
141
- }: {
142
- page: number;
143
- totalPages: number;
144
- total: number;
145
- pageSize: number;
146
- onPageChange: (p: number) => void;
147
- }) {
148
- const from = page * pageSize + 1;
149
- const to = Math.min((page + 1) * pageSize, total);
150
-
151
- return (
152
- <div className="flex items-center justify-between px-1 py-2 text-xs" style={{ color: "var(--text-tertiary)" }}>
153
- <span>
154
- {total > 0 ? `${from}\u2013${to} of ${total}` : "0 issues"}
155
- </span>
156
- <div className="flex items-center gap-1">
157
- <button
158
- disabled={page === 0}
159
- onClick={() => onPageChange(0)}
160
- className="px-2 py-1 rounded disabled:opacity-30"
161
- style={{ border: "1px solid var(--border-default)", background: "var(--bg-elevated)" }}
162
- >
163
- &laquo;&laquo;
164
- </button>
165
- <button
166
- disabled={page === 0}
167
- onClick={() => onPageChange(page - 1)}
168
- className="px-2 py-1 rounded disabled:opacity-30"
169
- style={{ border: "1px solid var(--border-default)", background: "var(--bg-elevated)" }}
170
- >
171
- &laquo;
172
- </button>
173
- <span className="px-2">
174
- {page + 1} / {totalPages || 1}
175
- </span>
176
- <button
177
- disabled={page >= totalPages - 1}
178
- onClick={() => onPageChange(page + 1)}
179
- className="px-2 py-1 rounded disabled:opacity-30"
180
- style={{ border: "1px solid var(--border-default)", background: "var(--bg-elevated)" }}
181
- >
182
- &raquo;
183
- </button>
184
- <button
185
- disabled={page >= totalPages - 1}
186
- onClick={() => onPageChange(totalPages - 1)}
187
- className="px-2 py-1 rounded disabled:opacity-30"
188
- style={{ border: "1px solid var(--border-default)", background: "var(--bg-elevated)" }}
189
- >
190
- &raquo;&raquo;
191
- </button>
192
- </div>
193
- </div>
194
- );
195
- }
196
-
197
- function IssueRow({
198
- issue,
199
- selected,
200
- onClick,
201
- }: {
202
- issue: Issue;
203
- selected: boolean;
204
- onClick: () => void;
205
- }) {
206
- const ts = new Date(issue.updated_at).toLocaleDateString();
207
-
208
- return (
209
- <button
210
- onClick={onClick}
211
- className="w-full flex items-center gap-3 px-4 py-2.5 text-sm transition-colors"
212
- style={{
213
- borderBottom: "1px solid var(--border-subtle)",
214
- background: selected ? "var(--bg-hover)" : "transparent",
215
- }}
216
- onMouseEnter={(e) => {
217
- if (!selected) e.currentTarget.style.background = "var(--bg-hover)";
218
- }}
219
- onMouseLeave={(e) => {
220
- if (!selected) e.currentTarget.style.background = "transparent";
221
- }}
222
- >
223
- <span className="font-mono text-xs w-36 shrink-0 text-left" style={{ color: "var(--text-tertiary)" }}>
224
- {issue.id}
225
- </span>
226
- <span className="w-28 shrink-0">
227
- <StatusBadge status={issue.status} />
228
- </span>
229
- <span className="w-12 shrink-0">
230
- <PriorityBadge priority={issue.priority} />
231
- </span>
232
- <span className="flex-1 text-left truncate" style={{ color: "var(--text-primary)" }}>
233
- {issue.title}
234
- {issue.issue_type === "epic" && issue.total_children != null && (
235
- <span
236
- className="ml-2 text-xs px-1.5 py-0.5 rounded"
237
- style={{ background: "rgba(0,0,0,0.04)", color: "var(--text-tertiary)" }}
238
- >
239
- {issue.total_children} sub
240
- </span>
241
- )}
242
- </span>
243
- <span className="w-16 shrink-0">
244
- <TypeBadge type={issue.issue_type} />
245
- </span>
246
- <span className="w-28 shrink-0 flex items-center gap-1.5 justify-end">
247
- {issue.assignee ? (
248
- <>
249
- <div
250
- className="flex items-center justify-center rounded-full text-[9px] font-bold shrink-0"
251
- style={{
252
- width: "20px",
253
- height: "20px",
254
- backgroundColor: getAvatarColor(issue.assignee),
255
- color: "var(--text-inverse)",
256
- }}
257
- >
258
- {getInitials(issue.assignee)}
259
- </div>
260
- <span className="text-xs truncate" style={{ color: "var(--text-secondary)" }}>
261
- {issue.assignee}
262
- </span>
263
- </>
264
- ) : (
265
- <span className="text-xs" style={{ color: "var(--text-tertiary)" }}>&mdash;</span>
266
- )}
267
- </span>
268
- <span className="text-xs w-20 text-right shrink-0" style={{ color: "var(--text-tertiary)" }}>
269
- {ts}
270
- </span>
271
- </button>
272
- );
273
- }
274
-
275
- export function List() {
276
- const [search, setSearch] = useState("");
277
- const [debouncedSearch, setDebouncedSearch] = useState("");
278
- const [statusFilter, setStatusFilter] = useState("all");
279
- const [typeFilter, setTypeFilter] = useState("all");
280
- const [page, setPage] = useState(0);
281
- const [selectedIndex, setSelectedIndex] = useState(-1);
282
- const listRef = useRef<HTMLDivElement>(null);
283
-
284
- // Debounce search input
285
- useEffect(() => {
286
- const timer = setTimeout(() => setDebouncedSearch(search), 300);
287
- return () => clearTimeout(timer);
288
- }, [search]);
289
-
290
- // Build server-side subscription params
291
- const subParams = useMemo(() => {
292
- const params: Record<string, string | number | boolean> = {};
293
- if (debouncedSearch.trim()) params.q = debouncedSearch.trim();
294
- if (statusFilter !== "all") params.status = statusFilter;
295
- if (typeFilter !== "all") params.type = typeFilter;
296
- return params;
297
- }, [debouncedSearch, statusFilter, typeFilter]);
298
-
299
- // Server-side search+filter via search-issues subscription
300
- const { issues: allFiltered, loading, refreshing, total } = useSubscription(
301
- "search-issues",
302
- subParams,
303
- );
304
-
305
- const totalPages = Math.ceil(total / PAGE_SIZE);
306
-
307
- // Client-side pagination of server-filtered results
308
- const paginatedIssues = useMemo(() => {
309
- const start = page * PAGE_SIZE;
310
- return allFiltered.slice(start, start + PAGE_SIZE);
311
- }, [allFiltered, page]);
312
-
313
- // Reset page when filters change
314
- useEffect(() => {
315
- setPage(0);
316
- setSelectedIndex(-1);
317
- }, [statusFilter, typeFilter, search]);
318
-
319
- // Reset selection on page change
320
- useEffect(() => {
321
- setSelectedIndex(-1);
322
- }, [page]);
323
-
324
- // Keyboard navigation
325
- useEffect(() => {
326
- const handler = (e: KeyboardEvent) => {
327
- if (
328
- e.target instanceof HTMLInputElement ||
329
- e.target instanceof HTMLTextAreaElement ||
330
- e.target instanceof HTMLSelectElement
331
- )
332
- return;
333
- if (e.key === "j")
334
- setSelectedIndex((i) => Math.min(i + 1, paginatedIssues.length - 1));
335
- if (e.key === "k") setSelectedIndex((i) => Math.max(i - 1, -1));
336
- if (e.key === "Enter" && paginatedIssues[selectedIndex]) {
337
- window.location.hash = `#/detail/${paginatedIssues[selectedIndex].id}`;
338
- }
339
- };
340
- window.addEventListener("keydown", handler);
341
- return () => window.removeEventListener("keydown", handler);
342
- }, [paginatedIssues, selectedIndex]);
343
-
344
- return (
345
- <div className="p-6 h-full flex flex-col" style={{ background: "var(--bg-base)" }}>
346
- <div className="mb-4">
347
- <h1
348
- className="text-xl font-bold"
349
- style={{ color: "var(--text-primary)" }}
350
- >
351
- Issues
352
- </h1>
353
- <p className="text-sm mt-0.5" style={{ color: "var(--text-tertiary)" }}>
354
- {loading ? "\u00A0" : `${total} issues`}
355
- </p>
356
- </div>
357
-
358
- <Toolbar
359
- search={search}
360
- setSearch={setSearch}
361
- statusFilter={statusFilter}
362
- setStatusFilter={setStatusFilter}
363
- typeFilter={typeFilter}
364
- setTypeFilter={setTypeFilter}
365
- />
366
-
367
- {/* Table header */}
368
- <div
369
- className="rounded-t-lg overflow-hidden"
370
- style={{
371
- background: "var(--bg-elevated)",
372
- border: "1px solid var(--border-subtle)",
373
- borderBottom: "none",
374
- boxShadow: "var(--shadow-card)",
375
- }}
376
- >
377
- <div
378
- className="flex items-center gap-3 px-4 py-2 text-xs font-semibold uppercase tracking-wider"
379
- style={{
380
- color: "var(--text-tertiary)",
381
- borderBottom: "1px solid var(--border-subtle)",
382
- fontSize: "11px",
383
- }}
384
- >
385
- <span className="w-36 shrink-0">ID</span>
386
- <span className="w-28 shrink-0">Status</span>
387
- <span className="w-12 shrink-0">Priority</span>
388
- <span className="flex-1">Title</span>
389
- <span className="w-16 shrink-0">Type</span>
390
- <span className="w-28 shrink-0 text-right">Assignee</span>
391
- <span className="w-20 shrink-0 text-right">Date</span>
392
- </div>
393
- </div>
394
-
395
- {/* Table body */}
396
- <div
397
- ref={listRef}
398
- className="flex-1 overflow-y-auto rounded-b-lg relative"
399
- style={{
400
- background: "var(--bg-elevated)",
401
- border: "1px solid var(--border-subtle)",
402
- borderTop: "none",
403
- boxShadow: "var(--shadow-card)",
404
- }}
405
- >
406
- {/* Initial load: skeleton shimmer */}
407
- {loading && (
408
- <>
409
- {Array.from({ length: 8 }).map((_, i) => (
410
- <SkeletonRow key={i} />
411
- ))}
412
- </>
413
- )}
414
-
415
- {/* Refresh overlay: dim existing list + centered spinner */}
416
- {refreshing && (
417
- <div
418
- className="absolute inset-0 z-10 flex items-center justify-center"
419
- style={{
420
- background: "rgba(253,251,247,0.6)",
421
- backdropFilter: "blur(1px)",
422
- }}
423
- >
424
- <Spinner />
425
- </div>
426
- )}
427
-
428
- {/* Data rows */}
429
- {!loading && (
430
- paginatedIssues.length === 0 ? (
431
- <p className="text-sm p-8 text-center" style={{ color: "var(--text-tertiary)" }}>
432
- No issues match your filters
433
- </p>
434
- ) : (
435
- paginatedIssues.map((issue, index) => (
436
- <IssueRow
437
- key={issue.id}
438
- issue={issue}
439
- selected={index === selectedIndex}
440
- onClick={() => {
441
- setSelectedIndex(index);
442
- window.location.hash = `#/detail/${issue.id}`;
443
- }}
444
- />
445
- ))
446
- )
447
- )}
448
- </div>
449
-
450
- {!loading && totalPages > 1 && (
451
- <Pagination
452
- page={page}
453
- totalPages={totalPages}
454
- total={total}
455
- pageSize={PAGE_SIZE}
456
- onPageChange={setPage}
457
- />
458
- )}
459
- </div>
460
- );
461
- }
@@ -1,68 +0,0 @@
1
- import type { Config } from "tailwindcss";
2
- import path from "path";
3
- import typography from "@tailwindcss/typography";
4
-
5
- const clientDir = __dirname;
6
-
7
- export default {
8
- content: [
9
- path.join(clientDir, "src/**/*.{ts,tsx}"),
10
- path.join(clientDir, "index.html"),
11
- ],
12
- theme: {
13
- extend: {
14
- fontFamily: {
15
- sans: ["'DM Sans'", "system-ui", "-apple-system", "sans-serif"],
16
- mono: ["'JetBrains Mono'", "ui-monospace", "monospace"],
17
- },
18
- colors: {
19
- status: {
20
- open: "var(--status-open)",
21
- "in-progress": "var(--status-in-progress)",
22
- blocked: "var(--status-blocked)",
23
- closed: "var(--status-closed)",
24
- },
25
- priority: {
26
- 0: "var(--priority-0)",
27
- 1: "var(--priority-1)",
28
- 2: "var(--priority-2)",
29
- 3: "var(--priority-3)",
30
- 4: "var(--priority-4)",
31
- },
32
- surface: {
33
- base: "var(--bg-base)",
34
- card: "var(--bg-surface)",
35
- elevated: "var(--bg-elevated)",
36
- hover: "var(--bg-hover)",
37
- },
38
- ink: {
39
- primary: "var(--text-primary)",
40
- secondary: "var(--text-secondary)",
41
- tertiary: "var(--text-tertiary)",
42
- },
43
- accent: {
44
- DEFAULT: "var(--accent)",
45
- soft: "var(--accent-soft)",
46
- },
47
- type: {
48
- task: "var(--type-task)",
49
- bug: "var(--type-bug)",
50
- feature: "var(--type-feature)",
51
- epic: "var(--type-epic)",
52
- chore: "var(--type-chore)",
53
- },
54
- },
55
- boxShadow: {
56
- sm: "var(--shadow-sm)",
57
- card: "var(--shadow-card)",
58
- md: "var(--shadow-md)",
59
- },
60
- borderRadius: {
61
- sm: "var(--radius-sm)",
62
- md: "var(--radius-md)",
63
- lg: "var(--radius-lg)",
64
- },
65
- },
66
- },
67
- plugins: [typography],
68
- } satisfies Config;
@@ -1,16 +0,0 @@
1
- {
2
- "compilerOptions": {
3
- "target": "ES2022",
4
- "module": "ESNext",
5
- "moduleResolution": "bundler",
6
- "jsx": "react-jsx",
7
- "strict": true,
8
- "esModuleInterop": true,
9
- "skipLibCheck": true,
10
- "outDir": "dist",
11
- "rootDir": "src",
12
- "baseUrl": ".",
13
- "paths": { "@/*": ["src/*"] }
14
- },
15
- "include": ["src"]
16
- }
@@ -1,20 +0,0 @@
1
- import { defineConfig } from "vite";
2
- import react from "@vitejs/plugin-react";
3
- import path from "path";
4
-
5
- export default defineConfig({
6
- plugins: [react()],
7
- root: path.resolve(__dirname),
8
- build: {
9
- outDir: path.resolve(__dirname, "../dist"),
10
- emptyOutDir: true,
11
- },
12
- server: {
13
- proxy: {
14
- "/ws": {
15
- target: "ws://localhost:3333",
16
- ws: true,
17
- },
18
- },
19
- },
20
- });