@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,200 +0,0 @@
1
- import { useMemo, useState, useRef, useEffect } from "react";
2
- import { useSubscription } from "../hooks/use-subscription";
3
- import { IssueCard } from "../components/IssueCard";
4
- import type { SubscriptionType } from "../lib/types";
5
-
6
- const COLUMN_PAGE_SIZE = 20;
7
-
8
- const STATUS_COLORS: Record<string, string> = {
9
- open: "var(--status-open)",
10
- in_progress: "var(--status-in-progress)",
11
- blocked: "var(--status-blocked)",
12
- closed: "var(--status-closed)",
13
- };
14
-
15
- function Column({
16
- title,
17
- subscriptionType,
18
- statusKey,
19
- onCardClick,
20
- isClosed = false,
21
- }: {
22
- title: string;
23
- subscriptionType: SubscriptionType;
24
- statusKey: string;
25
- onCardClick: (id: string) => void;
26
- isClosed?: boolean;
27
- }) {
28
- const [limit, setLimit] = useState(COLUMN_PAGE_SIZE);
29
- const params = useMemo(() => ({ limit, offset: 0 }), [limit]);
30
- const { issues, loading, total } = useSubscription(subscriptionType, params);
31
- const scrollRef = useRef<HTMLDivElement>(null);
32
- const [hasOverflow, setHasOverflow] = useState(false);
33
-
34
- const hasMore = total > issues.length;
35
- const remaining = total - issues.length;
36
- const dotColor = STATUS_COLORS[statusKey] ?? "var(--text-tertiary)";
37
-
38
- useEffect(() => {
39
- const el = scrollRef.current;
40
- if (!el) return;
41
- const check = () => setHasOverflow(el.scrollHeight > el.clientHeight);
42
- check();
43
- const observer = new ResizeObserver(check);
44
- observer.observe(el);
45
- return () => observer.disconnect();
46
- }, [issues.length]);
47
-
48
- return (
49
- <div className="flex-1 min-w-[280px] max-w-[360px] flex flex-col self-start">
50
- {/* Column header */}
51
- <div
52
- className="flex items-center gap-2.5 px-1 pb-3 mb-3"
53
- style={{ borderBottom: "1px solid var(--border-subtle)" }}
54
- >
55
- <div
56
- className="rounded-full shrink-0"
57
- style={{
58
- width: "10px",
59
- height: "10px",
60
- backgroundColor: dotColor,
61
- boxShadow: `0 0 0 3px ${dotColor}33`,
62
- }}
63
- />
64
- <h2
65
- className="text-sm font-semibold"
66
- style={{ color: "var(--text-primary)" }}
67
- >
68
- {title}
69
- </h2>
70
- <span
71
- className="text-xs font-medium px-2 py-0.5 rounded-full ml-auto"
72
- style={{
73
- backgroundColor: "rgba(0,0,0,0.05)",
74
- color: "var(--text-tertiary)",
75
- fontSize: "11px",
76
- }}
77
- >
78
- {loading ? "\u2026" : isClosed && total > issues.length ? `${issues.length} / ${total}` : total}
79
- </span>
80
- </div>
81
-
82
- {/* Card list */}
83
- <div
84
- ref={scrollRef}
85
- className="space-y-2 overflow-y-auto flex-1 pr-1 relative"
86
- style={{
87
- maxHeight: "calc(100vh - 160px)",
88
- maskImage: hasOverflow
89
- ? "linear-gradient(to bottom, transparent 0px, black 8px, black calc(100% - 8px), transparent 100%)"
90
- : undefined,
91
- WebkitMaskImage: hasOverflow
92
- ? "linear-gradient(to bottom, transparent 0px, black 8px, black calc(100% - 8px), transparent 100%)"
93
- : undefined,
94
- }}
95
- >
96
- {loading && issues.length === 0 && (
97
- <div className="py-8 text-center">
98
- <p className="text-xs" style={{ color: "var(--text-tertiary)" }}>
99
- Loading&hellip;
100
- </p>
101
- </div>
102
- )}
103
-
104
- {!loading && issues.length === 0 && (
105
- <div
106
- className="py-8 text-center"
107
- style={{
108
- border: "2px dashed var(--border-default)",
109
- borderRadius: "var(--radius-md)",
110
- minHeight: "100px",
111
- display: "flex",
112
- alignItems: "center",
113
- justifyContent: "center",
114
- }}
115
- >
116
- <p className="text-xs" style={{ color: "var(--text-tertiary)" }}>
117
- No items
118
- </p>
119
- </div>
120
- )}
121
-
122
- {issues.map((issue) => (
123
- <IssueCard
124
- key={issue.id}
125
- issue={issue}
126
- onClick={() => onCardClick(issue.id)}
127
- dimmed={isClosed}
128
- />
129
- ))}
130
-
131
- {hasMore && (
132
- <button
133
- onClick={() => setLimit((l) => l + COLUMN_PAGE_SIZE)}
134
- className="w-full text-xs py-2"
135
- style={{
136
- color: "var(--text-secondary)",
137
- transition: "color 120ms ease",
138
- }}
139
- onMouseEnter={(e) => { e.currentTarget.style.color = "var(--accent)"; }}
140
- onMouseLeave={(e) => { e.currentTarget.style.color = "var(--text-secondary)"; }}
141
- >
142
- Show {remaining} more&hellip;
143
- </button>
144
- )}
145
- </div>
146
- </div>
147
- );
148
- }
149
-
150
- export function Board() {
151
- const { total: totalIssues } = useSubscription("all-issues", { limit: 1, offset: 0 });
152
- const { total: totalEpics } = useSubscription("epics", { limit: 1, offset: 0 });
153
-
154
- const navigateToDetail = (id: string) => {
155
- window.location.hash = `#/detail/${id}`;
156
- };
157
-
158
- return (
159
- <div className="p-6" style={{ background: "var(--bg-base)" }}>
160
- <div className="mb-6">
161
- <h1
162
- className="text-xl font-bold"
163
- style={{ color: "var(--text-primary)" }}
164
- >
165
- Board
166
- </h1>
167
- <p className="text-sm mt-0.5" style={{ color: "var(--text-tertiary)" }}>
168
- {totalIssues} issues across {totalEpics} epics
169
- </p>
170
- </div>
171
- <div className="flex gap-4 overflow-x-auto items-start">
172
- <Column
173
- title="Open"
174
- subscriptionType="ready-issues"
175
- statusKey="open"
176
- onCardClick={navigateToDetail}
177
- />
178
- <Column
179
- title="In Progress"
180
- subscriptionType="in-progress-issues"
181
- statusKey="in_progress"
182
- onCardClick={navigateToDetail}
183
- />
184
- <Column
185
- title="Blocked"
186
- subscriptionType="blocked-issues"
187
- statusKey="blocked"
188
- onCardClick={navigateToDetail}
189
- />
190
- <Column
191
- title="Closed"
192
- subscriptionType="closed-issues"
193
- statusKey="closed"
194
- onCardClick={navigateToDetail}
195
- isClosed
196
- />
197
- </div>
198
- </div>
199
- );
200
- }
@@ -1,398 +0,0 @@
1
- import { useState } from "react";
2
- import { useSubscription } from "../hooks/use-subscription";
3
- import { useMutation } from "../hooks/use-mutation";
4
- import { StatusBadge } from "../components/StatusBadge";
5
- import { PriorityBadge } from "../components/PriorityBadge";
6
- import { TypeBadge } from "../components/TypeBadge";
7
- import { SectionEditor } from "../components/SectionEditor";
8
- import { getInitials, getAvatarColor } from "../lib/avatar";
9
- import type { Issue } from "../lib/types";
10
-
11
- function MetadataCard({ label, children }: { label: string; children: React.ReactNode }) {
12
- return (
13
- <div
14
- className="px-3 py-2.5 rounded-md"
15
- style={{ border: "1px solid var(--border-subtle)" }}
16
- >
17
- <h3
18
- className="font-semibold uppercase tracking-wider mb-1.5"
19
- style={{ fontSize: "11px", color: "var(--text-tertiary)" }}
20
- >
21
- {label}
22
- </h3>
23
- {children}
24
- </div>
25
- );
26
- }
27
-
28
- function MetadataSidebar({
29
- issue,
30
- onUpdate,
31
- }: {
32
- issue: Issue;
33
- onUpdate: ReturnType<typeof useMutation>;
34
- }) {
35
- const parentId = issue.parent_id || (issue as any).parent;
36
- const parentTitle = issue.parent_title || (issue as any).parent_title;
37
- const parentStatus = issue.parent_status || (issue as any).parent_status;
38
- const ts = new Date(issue.updated_at).toLocaleDateString();
39
-
40
- return (
41
- <aside
42
- className="shrink-0 p-4 space-y-3 overflow-y-auto"
43
- style={{
44
- width: "280px",
45
- borderLeft: "1px solid var(--border-subtle)",
46
- background: "var(--bg-base)",
47
- }}
48
- >
49
- <MetadataCard label="Status">
50
- <select
51
- value={issue.status}
52
- onChange={(e) =>
53
- onUpdate.updateStatus(
54
- issue.id,
55
- e.target.value as "open" | "in_progress" | "closed",
56
- )
57
- }
58
- className="w-full px-2 py-1 text-sm rounded-md outline-none"
59
- style={{
60
- border: "1px solid var(--border-default)",
61
- background: "var(--bg-elevated)",
62
- color: "var(--text-primary)",
63
- }}
64
- >
65
- <option value="open">Open</option>
66
- <option value="in_progress">In Progress</option>
67
- <option value="closed">Closed</option>
68
- </select>
69
- </MetadataCard>
70
-
71
- <MetadataCard label="Priority">
72
- <select
73
- value={issue.priority}
74
- onChange={(e) =>
75
- onUpdate.updatePriority(issue.id, Number(e.target.value))
76
- }
77
- className="w-full px-2 py-1 text-sm rounded-md outline-none"
78
- style={{
79
- border: "1px solid var(--border-default)",
80
- background: "var(--bg-elevated)",
81
- color: "var(--text-primary)",
82
- }}
83
- >
84
- {[0, 1, 2, 3, 4].map((p) => (
85
- <option key={p} value={p}>P{p}</option>
86
- ))}
87
- </select>
88
- </MetadataCard>
89
-
90
- <MetadataCard label="Type">
91
- <TypeBadge type={issue.issue_type} />
92
- </MetadataCard>
93
-
94
- <MetadataCard label="Assignee">
95
- {issue.assignee ? (
96
- <div className="flex items-center gap-2">
97
- <div
98
- className="flex items-center justify-center rounded-full text-[10px] font-bold shrink-0"
99
- style={{
100
- width: "24px",
101
- height: "24px",
102
- backgroundColor: getAvatarColor(issue.assignee),
103
- color: "var(--text-inverse)",
104
- }}
105
- >
106
- {getInitials(issue.assignee)}
107
- </div>
108
- <span className="text-sm" style={{ color: "var(--text-primary)" }}>{issue.assignee}</span>
109
- </div>
110
- ) : (
111
- <span className="text-sm" style={{ color: "var(--text-tertiary)" }}>Unassigned</span>
112
- )}
113
- </MetadataCard>
114
-
115
- <MetadataCard label="Updated">
116
- <span className="text-sm" style={{ color: "var(--text-secondary)" }}>{ts}</span>
117
- </MetadataCard>
118
-
119
- {issue.labels && issue.labels.length > 0 && (
120
- <MetadataCard label="Labels">
121
- <div className="flex flex-wrap gap-1">
122
- {issue.labels.map((l) => (
123
- <span
124
- key={l}
125
- className="px-2 py-0.5 text-xs rounded"
126
- style={{ background: "var(--bg-hover)", color: "var(--text-secondary)" }}
127
- >
128
- {l}
129
- </span>
130
- ))}
131
- </div>
132
- </MetadataCard>
133
- )}
134
-
135
- {/* Parent */}
136
- {parentId && (
137
- <MetadataCard label="Parent">
138
- <a
139
- href={`#/detail/${parentId}`}
140
- className="block rounded px-1 py-1 -mx-1 transition-colors"
141
- style={{ color: "var(--text-primary)" }}
142
- onMouseEnter={(e) => { e.currentTarget.style.background = "var(--bg-hover)"; }}
143
- onMouseLeave={(e) => { e.currentTarget.style.background = "transparent"; }}
144
- >
145
- <div className="flex items-center gap-1.5 mb-0.5">
146
- {parentStatus && <StatusBadge status={parentStatus} />}
147
- <span className="font-mono text-xs" style={{ color: "var(--text-tertiary)" }}>{parentId}</span>
148
- </div>
149
- {parentTitle && (
150
- <p className="text-xs leading-snug pl-0.5 line-clamp-2" style={{ color: "var(--text-secondary)" }}>
151
- {parentTitle}
152
- </p>
153
- )}
154
- </a>
155
- </MetadataCard>
156
- )}
157
-
158
- {/* Blocked By */}
159
- {(() => {
160
- const blocksDeps = (issue.dependencies || []).filter(
161
- (dep: any) => dep.type === "blocks" && dep.issue_id === issue.id
162
- );
163
- if (blocksDeps.length === 0) return null;
164
- return (
165
- <MetadataCard label="Blocked By">
166
- <div className="space-y-1">
167
- {blocksDeps.map((dep: any) => (
168
- <a
169
- key={dep.depends_on_id}
170
- href={`#/detail/${dep.depends_on_id}`}
171
- className="block text-xs font-mono px-1 py-0.5 transition-colors"
172
- style={{ color: "var(--accent)" }}
173
- onMouseEnter={(e) => { e.currentTarget.style.textDecoration = "underline"; }}
174
- onMouseLeave={(e) => { e.currentTarget.style.textDecoration = "none"; }}
175
- >
176
- {dep.depends_on_id}
177
- </a>
178
- ))}
179
- </div>
180
- </MetadataCard>
181
- );
182
- })()}
183
-
184
- {/* Children / Dependents (sidebar) */}
185
- {issue.dependents && issue.dependents.length > 0 && (
186
- <MetadataCard label="Children / Dependents">
187
- <div className="space-y-1">
188
- {issue.dependents.map((dep) => (
189
- <a
190
- key={dep.id}
191
- href={`#/detail/${dep.id}`}
192
- className="block text-xs rounded px-1 py-1.5 -mx-1 transition-colors"
193
- onMouseEnter={(e) => { e.currentTarget.style.background = "var(--bg-hover)"; }}
194
- onMouseLeave={(e) => { e.currentTarget.style.background = "transparent"; }}
195
- >
196
- <div className="flex items-center gap-1.5 mb-0.5">
197
- <StatusBadge status={dep.status} />
198
- <span className="font-mono" style={{ color: "var(--text-tertiary)" }}>{dep.id}</span>
199
- </div>
200
- {dep.title && (
201
- <p className="text-xs leading-snug pl-0.5 line-clamp-2" style={{ color: "var(--text-secondary)" }}>
202
- {dep.title}
203
- </p>
204
- )}
205
- </a>
206
- ))}
207
- </div>
208
- </MetadataCard>
209
- )}
210
- </aside>
211
- );
212
- }
213
-
214
- function CommentsSection({ issueId }: { issueId: string }) {
215
- const mutations = useMutation();
216
- const [newComment, setNewComment] = useState("");
217
-
218
- const handleAddComment = async () => {
219
- if (!newComment.trim()) return;
220
- await mutations.addComment(issueId, newComment.trim());
221
- setNewComment("");
222
- };
223
-
224
- return (
225
- <div
226
- className="rounded-lg p-4"
227
- style={{
228
- background: "var(--bg-elevated)",
229
- border: "1px solid var(--border-subtle)",
230
- boxShadow: "var(--shadow-card)",
231
- }}
232
- >
233
- <h3
234
- className="font-semibold uppercase tracking-wider mb-3"
235
- style={{ fontSize: "11px", color: "var(--text-tertiary)" }}
236
- >
237
- Comments
238
- </h3>
239
- <div className="flex gap-2">
240
- <textarea
241
- value={newComment}
242
- onChange={(e) => setNewComment(e.target.value)}
243
- placeholder="Add a comment... (Ctrl+Enter to submit)"
244
- rows={3}
245
- className="flex-1 p-2 text-sm rounded-md resize-y outline-none"
246
- style={{
247
- border: "1px solid var(--border-default)",
248
- background: "var(--bg-base)",
249
- color: "var(--text-primary)",
250
- }}
251
- onFocus={(e) => { e.currentTarget.style.borderColor = "var(--accent)"; }}
252
- onBlur={(e) => { e.currentTarget.style.borderColor = "var(--border-default)"; }}
253
- onKeyDown={(e) => {
254
- if (e.key === "Enter" && (e.ctrlKey || e.metaKey))
255
- handleAddComment();
256
- }}
257
- />
258
- </div>
259
- <button
260
- onClick={handleAddComment}
261
- className="mt-2 px-3 py-1.5 text-sm rounded-md font-medium transition-colors"
262
- style={{
263
- background: "var(--bg-hover)",
264
- color: "var(--text-primary)",
265
- border: "1px solid var(--border-default)",
266
- }}
267
- onMouseEnter={(e) => { e.currentTarget.style.borderColor = "var(--accent)"; }}
268
- onMouseLeave={(e) => { e.currentTarget.style.borderColor = "var(--border-default)"; }}
269
- >
270
- Add Comment
271
- </button>
272
- </div>
273
- );
274
- }
275
-
276
- export function Detail({ issueId }: { issueId: string }) {
277
- const { issues, loading } = useSubscription("issue-detail", { id: issueId });
278
- const mutations = useMutation();
279
- const issue = issues[0];
280
-
281
- if (loading) return <div className="p-6" style={{ color: "var(--text-tertiary)" }}>Loading...</div>;
282
- if (!issue)
283
- return <div className="p-6" style={{ color: "var(--text-tertiary)" }}>Issue not found</div>;
284
-
285
- return (
286
- <div className="flex h-full" style={{ background: "var(--bg-base)" }}>
287
- <div className="flex-1 overflow-y-auto p-6 space-y-5">
288
- {/* Breadcrumbs */}
289
- <div className="flex items-center gap-2 text-sm" style={{ color: "var(--text-tertiary)" }}>
290
- <a
291
- href="#/list"
292
- className="flex items-center gap-1 transition-colors"
293
- style={{ color: "var(--text-tertiary)" }}
294
- onMouseEnter={(e) => { e.currentTarget.style.color = "var(--accent)"; }}
295
- onMouseLeave={(e) => { e.currentTarget.style.color = "var(--text-tertiary)"; }}
296
- >
297
- &larr; List
298
- </a>
299
- <span>/</span>
300
- <span className="font-mono">{issue.id}</span>
301
- </div>
302
-
303
- {/* Title */}
304
- <h1 className="text-2xl font-bold" style={{ color: "var(--text-primary)" }}>
305
- {issue.title}
306
- </h1>
307
-
308
- {/* Description */}
309
- <SectionEditor
310
- label="Description"
311
- value={issue.description || ""}
312
- onSave={(v) =>
313
- mutations.editText({
314
- id: issue.id,
315
- field: "description",
316
- value: v,
317
- })
318
- }
319
- />
320
-
321
- {/* Acceptance Criteria */}
322
- <SectionEditor
323
- label="Acceptance Criteria"
324
- value={issue.acceptance || ""}
325
- placeholder="Add acceptance criteria..."
326
- onSave={(v) =>
327
- mutations.editText({
328
- id: issue.id,
329
- field: "acceptance",
330
- value: v,
331
- })
332
- }
333
- />
334
-
335
- {/* Notes */}
336
- <SectionEditor
337
- label="Notes"
338
- value={issue.notes || ""}
339
- placeholder="Add notes..."
340
- onSave={(v) =>
341
- mutations.editText({ id: issue.id, field: "notes", value: v })
342
- }
343
- />
344
-
345
- {/* Design */}
346
- <SectionEditor
347
- label="Design"
348
- value={issue.design || ""}
349
- placeholder="Add design notes..."
350
- onSave={(v) =>
351
- mutations.editText({ id: issue.id, field: "design", value: v })
352
- }
353
- />
354
-
355
- {/* Children (epics only) */}
356
- {issue.issue_type === "epic" && issue.dependents && issue.dependents.length > 0 && (
357
- <div
358
- className="rounded-lg p-4"
359
- style={{
360
- background: "var(--bg-elevated)",
361
- border: "1px solid var(--border-subtle)",
362
- boxShadow: "var(--shadow-card)",
363
- }}
364
- >
365
- <h3
366
- className="font-semibold uppercase tracking-wider mb-3"
367
- style={{ fontSize: "11px", color: "var(--text-tertiary)" }}
368
- >
369
- Children ({issue.closed_children ?? 0} / {issue.total_children ?? issue.dependents.length})
370
- </h3>
371
- <div className="space-y-0.5">
372
- {issue.dependents.map((child) => (
373
- <a
374
- key={child.id}
375
- href={`#/detail/${child.id}`}
376
- className="flex items-center gap-2.5 px-2 py-2 rounded-md transition-colors"
377
- style={{ opacity: child.status === "closed" ? 0.6 : 1 }}
378
- onMouseEnter={(e) => { e.currentTarget.style.background = "var(--bg-hover)"; }}
379
- onMouseLeave={(e) => { e.currentTarget.style.background = "transparent"; }}
380
- >
381
- <StatusBadge status={child.status} />
382
- <span className="text-xs font-mono" style={{ color: "var(--text-tertiary)" }}>{child.id}</span>
383
- <span className="text-sm truncate" style={{ color: "var(--text-primary)" }}>{child.title}</span>
384
- </a>
385
- ))}
386
- </div>
387
- </div>
388
- )}
389
-
390
- {/* Comments */}
391
- <CommentsSection issueId={issue.id} />
392
- </div>
393
-
394
- {/* Metadata sidebar */}
395
- <MetadataSidebar issue={issue} onUpdate={mutations} />
396
- </div>
397
- );
398
- }