@treenity/mods 3.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (486) hide show
  1. package/board/board.test.ts +212 -0
  2. package/board/client.ts +1 -0
  3. package/board/seed.ts +26 -0
  4. package/board/server.ts +5 -0
  5. package/board/types.ts +87 -0
  6. package/board/view.tsx +574 -0
  7. package/brahman/CLAUDE.md +18 -0
  8. package/brahman/brahman.test.ts +855 -0
  9. package/brahman/client.ts +2 -0
  10. package/brahman/helpers.ts +374 -0
  11. package/brahman/server.ts +2 -0
  12. package/brahman/service.ts +328 -0
  13. package/brahman/types.ts +727 -0
  14. package/brahman/view.tsx +73 -0
  15. package/brahman/views/action-cards.tsx +615 -0
  16. package/brahman/views/bot-view.tsx +76 -0
  17. package/brahman/views/chat-editor.tsx +782 -0
  18. package/brahman/views/chat-preview.tsx +266 -0
  19. package/brahman/views/menu-editor.tsx +451 -0
  20. package/brahman/views/page-layout.tsx +285 -0
  21. package/brahman/views/tstring-input.tsx +84 -0
  22. package/cafe/CLAUDE.md +7 -0
  23. package/cafe/seed.ts +8 -0
  24. package/cafe/server.ts +2 -0
  25. package/cafe/types.ts +32 -0
  26. package/canary/seed.ts +8 -0
  27. package/canary/server.ts +3 -0
  28. package/canary/service.ts +101 -0
  29. package/canary/types.ts +16 -0
  30. package/dist/board/client.d.ts +2 -0
  31. package/dist/board/client.d.ts.map +1 -0
  32. package/dist/board/client.js +2 -0
  33. package/dist/board/client.js.map +1 -0
  34. package/dist/board/seed.d.ts +2 -0
  35. package/dist/board/seed.d.ts.map +1 -0
  36. package/dist/board/seed.js +23 -0
  37. package/dist/board/seed.js.map +1 -0
  38. package/dist/board/server.d.ts +3 -0
  39. package/dist/board/server.d.ts.map +1 -0
  40. package/dist/board/server.js +5 -0
  41. package/dist/board/server.js.map +1 -0
  42. package/dist/board/types.d.ts +45 -0
  43. package/dist/board/types.d.ts.map +1 -0
  44. package/dist/board/types.js +82 -0
  45. package/dist/board/types.js.map +1 -0
  46. package/dist/board/view.d.ts +2 -0
  47. package/dist/board/view.d.ts.map +1 -0
  48. package/dist/board/view.js +254 -0
  49. package/dist/board/view.js.map +1 -0
  50. package/dist/brahman/client.d.ts +3 -0
  51. package/dist/brahman/client.d.ts.map +1 -0
  52. package/dist/brahman/client.js +3 -0
  53. package/dist/brahman/client.js.map +1 -0
  54. package/dist/brahman/helpers.d.ts +51 -0
  55. package/dist/brahman/helpers.d.ts.map +1 -0
  56. package/dist/brahman/helpers.js +321 -0
  57. package/dist/brahman/helpers.js.map +1 -0
  58. package/dist/brahman/server.d.ts +3 -0
  59. package/dist/brahman/server.d.ts.map +1 -0
  60. package/dist/brahman/server.js +3 -0
  61. package/dist/brahman/server.js.map +1 -0
  62. package/dist/brahman/service.d.ts +2 -0
  63. package/dist/brahman/service.d.ts.map +1 -0
  64. package/dist/brahman/service.js +310 -0
  65. package/dist/brahman/service.js.map +1 -0
  66. package/dist/brahman/types.d.ts +335 -0
  67. package/dist/brahman/types.d.ts.map +1 -0
  68. package/dist/brahman/types.js +633 -0
  69. package/dist/brahman/types.js.map +1 -0
  70. package/dist/brahman/view.d.ts +2 -0
  71. package/dist/brahman/view.d.ts.map +1 -0
  72. package/dist/brahman/view.js +47 -0
  73. package/dist/brahman/view.js.map +1 -0
  74. package/dist/brahman/views/action-cards.d.ts +60 -0
  75. package/dist/brahman/views/action-cards.d.ts.map +1 -0
  76. package/dist/brahman/views/action-cards.js +283 -0
  77. package/dist/brahman/views/action-cards.js.map +1 -0
  78. package/dist/brahman/views/bot-view.d.ts +5 -0
  79. package/dist/brahman/views/bot-view.d.ts.map +1 -0
  80. package/dist/brahman/views/bot-view.js +14 -0
  81. package/dist/brahman/views/bot-view.js.map +1 -0
  82. package/dist/brahman/views/chat-editor.d.ts +5 -0
  83. package/dist/brahman/views/chat-editor.d.ts.map +1 -0
  84. package/dist/brahman/views/chat-editor.js +306 -0
  85. package/dist/brahman/views/chat-editor.js.map +1 -0
  86. package/dist/brahman/views/chat-preview.d.ts +5 -0
  87. package/dist/brahman/views/chat-preview.d.ts.map +1 -0
  88. package/dist/brahman/views/chat-preview.js +145 -0
  89. package/dist/brahman/views/chat-preview.js.map +1 -0
  90. package/dist/brahman/views/menu-editor.d.ts +11 -0
  91. package/dist/brahman/views/menu-editor.d.ts.map +1 -0
  92. package/dist/brahman/views/menu-editor.js +171 -0
  93. package/dist/brahman/views/menu-editor.js.map +1 -0
  94. package/dist/brahman/views/page-layout.d.ts +5 -0
  95. package/dist/brahman/views/page-layout.d.ts.map +1 -0
  96. package/dist/brahman/views/page-layout.js +114 -0
  97. package/dist/brahman/views/page-layout.js.map +1 -0
  98. package/dist/brahman/views/tstring-input.d.ts +15 -0
  99. package/dist/brahman/views/tstring-input.d.ts.map +1 -0
  100. package/dist/brahman/views/tstring-input.js +23 -0
  101. package/dist/brahman/views/tstring-input.js.map +1 -0
  102. package/dist/cafe/seed.d.ts +2 -0
  103. package/dist/cafe/seed.d.ts.map +1 -0
  104. package/dist/cafe/seed.js +7 -0
  105. package/dist/cafe/seed.js.map +1 -0
  106. package/dist/cafe/server.d.ts +3 -0
  107. package/dist/cafe/server.d.ts.map +1 -0
  108. package/dist/cafe/server.js +3 -0
  109. package/dist/cafe/server.js.map +1 -0
  110. package/dist/cafe/types.d.ts +2 -0
  111. package/dist/cafe/types.d.ts.map +1 -0
  112. package/dist/cafe/types.js +31 -0
  113. package/dist/cafe/types.js.map +1 -0
  114. package/dist/canary/seed.d.ts +2 -0
  115. package/dist/canary/seed.d.ts.map +1 -0
  116. package/dist/canary/seed.js +7 -0
  117. package/dist/canary/seed.js.map +1 -0
  118. package/dist/canary/server.d.ts +4 -0
  119. package/dist/canary/server.d.ts.map +1 -0
  120. package/dist/canary/server.js +4 -0
  121. package/dist/canary/server.js.map +1 -0
  122. package/dist/canary/service.d.ts +2 -0
  123. package/dist/canary/service.d.ts.map +1 -0
  124. package/dist/canary/service.js +101 -0
  125. package/dist/canary/service.js.map +1 -0
  126. package/dist/canary/types.d.ts +9 -0
  127. package/dist/canary/types.d.ts.map +1 -0
  128. package/dist/canary/types.js +13 -0
  129. package/dist/canary/types.js.map +1 -0
  130. package/dist/doc/client.d.ts +3 -0
  131. package/dist/doc/client.d.ts.map +1 -0
  132. package/dist/doc/client.js +5 -0
  133. package/dist/doc/client.js.map +1 -0
  134. package/dist/doc/fs-codec.d.ts +2 -0
  135. package/dist/doc/fs-codec.d.ts.map +1 -0
  136. package/dist/doc/fs-codec.js +44 -0
  137. package/dist/doc/fs-codec.js.map +1 -0
  138. package/dist/doc/markdown.d.ts +13 -0
  139. package/dist/doc/markdown.d.ts.map +1 -0
  140. package/dist/doc/markdown.js +250 -0
  141. package/dist/doc/markdown.js.map +1 -0
  142. package/dist/doc/prefab.d.ts +2 -0
  143. package/dist/doc/prefab.d.ts.map +1 -0
  144. package/dist/doc/prefab.js +23 -0
  145. package/dist/doc/prefab.js.map +1 -0
  146. package/dist/doc/renderers.d.ts +2 -0
  147. package/dist/doc/renderers.d.ts.map +1 -0
  148. package/dist/doc/renderers.js +94 -0
  149. package/dist/doc/renderers.js.map +1 -0
  150. package/dist/doc/seed.d.ts +2 -0
  151. package/dist/doc/seed.d.ts.map +1 -0
  152. package/dist/doc/seed.js +9 -0
  153. package/dist/doc/seed.js.map +1 -0
  154. package/dist/doc/server.d.ts +6 -0
  155. package/dist/doc/server.d.ts.map +1 -0
  156. package/dist/doc/server.js +6 -0
  157. package/dist/doc/server.js.map +1 -0
  158. package/dist/doc/slash-command.d.ts +3 -0
  159. package/dist/doc/slash-command.d.ts.map +1 -0
  160. package/dist/doc/slash-command.js +123 -0
  161. package/dist/doc/slash-command.js.map +1 -0
  162. package/dist/doc/slash-menu.d.ts +15 -0
  163. package/dist/doc/slash-menu.d.ts.map +1 -0
  164. package/dist/doc/slash-menu.js +54 -0
  165. package/dist/doc/slash-menu.js.map +1 -0
  166. package/dist/doc/text.d.ts +2 -0
  167. package/dist/doc/text.d.ts.map +1 -0
  168. package/dist/doc/text.js +22 -0
  169. package/dist/doc/text.js.map +1 -0
  170. package/dist/doc/toolbar.d.ts +5 -0
  171. package/dist/doc/toolbar.d.ts.map +1 -0
  172. package/dist/doc/toolbar.js +15 -0
  173. package/dist/doc/toolbar.js.map +1 -0
  174. package/dist/doc/treenity-block-view.d.ts +2 -0
  175. package/dist/doc/treenity-block-view.d.ts.map +1 -0
  176. package/dist/doc/treenity-block-view.js +37 -0
  177. package/dist/doc/treenity-block-view.js.map +1 -0
  178. package/dist/doc/treenity-block.d.ts +3 -0
  179. package/dist/doc/treenity-block.d.ts.map +1 -0
  180. package/dist/doc/treenity-block.js +27 -0
  181. package/dist/doc/treenity-block.js.map +1 -0
  182. package/dist/doc/types.d.ts +2 -0
  183. package/dist/doc/types.d.ts.map +1 -0
  184. package/dist/doc/types.js +10 -0
  185. package/dist/doc/types.js.map +1 -0
  186. package/dist/launcher/client.d.ts +5 -0
  187. package/dist/launcher/client.d.ts.map +1 -0
  188. package/dist/launcher/client.js +5 -0
  189. package/dist/launcher/client.js.map +1 -0
  190. package/dist/launcher/icons.d.ts +2 -0
  191. package/dist/launcher/icons.d.ts.map +1 -0
  192. package/dist/launcher/icons.js +57 -0
  193. package/dist/launcher/icons.js.map +1 -0
  194. package/dist/launcher/seed.d.ts +2 -0
  195. package/dist/launcher/seed.d.ts.map +1 -0
  196. package/dist/launcher/seed.js +36 -0
  197. package/dist/launcher/seed.js.map +1 -0
  198. package/dist/launcher/server.d.ts +3 -0
  199. package/dist/launcher/server.d.ts.map +1 -0
  200. package/dist/launcher/server.js +3 -0
  201. package/dist/launcher/server.js.map +1 -0
  202. package/dist/launcher/types.d.ts +21 -0
  203. package/dist/launcher/types.d.ts.map +1 -0
  204. package/dist/launcher/types.js +47 -0
  205. package/dist/launcher/types.js.map +1 -0
  206. package/dist/launcher/view.d.ts +2 -0
  207. package/dist/launcher/view.d.ts.map +1 -0
  208. package/dist/launcher/view.js +187 -0
  209. package/dist/launcher/view.js.map +1 -0
  210. package/dist/launcher/widgets.d.ts +2 -0
  211. package/dist/launcher/widgets.d.ts.map +1 -0
  212. package/dist/launcher/widgets.js +73 -0
  213. package/dist/launcher/widgets.js.map +1 -0
  214. package/dist/mindmap/branch.d.ts +17 -0
  215. package/dist/mindmap/branch.d.ts.map +1 -0
  216. package/dist/mindmap/branch.js +47 -0
  217. package/dist/mindmap/branch.js.map +1 -0
  218. package/dist/mindmap/client.d.ts +3 -0
  219. package/dist/mindmap/client.d.ts.map +1 -0
  220. package/dist/mindmap/client.js +3 -0
  221. package/dist/mindmap/client.js.map +1 -0
  222. package/dist/mindmap/radial-tree.d.ts +14 -0
  223. package/dist/mindmap/radial-tree.d.ts.map +1 -0
  224. package/dist/mindmap/radial-tree.js +184 -0
  225. package/dist/mindmap/radial-tree.js.map +1 -0
  226. package/dist/mindmap/sidebar.d.ts +8 -0
  227. package/dist/mindmap/sidebar.d.ts.map +1 -0
  228. package/dist/mindmap/sidebar.js +24 -0
  229. package/dist/mindmap/sidebar.js.map +1 -0
  230. package/dist/mindmap/types.d.ts +8 -0
  231. package/dist/mindmap/types.d.ts.map +1 -0
  232. package/dist/mindmap/types.js +10 -0
  233. package/dist/mindmap/types.js.map +1 -0
  234. package/dist/mindmap/use-tree-data.d.ts +14 -0
  235. package/dist/mindmap/use-tree-data.d.ts.map +1 -0
  236. package/dist/mindmap/use-tree-data.js +95 -0
  237. package/dist/mindmap/use-tree-data.js.map +1 -0
  238. package/dist/mindmap/view.d.ts +3 -0
  239. package/dist/mindmap/view.d.ts.map +1 -0
  240. package/dist/mindmap/view.js +141 -0
  241. package/dist/mindmap/view.js.map +1 -0
  242. package/dist/sensor-demo/client.d.ts +2 -0
  243. package/dist/sensor-demo/client.d.ts.map +1 -0
  244. package/dist/sensor-demo/client.js +2 -0
  245. package/dist/sensor-demo/client.js.map +1 -0
  246. package/dist/sensor-demo/server.d.ts +2 -0
  247. package/dist/sensor-demo/server.d.ts.map +1 -0
  248. package/dist/sensor-demo/server.js +2 -0
  249. package/dist/sensor-demo/server.js.map +1 -0
  250. package/dist/sensor-demo/service.d.ts +2 -0
  251. package/dist/sensor-demo/service.d.ts.map +1 -0
  252. package/dist/sensor-demo/service.js +27 -0
  253. package/dist/sensor-demo/service.js.map +1 -0
  254. package/dist/sensor-demo/types.d.ts +15 -0
  255. package/dist/sensor-demo/types.d.ts.map +1 -0
  256. package/dist/sensor-demo/types.js +18 -0
  257. package/dist/sensor-demo/types.js.map +1 -0
  258. package/dist/sensor-demo/view.d.ts +2 -0
  259. package/dist/sensor-demo/view.d.ts.map +1 -0
  260. package/dist/sensor-demo/view.js +33 -0
  261. package/dist/sensor-demo/view.js.map +1 -0
  262. package/dist/sensor-generator/action.d.ts +2 -0
  263. package/dist/sensor-generator/action.d.ts.map +1 -0
  264. package/dist/sensor-generator/action.js +23 -0
  265. package/dist/sensor-generator/action.js.map +1 -0
  266. package/dist/sensor-generator/client.d.ts +2 -0
  267. package/dist/sensor-generator/client.d.ts.map +1 -0
  268. package/dist/sensor-generator/client.js +2 -0
  269. package/dist/sensor-generator/client.js.map +1 -0
  270. package/dist/sensor-generator/server.d.ts +2 -0
  271. package/dist/sensor-generator/server.d.ts.map +1 -0
  272. package/dist/sensor-generator/server.js +2 -0
  273. package/dist/sensor-generator/server.js.map +1 -0
  274. package/dist/sensor-generator/view.d.ts +2 -0
  275. package/dist/sensor-generator/view.d.ts.map +1 -0
  276. package/dist/sensor-generator/view.js +64 -0
  277. package/dist/sensor-generator/view.js.map +1 -0
  278. package/dist/sim/client.d.ts +2 -0
  279. package/dist/sim/client.d.ts.map +1 -0
  280. package/dist/sim/client.js +2 -0
  281. package/dist/sim/client.js.map +1 -0
  282. package/dist/sim/seed.d.ts +2 -0
  283. package/dist/sim/seed.d.ts.map +1 -0
  284. package/dist/sim/seed.js +50 -0
  285. package/dist/sim/seed.js.map +1 -0
  286. package/dist/sim/server.d.ts +3 -0
  287. package/dist/sim/server.d.ts.map +1 -0
  288. package/dist/sim/server.js +3 -0
  289. package/dist/sim/server.js.map +1 -0
  290. package/dist/sim/service.d.ts +4 -0
  291. package/dist/sim/service.d.ts.map +1 -0
  292. package/dist/sim/service.js +528 -0
  293. package/dist/sim/service.js.map +1 -0
  294. package/dist/sim/types.d.ts +63 -0
  295. package/dist/sim/types.d.ts.map +1 -0
  296. package/dist/sim/types.js +57 -0
  297. package/dist/sim/types.js.map +1 -0
  298. package/dist/sim/view.d.ts +2 -0
  299. package/dist/sim/view.d.ts.map +1 -0
  300. package/dist/sim/view.js +205 -0
  301. package/dist/sim/view.js.map +1 -0
  302. package/dist/table/client.d.ts +4 -0
  303. package/dist/table/client.d.ts.map +1 -0
  304. package/dist/table/client.js +4 -0
  305. package/dist/table/client.js.map +1 -0
  306. package/dist/table/edit.d.ts +2 -0
  307. package/dist/table/edit.d.ts.map +1 -0
  308. package/dist/table/edit.js +115 -0
  309. package/dist/table/edit.js.map +1 -0
  310. package/dist/table/server.d.ts +2 -0
  311. package/dist/table/server.d.ts.map +1 -0
  312. package/dist/table/server.js +2 -0
  313. package/dist/table/server.js.map +1 -0
  314. package/dist/table/types.d.ts +18 -0
  315. package/dist/table/types.d.ts.map +1 -0
  316. package/dist/table/types.js +13 -0
  317. package/dist/table/types.js.map +1 -0
  318. package/dist/table/use-debounced-sync.d.ts +8 -0
  319. package/dist/table/use-debounced-sync.d.ts.map +1 -0
  320. package/dist/table/use-debounced-sync.js +56 -0
  321. package/dist/table/use-debounced-sync.js.map +1 -0
  322. package/dist/table/view.d.ts +2 -0
  323. package/dist/table/view.d.ts.map +1 -0
  324. package/dist/table/view.js +199 -0
  325. package/dist/table/view.js.map +1 -0
  326. package/dist/tasks/server.d.ts +2 -0
  327. package/dist/tasks/server.d.ts.map +1 -0
  328. package/dist/tasks/server.js +2 -0
  329. package/dist/tasks/server.js.map +1 -0
  330. package/dist/tasks/types.d.ts +22 -0
  331. package/dist/tasks/types.d.ts.map +1 -0
  332. package/dist/tasks/types.js +34 -0
  333. package/dist/tasks/types.js.map +1 -0
  334. package/dist/three/client.d.ts +2 -0
  335. package/dist/three/client.d.ts.map +1 -0
  336. package/dist/three/client.js +2 -0
  337. package/dist/three/client.js.map +1 -0
  338. package/dist/three/seed.d.ts +2 -0
  339. package/dist/three/seed.d.ts.map +1 -0
  340. package/dist/three/seed.js +45 -0
  341. package/dist/three/seed.js.map +1 -0
  342. package/dist/three/server.d.ts +3 -0
  343. package/dist/three/server.d.ts.map +1 -0
  344. package/dist/three/server.js +3 -0
  345. package/dist/three/server.js.map +1 -0
  346. package/dist/three/types.d.ts +178 -0
  347. package/dist/three/types.d.ts.map +1 -0
  348. package/dist/three/types.js +209 -0
  349. package/dist/three/types.js.map +1 -0
  350. package/dist/three/view.d.ts +2 -0
  351. package/dist/three/view.d.ts.map +1 -0
  352. package/dist/three/view.js +307 -0
  353. package/dist/three/view.js.map +1 -0
  354. package/dist/todo/client.d.ts +3 -0
  355. package/dist/todo/client.d.ts.map +1 -0
  356. package/dist/todo/client.js +3 -0
  357. package/dist/todo/client.js.map +1 -0
  358. package/dist/todo/seed.d.ts +2 -0
  359. package/dist/todo/seed.d.ts.map +1 -0
  360. package/dist/todo/seed.js +8 -0
  361. package/dist/todo/seed.js.map +1 -0
  362. package/dist/todo/server.d.ts +3 -0
  363. package/dist/todo/server.d.ts.map +1 -0
  364. package/dist/todo/server.js +3 -0
  365. package/dist/todo/server.js.map +1 -0
  366. package/dist/todo/types.d.ts +15 -0
  367. package/dist/todo/types.d.ts.map +1 -0
  368. package/dist/todo/types.js +29 -0
  369. package/dist/todo/types.js.map +1 -0
  370. package/dist/todo/view.d.ts +2 -0
  371. package/dist/todo/view.d.ts.map +1 -0
  372. package/dist/todo/view.js +27 -0
  373. package/dist/todo/view.js.map +1 -0
  374. package/dist/whisper/client.d.ts +2 -0
  375. package/dist/whisper/client.d.ts.map +1 -0
  376. package/dist/whisper/client.js +2 -0
  377. package/dist/whisper/client.js.map +1 -0
  378. package/dist/whisper/inbox.d.ts +2 -0
  379. package/dist/whisper/inbox.d.ts.map +1 -0
  380. package/dist/whisper/inbox.js +50 -0
  381. package/dist/whisper/inbox.js.map +1 -0
  382. package/dist/whisper/route.d.ts +10 -0
  383. package/dist/whisper/route.d.ts.map +1 -0
  384. package/dist/whisper/route.js +154 -0
  385. package/dist/whisper/route.js.map +1 -0
  386. package/dist/whisper/seed.d.ts +2 -0
  387. package/dist/whisper/seed.d.ts.map +1 -0
  388. package/dist/whisper/seed.js +14 -0
  389. package/dist/whisper/seed.js.map +1 -0
  390. package/dist/whisper/server.d.ts +5 -0
  391. package/dist/whisper/server.d.ts.map +1 -0
  392. package/dist/whisper/server.js +5 -0
  393. package/dist/whisper/server.js.map +1 -0
  394. package/dist/whisper/service.d.ts +2 -0
  395. package/dist/whisper/service.d.ts.map +1 -0
  396. package/dist/whisper/service.js +26 -0
  397. package/dist/whisper/service.js.map +1 -0
  398. package/dist/whisper/types.d.ts +38 -0
  399. package/dist/whisper/types.d.ts.map +1 -0
  400. package/dist/whisper/types.js +45 -0
  401. package/dist/whisper/types.js.map +1 -0
  402. package/dist/whisper/view.d.ts +2 -0
  403. package/dist/whisper/view.d.ts.map +1 -0
  404. package/dist/whisper/view.js +40 -0
  405. package/dist/whisper/view.js.map +1 -0
  406. package/doc/CLAUDE.md +49 -0
  407. package/doc/client.ts +5 -0
  408. package/doc/editor.css +283 -0
  409. package/doc/fs-codec.test.ts +119 -0
  410. package/doc/fs-codec.ts +50 -0
  411. package/doc/markdown.test.ts +152 -0
  412. package/doc/markdown.ts +284 -0
  413. package/doc/prefab.ts +26 -0
  414. package/doc/renderers.tsx +126 -0
  415. package/doc/seed.ts +10 -0
  416. package/doc/server.ts +5 -0
  417. package/doc/slash-command.ts +136 -0
  418. package/doc/slash-menu.tsx +91 -0
  419. package/doc/text.ts +20 -0
  420. package/doc/toolbar.tsx +86 -0
  421. package/doc/treenity-block-view.tsx +116 -0
  422. package/doc/treenity-block.ts +31 -0
  423. package/doc/types.ts +10 -0
  424. package/launcher/client.ts +4 -0
  425. package/launcher/icons.tsx +234 -0
  426. package/launcher/launcher.css +64 -0
  427. package/launcher/seed.ts +41 -0
  428. package/launcher/server.ts +2 -0
  429. package/launcher/types.ts +53 -0
  430. package/launcher/view.tsx +401 -0
  431. package/launcher/widgets.tsx +171 -0
  432. package/mindmap/branch.tsx +163 -0
  433. package/mindmap/client.ts +2 -0
  434. package/mindmap/mindmap.css +243 -0
  435. package/mindmap/sidebar.tsx +127 -0
  436. package/mindmap/types.ts +10 -0
  437. package/mindmap/view.tsx +247 -0
  438. package/package.json +75 -0
  439. package/sensor-demo/CLAUDE.md +3 -0
  440. package/sensor-demo/client.ts +1 -0
  441. package/sensor-demo/server.ts +1 -0
  442. package/sensor-demo/service.ts +27 -0
  443. package/sensor-demo/types.ts +20 -0
  444. package/sensor-demo/view.tsx +64 -0
  445. package/sensor-generator/CLAUDE.md +3 -0
  446. package/sensor-generator/action.ts +28 -0
  447. package/sensor-generator/client.ts +1 -0
  448. package/sensor-generator/server.ts +1 -0
  449. package/sensor-generator/view.tsx +107 -0
  450. package/sim/CLAUDE.md +16 -0
  451. package/sim/client.ts +1 -0
  452. package/sim/seed.ts +55 -0
  453. package/sim/server.ts +2 -0
  454. package/sim/service.ts +594 -0
  455. package/sim/sim.test.ts +282 -0
  456. package/sim/types.ts +87 -0
  457. package/sim/view.tsx +446 -0
  458. package/table/client.ts +3 -0
  459. package/table/edit.tsx +241 -0
  460. package/table/server.ts +1 -0
  461. package/table/types.ts +22 -0
  462. package/table/use-debounced-sync.ts +67 -0
  463. package/table/view.tsx +339 -0
  464. package/tasks/CLAUDE.md +6 -0
  465. package/tasks/server.ts +1 -0
  466. package/tasks/types.ts +36 -0
  467. package/three/CLAUDE.md +6 -0
  468. package/three/client.ts +1 -0
  469. package/three/seed.ts +54 -0
  470. package/three/server.ts +2 -0
  471. package/three/types.ts +238 -0
  472. package/three/view.tsx +453 -0
  473. package/todo/client.ts +2 -0
  474. package/todo/seed.ts +9 -0
  475. package/todo/server.ts +2 -0
  476. package/todo/types.ts +32 -0
  477. package/todo/view.tsx +67 -0
  478. package/whisper/CLAUDE.md +16 -0
  479. package/whisper/client.ts +1 -0
  480. package/whisper/inbox.ts +54 -0
  481. package/whisper/route.ts +182 -0
  482. package/whisper/seed.ts +15 -0
  483. package/whisper/server.ts +4 -0
  484. package/whisper/service.ts +29 -0
  485. package/whisper/types.ts +51 -0
  486. package/whisper/view.tsx +81 -0
@@ -0,0 +1,284 @@
1
+ // Markdown ↔ Tiptap JSON converters
2
+ // Used by: text.ts (Tiptap→md), fs-codec decode/encode (.md files ↔ doc.page nodes)
3
+
4
+ export type TiptapNode = {
5
+ type: string;
6
+ attrs?: Record<string, unknown>;
7
+ content?: TiptapNode[];
8
+ text?: string;
9
+ marks?: { type: string }[];
10
+ };
11
+
12
+ // ── Tiptap JSON → Markdown ──
13
+
14
+ export function inlineToMd(node: TiptapNode): string {
15
+ if (node.text) {
16
+ let t = node.text;
17
+ if (node.marks?.some((m) => m.type === 'bold')) t = `**${t}**`;
18
+ if (node.marks?.some((m) => m.type === 'italic')) t = `*${t}*`;
19
+ if (node.marks?.some((m) => m.type === 'code')) t = `\`${t}\``;
20
+ return t;
21
+ }
22
+ return (node.content ?? []).map(inlineToMd).join('');
23
+ }
24
+
25
+ export function tiptapToMd(node: TiptapNode): string {
26
+ const children = node.content ?? [];
27
+
28
+ switch (node.type) {
29
+ case 'doc':
30
+ return children.map((c) => tiptapToMd(c)).join('\n\n');
31
+
32
+ case 'heading': {
33
+ const level = (node.attrs?.level as number) ?? 1;
34
+ return '#'.repeat(level) + ' ' + children.map(inlineToMd).join('');
35
+ }
36
+
37
+ case 'paragraph':
38
+ return children.map(inlineToMd).join('');
39
+
40
+ case 'bulletList':
41
+ return children.map((c) => tiptapToMd(c)).join('\n');
42
+
43
+ case 'orderedList':
44
+ return children.map((c, i) => `${i + 1}. ${tiptapToMd(c).replace(/^- /, '')}`).join('\n');
45
+
46
+ case 'taskList':
47
+ return children.map((c) => tiptapToMd(c)).join('\n');
48
+
49
+ case 'taskItem': {
50
+ const checked = node.attrs?.checked ? 'x' : ' ';
51
+ const inner = children.map((c) => tiptapToMd(c)).join('\n');
52
+ return `- [${checked}] ${inner}`;
53
+ }
54
+
55
+ case 'listItem': {
56
+ const inner = children.map((c) => tiptapToMd(c)).join('\n');
57
+ return '- ' + inner;
58
+ }
59
+
60
+ case 'table': {
61
+ const rows = children;
62
+ if (!rows.length) return '';
63
+ const mdRows = rows.map((row) => {
64
+ const cells = (row.content ?? []).map((cell) => {
65
+ const text = (cell.content ?? []).map((c) => tiptapToMd(c)).join('').replace(/\|/g, '\\|');
66
+ return text.trim();
67
+ });
68
+ return '| ' + cells.join(' | ') + ' |';
69
+ });
70
+ // Insert separator after header row
71
+ const firstRow = rows[0];
72
+ const colCount = (firstRow.content ?? []).length;
73
+ const sep = '| ' + Array(colCount).fill('---').join(' | ') + ' |';
74
+ return [mdRows[0], sep, ...mdRows.slice(1)].join('\n');
75
+ }
76
+
77
+ case 'tableRow':
78
+ case 'tableCell':
79
+ case 'tableHeader':
80
+ return children.map((c) => tiptapToMd(c)).join('');
81
+
82
+ case 'codeBlock': {
83
+ const lang = (node.attrs?.language as string) ?? '';
84
+ return '```' + lang + '\n' + children.map(inlineToMd).join('') + '\n```';
85
+ }
86
+
87
+ case 'blockquote':
88
+ return children.map((c) => '> ' + tiptapToMd(c)).join('\n');
89
+
90
+ case 'horizontalRule':
91
+ return '---';
92
+
93
+ case 'treenityBlock': {
94
+ const ref = node.attrs?.ref as string | null;
95
+ const type = node.attrs?.type as string | null;
96
+ if (ref) return `[Component: ${type ?? 'unknown'} at ${ref}]`;
97
+ return `[Component: ${type ?? 'unknown'} (inline)]`;
98
+ }
99
+
100
+ default:
101
+ return children.map((c) => tiptapToMd(c)).join('');
102
+ }
103
+ }
104
+
105
+ // ── Markdown → Tiptap JSON ──
106
+
107
+ export function mdToTiptap(markdown: string): TiptapNode {
108
+ const lines = markdown.split('\n');
109
+ const blocks: TiptapNode[] = [];
110
+ let i = 0;
111
+
112
+ while (i < lines.length) {
113
+ const line = lines[i];
114
+
115
+ // Empty line → skip
116
+ if (!line.trim()) { i++; continue; }
117
+
118
+ // Heading
119
+ const headingMatch = line.match(/^(#{1,6})\s+(.+)$/);
120
+ if (headingMatch) {
121
+ blocks.push({
122
+ type: 'heading',
123
+ attrs: { level: headingMatch[1].length },
124
+ content: parseInline(headingMatch[2]),
125
+ });
126
+ i++;
127
+ continue;
128
+ }
129
+
130
+ // Horizontal rule
131
+ if (/^---+$/.test(line.trim()) || /^\*\*\*+$/.test(line.trim())) {
132
+ blocks.push({ type: 'horizontalRule' });
133
+ i++;
134
+ continue;
135
+ }
136
+
137
+ // Code block
138
+ if (line.trimStart().startsWith('```')) {
139
+ const lang = line.trim().slice(3).trim();
140
+ const codeLines: string[] = [];
141
+ i++;
142
+ while (i < lines.length && !lines[i].trimStart().startsWith('```')) {
143
+ codeLines.push(lines[i]);
144
+ i++;
145
+ }
146
+ i++; // skip closing ```
147
+ blocks.push({
148
+ type: 'codeBlock',
149
+ attrs: lang ? { language: lang } : {},
150
+ content: [{ type: 'text', text: codeLines.join('\n') }],
151
+ });
152
+ continue;
153
+ }
154
+
155
+ // Task list: - [ ] or - [x]
156
+ if (/^[-*]\s\[[x ]\]\s/i.test(line.trimStart())) {
157
+ const items: TiptapNode[] = [];
158
+ while (i < lines.length && /^[-*]\s\[[x ]\]\s/i.test(lines[i].trimStart())) {
159
+ const m = lines[i].trimStart().match(/^[-*]\s\[([x ])\]\s+(.*)/i)!;
160
+ const checked = m[1].toLowerCase() === 'x';
161
+ items.push({
162
+ type: 'taskItem',
163
+ attrs: { checked },
164
+ content: [{ type: 'paragraph', content: parseInline(m[2]) }],
165
+ });
166
+ i++;
167
+ }
168
+ blocks.push({ type: 'taskList', content: items });
169
+ continue;
170
+ }
171
+
172
+ // Bullet list
173
+ if (/^[-*]\s/.test(line.trimStart())) {
174
+ const items: TiptapNode[] = [];
175
+ while (i < lines.length && /^[-*]\s/.test(lines[i].trimStart()) && !/^[-*]\s\[[x ]\]\s/i.test(lines[i].trimStart())) {
176
+ const text = lines[i].trimStart().replace(/^[-*]\s+/, '');
177
+ items.push({
178
+ type: 'listItem',
179
+ content: [{ type: 'paragraph', content: parseInline(text) }],
180
+ });
181
+ i++;
182
+ }
183
+ if (items.length) blocks.push({ type: 'bulletList', content: items });
184
+ continue;
185
+ }
186
+
187
+ // Ordered list
188
+ if (/^\d+\.\s/.test(line.trimStart())) {
189
+ const items: TiptapNode[] = [];
190
+ while (i < lines.length && /^\d+\.\s/.test(lines[i].trimStart())) {
191
+ const text = lines[i].trimStart().replace(/^\d+\.\s+/, '');
192
+ items.push({
193
+ type: 'listItem',
194
+ content: [{ type: 'paragraph', content: parseInline(text) }],
195
+ });
196
+ i++;
197
+ }
198
+ blocks.push({ type: 'orderedList', content: items });
199
+ continue;
200
+ }
201
+
202
+ // Blockquote
203
+ if (line.trimStart().startsWith('> ')) {
204
+ const quoteLines: string[] = [];
205
+ while (i < lines.length && lines[i].trimStart().startsWith('> ')) {
206
+ quoteLines.push(lines[i].trimStart().slice(2));
207
+ i++;
208
+ }
209
+ blocks.push({
210
+ type: 'blockquote',
211
+ content: [{ type: 'paragraph', content: parseInline(quoteLines.join(' ')) }],
212
+ });
213
+ continue;
214
+ }
215
+
216
+ // Table: lines starting with |
217
+ if (line.trimStart().startsWith('|')) {
218
+ const tableLines: string[] = [];
219
+ while (i < lines.length && lines[i].trimStart().startsWith('|')) {
220
+ tableLines.push(lines[i].trim());
221
+ i++;
222
+ }
223
+ // Filter out separator rows (| --- | --- |)
224
+ const dataRows = tableLines.filter((r) => !/^\|[-\s|:]+\|$/.test(r));
225
+ if (dataRows.length) {
226
+ const tiptapRows: TiptapNode[] = dataRows.map((row, rowIdx) => {
227
+ const cells = row.replace(/^\||\|$/g, '').split('|').map((c) => c.trim());
228
+ const cellType = rowIdx === 0 ? 'tableHeader' : 'tableCell';
229
+ return {
230
+ type: 'tableRow',
231
+ content: cells.map((cellText) => ({
232
+ type: cellType,
233
+ attrs: { colspan: 1, rowspan: 1 },
234
+ content: [{ type: 'paragraph', content: parseInline(cellText) }],
235
+ })),
236
+ };
237
+ });
238
+ blocks.push({ type: 'table', content: tiptapRows });
239
+ }
240
+ continue;
241
+ }
242
+
243
+ // Paragraph (default)
244
+ const paraLines: string[] = [];
245
+ while (i < lines.length && lines[i].trim() && !lines[i].match(/^#{1,6}\s/) && !lines[i].match(/^[-*]\s/) && !lines[i].match(/^\d+\.\s/) && !lines[i].trimStart().startsWith('```') && !lines[i].trimStart().startsWith('> ') && !lines[i].trimStart().startsWith('|') && !/^---+$/.test(lines[i].trim())) {
246
+ paraLines.push(lines[i]);
247
+ i++;
248
+ }
249
+ if (paraLines.length) {
250
+ blocks.push({ type: 'paragraph', content: parseInline(paraLines.join(' ')) });
251
+ }
252
+ }
253
+
254
+ return { type: 'doc', content: blocks.length ? blocks : [{ type: 'paragraph' }] };
255
+ }
256
+
257
+ function parseInline(text: string): TiptapNode[] {
258
+ const nodes: TiptapNode[] = [];
259
+ // Simple regex-based inline parser: **bold**, *italic*, `code`
260
+ const re = /(\*\*(.+?)\*\*|\*(.+?)\*|`([^`]+?)`)/g;
261
+ let last = 0;
262
+
263
+ for (const match of text.matchAll(re)) {
264
+ if (match.index! > last) {
265
+ nodes.push({ type: 'text', text: text.slice(last, match.index!) });
266
+ }
267
+
268
+ if (match[2]) {
269
+ nodes.push({ type: 'text', text: match[2], marks: [{ type: 'bold' }] });
270
+ } else if (match[3]) {
271
+ nodes.push({ type: 'text', text: match[3], marks: [{ type: 'italic' }] });
272
+ } else if (match[4]) {
273
+ nodes.push({ type: 'text', text: match[4], marks: [{ type: 'code' }] });
274
+ }
275
+
276
+ last = match.index! + match[0].length;
277
+ }
278
+
279
+ if (last < text.length) {
280
+ nodes.push({ type: 'text', text: text.slice(last) });
281
+ }
282
+
283
+ return nodes.length ? nodes : [{ type: 'text', text: text || '' }];
284
+ }
package/doc/prefab.ts ADDED
@@ -0,0 +1,26 @@
1
+ import { type NodeData } from '@treenity/core';
2
+ import { registerPrefab } from '@treenity/core/mod';
3
+ import { mdToTiptap } from './markdown';
4
+
5
+ // FS mount-point for a docs directory
6
+ // Setup accepts { root: string } to set the filesystem path
7
+ registerPrefab('doc', 'library', [
8
+ { $path: '.', $type: 'mount-point', mount: { $type: 't.mount.fs' }, root: '' },
9
+ ] as NodeData[], (nodes, params) => {
10
+ const p = params as { root?: string } | undefined;
11
+ if (!p?.root) return nodes;
12
+ return nodes.map(n => n.$path === '.' ? { ...n, root: p.root } : n);
13
+ });
14
+
15
+ // Sample doc.page node
16
+ const welcomeMd = `A rich document with **bold**, *italic*, and \`code\`.
17
+
18
+ - Stored as Tiptap JSON
19
+ - Editable with WYSIWYG toolbar
20
+ - Supports embedded Treenity components via /slash commands`;
21
+
22
+ const welcomeContent = JSON.stringify(mdToTiptap(welcomeMd));
23
+
24
+ registerPrefab('doc', 'demo', [
25
+ { $path: 'welcome', $type: 'doc.page', title: 'Welcome to Docs', content: welcomeContent },
26
+ ] as NodeData[]);
@@ -0,0 +1,126 @@
1
+ import { CodeBlockLowlight } from '@tiptap/extension-code-block-lowlight';
2
+ import { Table } from '@tiptap/extension-table';
3
+ import TableCell from '@tiptap/extension-table-cell';
4
+ import TableHeader from '@tiptap/extension-table-header';
5
+ import TableRow from '@tiptap/extension-table-row';
6
+ import TaskItem from '@tiptap/extension-task-item';
7
+ import TaskList from '@tiptap/extension-task-list';
8
+ import { EditorContent, useEditor } from '@tiptap/react';
9
+ import StarterKit from '@tiptap/starter-kit';
10
+ import { register } from '@treenity/core';
11
+ import { common, createLowlight } from 'lowlight';
12
+ import { useCallback, useEffect, useRef } from 'react';
13
+ import { SlashCommand } from './slash-command';
14
+ import { Toolbar } from './toolbar';
15
+ import { TreenityBlock } from './treenity-block';
16
+
17
+ const lowlight = createLowlight(common);
18
+
19
+ type BlockProps = { value: any; onChange?: (data: any) => void };
20
+
21
+ function DocPageView({ value, onChange }: BlockProps) {
22
+ const suppressRef = useRef(false);
23
+ const contentRef = useRef(value.content);
24
+
25
+ const handleUpdate = useCallback(({ editor }: any) => {
26
+ if (suppressRef.current) return;
27
+ const json = JSON.stringify(editor.getJSON());
28
+ contentRef.current = json;
29
+ onChange?.({ ...value, content: json });
30
+ }, [onChange, value]);
31
+
32
+ const editor = useEditor({
33
+ extensions: [
34
+ StarterKit.configure({ codeBlock: false }),
35
+ CodeBlockLowlight.configure({ lowlight }),
36
+ TaskList,
37
+ TaskItem.configure({ nested: true }),
38
+ Table.configure({ resizable: false }),
39
+ TableRow,
40
+ TableCell,
41
+ TableHeader,
42
+ TreenityBlock,
43
+ SlashCommand,
44
+ ],
45
+ content: parseContent(value.content),
46
+ editable: !!onChange,
47
+ onUpdate: handleUpdate,
48
+ });
49
+
50
+ // Sync editor content when node changes (navigating between docs)
51
+ useEffect(() => {
52
+ if (!editor || value.content === contentRef.current) return;
53
+ contentRef.current = value.content;
54
+ suppressRef.current = true;
55
+ editor.commands.setContent(parseContent(value.content));
56
+ suppressRef.current = false;
57
+ }, [editor, value.content]);
58
+
59
+ // Sync editable state
60
+ useEffect(() => {
61
+ if (editor && editor.isEditable !== !!onChange) {
62
+ editor.setEditable(!!onChange);
63
+ }
64
+ }, [editor, onChange]);
65
+
66
+ if (!editor) return null;
67
+
68
+ return (
69
+ <div className="max-w-3xl mx-auto py-6 px-4">
70
+ {/* Title */}
71
+ <div className="mb-5">
72
+ {onChange ? (
73
+ <input
74
+ type="text"
75
+ value={value.title || ''}
76
+ onChange={(e) => onChange({ ...value, title: e.target.value })}
77
+ placeholder="Untitled"
78
+ className="w-full text-2xl font-semibold tracking-tight bg-transparent border-none outline-none text-[var(--text)] placeholder:text-[var(--text-3)]"
79
+ />
80
+ ) : (
81
+ value.title && <h1 className="text-2xl font-semibold tracking-tight text-[var(--text)]">{value.title}</h1>
82
+ )}
83
+ </div>
84
+
85
+ {/* Toolbar — only in edit mode */}
86
+ {onChange && <Toolbar editor={editor} />}
87
+
88
+ {/* Editor content */}
89
+ <div
90
+ className={`min-h-[300px] ${onChange ? 'pt-4' : ''}`}
91
+ onDrop={(e) => {
92
+ if (!editor || !onChange) return;
93
+ const path = e.dataTransfer.getData('application/treenity-path');
94
+ if (!path) return;
95
+ e.preventDefault();
96
+ e.stopPropagation();
97
+ editor.chain().focus().insertContent({
98
+ type: 'treenityBlock',
99
+ attrs: { ref: path, type: null, props: {} },
100
+ }).run();
101
+ }}
102
+ onDragOver={(e) => {
103
+ if (e.dataTransfer.types.includes('application/treenity-path')) {
104
+ e.preventDefault();
105
+ e.dataTransfer.dropEffect = 'copy';
106
+ }
107
+ }}
108
+ >
109
+ <EditorContent editor={editor} />
110
+ </div>
111
+ </div>
112
+ );
113
+ }
114
+
115
+ function parseContent(content: string | undefined): any {
116
+ if (!content) return undefined;
117
+ try {
118
+ return JSON.parse(content);
119
+ } catch {
120
+ return { type: 'doc', content: [{ type: 'paragraph', content: [{ type: 'text', text: content }] }] };
121
+ }
122
+ }
123
+
124
+ export function registerDocViews() {
125
+ register('doc.page', 'react', DocPageView);
126
+ }
package/doc/seed.ts ADDED
@@ -0,0 +1,10 @@
1
+ import { type NodeData } from '@treenity/core';
2
+ import { registerPrefab } from '@treenity/core/mod';
3
+ import { join } from 'node:path';
4
+
5
+ registerPrefab('doc', 'seed', [
6
+ { $path: 'docs', $type: 'mount-point', mount: { $type: 't.mount.fs' }, root: '' },
7
+ ] as NodeData[], (nodes) => {
8
+ const root = process.env.DOCS_ROOT || join(process.cwd(), 'docs');
9
+ return nodes.map(n => n.$path === 'docs' ? { ...n, root } : n);
10
+ }, { tier: 'core' });
package/doc/server.ts ADDED
@@ -0,0 +1,5 @@
1
+ import './types';
2
+ import './text';
3
+ import './fs-codec';
4
+ import './prefab';
5
+ import './seed';
@@ -0,0 +1,136 @@
1
+ import { Extension } from '@tiptap/core';
2
+ import { ReactRenderer } from '@tiptap/react';
3
+ import Suggestion, { type SuggestionOptions } from '@tiptap/suggestion';
4
+ import { SlashMenu, type SlashMenuItem } from './slash-menu';
5
+
6
+ const defaultItems: SlashMenuItem[] = [
7
+ {
8
+ title: 'Heading 1',
9
+ group: 'Text',
10
+ command: ({ editor, range }) => editor.chain().focus().deleteRange(range).toggleHeading({ level: 1 }).run(),
11
+ },
12
+ {
13
+ title: 'Heading 2',
14
+ group: 'Text',
15
+ command: ({ editor, range }) => editor.chain().focus().deleteRange(range).toggleHeading({ level: 2 }).run(),
16
+ },
17
+ {
18
+ title: 'Heading 3',
19
+ group: 'Text',
20
+ command: ({ editor, range }) => editor.chain().focus().deleteRange(range).toggleHeading({ level: 3 }).run(),
21
+ },
22
+ {
23
+ title: 'Bullet List',
24
+ group: 'Text',
25
+ command: ({ editor, range }) => editor.chain().focus().deleteRange(range).toggleBulletList().run(),
26
+ },
27
+ {
28
+ title: 'Ordered List',
29
+ group: 'Text',
30
+ command: ({ editor, range }) => editor.chain().focus().deleteRange(range).toggleOrderedList().run(),
31
+ },
32
+ {
33
+ title: 'Code Block',
34
+ group: 'Text',
35
+ command: ({ editor, range }) => editor.chain().focus().deleteRange(range).toggleCodeBlock().run(),
36
+ },
37
+ {
38
+ title: 'Quote',
39
+ group: 'Text',
40
+ command: ({ editor, range }) => editor.chain().focus().deleteRange(range).toggleBlockquote().run(),
41
+ },
42
+ {
43
+ title: 'Divider',
44
+ group: 'Text',
45
+ command: ({ editor, range }) => editor.chain().focus().deleteRange(range).setHorizontalRule().run(),
46
+ },
47
+ {
48
+ title: 'Component',
49
+ group: 'Treenity',
50
+ command: () => {},
51
+ pickComponent: true,
52
+ },
53
+ ];
54
+
55
+ export const SlashCommand = Extension.create({
56
+ name: 'slashCommand',
57
+
58
+ addOptions() {
59
+ return {
60
+ suggestion: {
61
+ char: '/',
62
+ command: ({ editor, range, props }: any) => {
63
+ props.command({ editor, range });
64
+ },
65
+ } as Partial<SuggestionOptions>,
66
+ };
67
+ },
68
+
69
+ addProseMirrorPlugins() {
70
+ return [
71
+ Suggestion({
72
+ editor: this.editor,
73
+ ...this.options.suggestion,
74
+ items: ({ query }: { query: string }) =>
75
+ defaultItems.filter((item) =>
76
+ item.title.toLowerCase().includes(query.toLowerCase()),
77
+ ),
78
+ render: () => {
79
+ let component: ReactRenderer;
80
+
81
+ return {
82
+ onStart: (props: any) => {
83
+ component = new ReactRenderer(SlashMenu, {
84
+ props,
85
+ editor: props.editor,
86
+ });
87
+ // ReactRenderer.element is a detached div — must add to DOM
88
+ document.body.appendChild(component.element);
89
+ positionElement(component.element, props.clientRect);
90
+ },
91
+
92
+ onUpdate: (props: any) => {
93
+ component.updateProps(props);
94
+ positionElement(component.element, props.clientRect);
95
+ },
96
+
97
+ onKeyDown: (props: any) => {
98
+ if (props.event.key === 'Escape') {
99
+ component?.element?.remove();
100
+ component?.destroy();
101
+ return true;
102
+ }
103
+ return (component?.ref as any)?.onKeyDown?.(props) ?? false;
104
+ },
105
+
106
+ onExit: () => {
107
+ component?.element?.remove();
108
+ component?.destroy();
109
+ },
110
+ };
111
+ },
112
+ }),
113
+ ];
114
+ },
115
+ });
116
+
117
+ function positionElement(el: HTMLElement, clientRect: (() => DOMRect | null) | null) {
118
+ const rect = clientRect?.();
119
+ if (!rect) return;
120
+
121
+ const menuH = 300; // approximate max height
122
+ const menuW = 210;
123
+ const vw = window.innerWidth;
124
+ const vh = window.innerHeight;
125
+
126
+ const left = Math.min(rect.left, vw - menuW - 8);
127
+ const fitsBelow = rect.bottom + 4 + menuH < vh;
128
+ const top = fitsBelow ? rect.bottom + 4 : rect.top - menuH - 4;
129
+
130
+ Object.assign(el.style, {
131
+ position: 'fixed',
132
+ left: `${Math.max(4, left)}px`,
133
+ top: `${Math.max(4, top)}px`,
134
+ zIndex: '9999',
135
+ });
136
+ }
@@ -0,0 +1,91 @@
1
+ import { MiniTree } from '@treenity/react/mods/editor-ui/form-fields';
2
+ import { forwardRef, useEffect, useImperativeHandle, useState } from 'react';
3
+
4
+ export type SlashMenuItem = {
5
+ title: string;
6
+ group: string;
7
+ command: (props: any) => void;
8
+ pickComponent?: boolean;
9
+ };
10
+
11
+ type SlashMenuProps = {
12
+ items: SlashMenuItem[];
13
+ command: (item: any) => void;
14
+ editor: any;
15
+ range: any;
16
+ };
17
+
18
+ export const SlashMenu = forwardRef<any, SlashMenuProps>(({ items, command, editor, range }, ref) => {
19
+ const [selectedIndex, setSelectedIndex] = useState(0);
20
+ const [showPicker, setShowPicker] = useState(false);
21
+
22
+ useEffect(() => setSelectedIndex(0), [items]);
23
+
24
+ useImperativeHandle(ref, () => ({
25
+ onKeyDown: ({ event }: { event: KeyboardEvent }) => {
26
+ if (event.key === 'ArrowDown') {
27
+ event.preventDefault();
28
+ setSelectedIndex((i) => (i + 1) % items.length);
29
+ return true;
30
+ }
31
+ if (event.key === 'ArrowUp') {
32
+ event.preventDefault();
33
+ setSelectedIndex((i) => (i - 1 + items.length) % items.length);
34
+ return true;
35
+ }
36
+ if (event.key === 'Enter') {
37
+ event.preventDefault();
38
+ selectItem(selectedIndex);
39
+ return true;
40
+ }
41
+ return false;
42
+ },
43
+ }));
44
+
45
+ const selectItem = (index: number) => {
46
+ const item = items[index];
47
+ if (!item) return;
48
+ if (item.pickComponent) {
49
+ setShowPicker(true);
50
+ return;
51
+ }
52
+ command(item);
53
+ };
54
+
55
+ const handlePickNode = (path: string) => {
56
+ setShowPicker(false);
57
+ editor.chain().focus().deleteRange(range).insertContent({
58
+ type: 'treenityBlock',
59
+ attrs: { ref: path, type: null, props: {} },
60
+ }).run();
61
+ };
62
+
63
+ if (showPicker) {
64
+ return <MiniTree onSelect={(path) => { handlePickNode(path); setShowPicker(false); }} />;
65
+ }
66
+
67
+ if (!items.length) {
68
+ return (
69
+ <div className="slash-menu">
70
+ <div className="slash-menu-empty">No commands</div>
71
+ </div>
72
+ );
73
+ }
74
+
75
+ return (
76
+ <div className="slash-menu">
77
+ {items.map((item, i) => (
78
+ <button
79
+ key={item.title}
80
+ onMouseDown={(e) => { e.preventDefault(); selectItem(i); }}
81
+ className={`slash-menu-item${i === selectedIndex ? ' selected' : ''}`}
82
+ >
83
+ <span className="slash-menu-title">{item.title}</span>
84
+ <span className="slash-menu-group">{item.group}</span>
85
+ </button>
86
+ ))}
87
+ </div>
88
+ );
89
+ });
90
+
91
+ SlashMenu.displayName = 'SlashMenu';