@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
package/sim/view.tsx ADDED
@@ -0,0 +1,446 @@
1
+ // AgentSim views — World map + Agent card + Item card
2
+ // WorldView: controls, spatial map with entities + proximity links, event log
3
+ // AgentView: profile, editable prompt, memory, event inbox
4
+ // ItemView: description card
5
+
6
+ import { getComponent, type NodeData, register } from '@treenity/core';
7
+ import { useCurrentNode } from '@treenity/react/context';
8
+ import { useChildren } from '@treenity/react/hooks';
9
+ import { trpc } from '@treenity/react/trpc';
10
+ import { useCallback, useMemo, useRef, useState } from 'react';
11
+ import {
12
+ type EventEntry,
13
+ SimAi,
14
+ SimConfig,
15
+ SimDescriptive,
16
+ SimEvents,
17
+ SimMemory,
18
+ SimNearby,
19
+ SimPosition,
20
+ SimRound,
21
+ } from './types';
22
+
23
+ type Link = { x1: number; y1: number; x2: number; y2: number; opacity: number };
24
+
25
+ function buildLinks(entities: NodeData[]): Link[] {
26
+ const posByName = new Map<string, SimPosition & { $type: string }>();
27
+ for (const a of entities) {
28
+ const d = getComponent(a, SimDescriptive);
29
+ const p = getComponent(a, SimPosition);
30
+ if (d && p) posByName.set(d.name, p);
31
+ }
32
+ const links: Link[] = [];
33
+ const seen = new Set<string>();
34
+ for (const a of entities) {
35
+ const desc = getComponent(a, SimDescriptive);
36
+ const pos = getComponent(a, SimPosition);
37
+ const nearby = getComponent(a, SimNearby);
38
+ if (!desc || !pos || !nearby?.agents) continue;
39
+ for (const other of nearby.agents) {
40
+ const key = [desc.name, other].sort().join('|');
41
+ if (seen.has(key)) continue;
42
+ seen.add(key);
43
+ const op = posByName.get(other);
44
+ if (!op) continue;
45
+ const d = Math.sqrt((pos.x - op.x) ** 2 + (pos.y - op.y) ** 2);
46
+ const maxR = Math.max(pos.radius, op.radius);
47
+ links.push({
48
+ x1: pos.x, y1: pos.y, x2: op.x, y2: op.y,
49
+ opacity: maxR > 0 ? 0.15 + 0.35 * (1 - d / maxR) : 0.3,
50
+ });
51
+ }
52
+ }
53
+ return links;
54
+ }
55
+
56
+ const btn = {
57
+ padding: '4px 12px',
58
+ borderRadius: 'var(--radius)',
59
+ border: '1px solid var(--border)',
60
+ background: 'var(--surface)',
61
+ color: 'var(--text)',
62
+ cursor: 'pointer',
63
+ fontSize: 11,
64
+ } as const;
65
+
66
+ function formatAction(e: EventEntry): string {
67
+ const msg = (e.data as any)?.message;
68
+ if (e.action.startsWith('speak') && msg) return `"${msg}"`;
69
+ if (e.action === 'move') return `→ (${(e.data as any).x}, ${(e.data as any).y})`;
70
+ if (e.action.includes('→')) return e.action;
71
+ return e.action;
72
+ }
73
+
74
+ // ── WorldView ──
75
+
76
+ function errMsg(e: unknown): string {
77
+ if (e instanceof Error) return e.message;
78
+ return String(e);
79
+ }
80
+
81
+ function WorldView() {
82
+ const node = useCurrentNode();
83
+ const children = useChildren(node.$path, { watchNew: true });
84
+ const [selected, setSelected] = useState<string | null>(null);
85
+ const [error, setError] = useState<string | null>(null);
86
+ const mapRef = useRef<HTMLDivElement>(null);
87
+
88
+ const call = useCallback((p: Parameters<typeof trpc.execute.mutate>[0]) => {
89
+ setError(null);
90
+ trpc.execute.mutate(p).catch((e) => setError(errMsg(e)));
91
+ }, []);
92
+
93
+ const cfg = getComponent(node, SimConfig);
94
+ const round = getComponent(node, SimRound);
95
+ const entities = children.filter((n: NodeData) => n.$type === 'sim.agent' || n.$type === 'sim.item');
96
+ const W = cfg?.width ?? 600;
97
+ const H = cfg?.height ?? 400;
98
+ const log = round?.log ?? [];
99
+ const lastRound = (round?.current ?? 1) - 1;
100
+ const links = useMemo(() => buildLinks(entities), [entities]);
101
+
102
+ const toggle = useCallback(() => {
103
+ call({ path: node.$path, action: cfg?.running ? 'stop' : 'start' });
104
+ }, [node.$path, cfg?.running, call]);
105
+
106
+ const onMapClick = useCallback(
107
+ (e: React.MouseEvent<HTMLDivElement>) => {
108
+ if (!selected || !mapRef.current) return;
109
+ const rect = mapRef.current.getBoundingClientRect();
110
+ const x = Math.round(e.clientX - rect.left);
111
+ const y = Math.round(e.clientY - rect.top);
112
+ call({ path: selected, action: 'move', data: { x, y } });
113
+ },
114
+ [selected, call],
115
+ );
116
+
117
+ const setSpeed = useCallback(
118
+ (delay: number) => {
119
+ call({ path: node.$path, action: 'set-config', data: { roundDelay: delay } });
120
+ },
121
+ [node.$path, call],
122
+ );
123
+
124
+ return (
125
+ <div style={{ fontFamily: 'var(--mono)', fontSize: 13 }}>
126
+ {/* Controls */}
127
+ <div style={{ display: 'flex', alignItems: 'center', gap: 8, marginBottom: 12, flexWrap: 'wrap' }}>
128
+ <button
129
+ onClick={toggle}
130
+ style={{
131
+ ...btn,
132
+ background: cfg?.running ? 'var(--danger, #c44)' : 'var(--accent)',
133
+ color: '#fff',
134
+ fontWeight: 600,
135
+ }}
136
+ >
137
+ {cfg?.running ? 'Stop' : 'Start'}
138
+ </button>
139
+ <span style={{ color: 'var(--text-2)' }}>R{round?.current ?? 0}</span>
140
+ {round?.phase === 'thinking' && (
141
+ <span style={{ color: 'var(--accent)' }}>thinking...</span>
142
+ )}
143
+ <span style={{ color: 'var(--text-3)', fontSize: 11 }}>|</span>
144
+ {[1000, 3000, 5000, 10000].map((d) => (
145
+ <button
146
+ key={d}
147
+ onClick={() => setSpeed(d)}
148
+ style={{
149
+ ...btn,
150
+ background: cfg?.roundDelay === d ? 'var(--accent)' : 'var(--surface)',
151
+ color: cfg?.roundDelay === d ? '#fff' : 'var(--text-2)',
152
+ }}
153
+ >
154
+ {d / 1000}s
155
+ </button>
156
+ ))}
157
+ <span style={{ color: 'var(--text-3)', marginLeft: 'auto', fontSize: 11 }}>
158
+ {entities.length} entities
159
+ {selected && ` · click map to move`}
160
+ </span>
161
+ </div>
162
+
163
+ {error && (
164
+ <div onClick={() => setError(null)} style={{ padding: '6px 12px', marginBottom: 8, background: 'var(--danger, #c44)', color: '#fff', borderRadius: 'var(--radius)', fontSize: 12, cursor: 'pointer' }}>
165
+ {error}
166
+ </div>
167
+ )}
168
+
169
+ {/* Map */}
170
+ <div
171
+ ref={mapRef}
172
+ onClick={onMapClick}
173
+ style={{
174
+ position: 'relative',
175
+ width: W,
176
+ height: H,
177
+ background: 'var(--surface)',
178
+ border: '1px solid var(--border)',
179
+ borderRadius: 'var(--radius)',
180
+ overflow: 'hidden',
181
+ cursor: selected ? 'crosshair' : 'default',
182
+ }}
183
+ >
184
+ <svg
185
+ style={{ position: 'absolute', inset: 0, pointerEvents: 'none', zIndex: 1 }}
186
+ width={W}
187
+ height={H}
188
+ >
189
+ {links.map((l, i) => (
190
+ <line
191
+ key={i}
192
+ x1={l.x1} y1={l.y1} x2={l.x2} y2={l.y2}
193
+ stroke="var(--accent, #6366f1)"
194
+ strokeWidth={1.5}
195
+ opacity={l.opacity}
196
+ />
197
+ ))}
198
+ </svg>
199
+
200
+ {entities.map((a: NodeData) => {
201
+ const pos = getComponent(a, SimPosition);
202
+ const desc = getComponent(a, SimDescriptive);
203
+ if (!pos) return null;
204
+ const isSel = selected === a.$path;
205
+ const isItem = a.$type === 'sim.item';
206
+ return (
207
+ <div key={a.$path}>
208
+ <div
209
+ style={{
210
+ position: 'absolute',
211
+ left: pos.x - pos.radius,
212
+ top: pos.y - pos.radius,
213
+ width: pos.radius * 2,
214
+ height: pos.radius * 2,
215
+ borderRadius: '50%',
216
+ border: `1px dashed ${isSel ? 'var(--accent)' : 'var(--border)'}`,
217
+ opacity: isSel ? 0.5 : 0.2,
218
+ pointerEvents: 'none',
219
+ }}
220
+ />
221
+ <div
222
+ onClick={(e) => { e.stopPropagation(); setSelected(isSel ? null : a.$path); }}
223
+ style={{
224
+ position: 'absolute',
225
+ left: pos.x - (isItem ? 12 : 16),
226
+ top: pos.y - (isItem ? 12 : 16),
227
+ width: isItem ? 24 : 32,
228
+ height: isItem ? 24 : 32,
229
+ borderRadius: isItem ? 'var(--radius)' : '50%',
230
+ background: isItem ? 'var(--surface)' : 'var(--accent)',
231
+ border: isSel ? '2px solid #fff' : `1px solid ${isItem ? 'var(--border)' : 'transparent'}`,
232
+ boxShadow: isSel ? '0 0 0 2px var(--accent)' : 'none',
233
+ display: 'flex',
234
+ alignItems: 'center',
235
+ justifyContent: 'center',
236
+ fontSize: isItem ? 14 : 16,
237
+ zIndex: 2,
238
+ cursor: 'pointer',
239
+ }}
240
+ title={`${desc?.name} (${pos.x}, ${pos.y}) r=${pos.radius} [${a.$type}]`}
241
+ >
242
+ {desc?.icon ?? '?'}
243
+ </div>
244
+ <div
245
+ style={{
246
+ position: 'absolute',
247
+ left: pos.x,
248
+ top: pos.y + (isItem ? 14 : 18),
249
+ transform: 'translateX(-50%)',
250
+ fontSize: 10,
251
+ color: isSel ? 'var(--accent)' : 'var(--text-2)',
252
+ fontWeight: isSel ? 600 : 400,
253
+ whiteSpace: 'nowrap',
254
+ zIndex: 2,
255
+ pointerEvents: 'none',
256
+ }}
257
+ >
258
+ {desc?.name}
259
+ </div>
260
+ </div>
261
+ );
262
+ })}
263
+
264
+ {entities.length === 0 && (
265
+ <div style={{ position: 'absolute', inset: 0, display: 'flex', alignItems: 'center', justifyContent: 'center', color: 'var(--text-3)' }}>
266
+ No entities — add sim.agent or sim.item children
267
+ </div>
268
+ )}
269
+ </div>
270
+
271
+ {/* Event Log */}
272
+ <div style={{ marginTop: 12, maxHeight: 240, overflowY: 'auto' }}>
273
+ <div style={{ fontSize: 11, color: 'var(--text-3)', textTransform: 'uppercase', letterSpacing: 0.5, marginBottom: 6 }}>
274
+ Event Log ({log.length})
275
+ </div>
276
+ {log.length === 0 && (
277
+ <div style={{ color: 'var(--text-3)', padding: 8 }}>No events yet — press Start</div>
278
+ )}
279
+ {[...log].reverse().map((e, i) => (
280
+ <div
281
+ key={`${e.ts}-${i}`}
282
+ style={{
283
+ display: 'flex',
284
+ gap: 8,
285
+ padding: '4px 8px',
286
+ background: e.round === lastRound ? 'var(--accent-bg, rgba(99,102,241,0.08))' : 'transparent',
287
+ borderRadius: 'var(--radius)',
288
+ fontSize: 12,
289
+ }}
290
+ >
291
+ <span style={{ minWidth: 28, color: 'var(--text-3)' }}>R{e.round}</span>
292
+ <span>{e.icon}</span>
293
+ <span style={{ color: 'var(--accent)', minWidth: 48 }}>{e.agent}</span>
294
+ <span style={{ color: 'var(--text-2)', flex: 1 }}>{formatAction(e)}</span>
295
+ {e.heardBy?.length ? (
296
+ <span style={{ color: 'var(--text-3)', fontSize: 10 }}>
297
+ heard: {e.heardBy.join(', ')}
298
+ </span>
299
+ ) : null}
300
+ </div>
301
+ ))}
302
+ </div>
303
+ </div>
304
+ );
305
+ }
306
+
307
+ // ── AgentView ──
308
+
309
+ function AgentView() {
310
+ const node = useCurrentNode();
311
+ const desc = getComponent(node, SimDescriptive);
312
+ const pos = getComponent(node, SimPosition);
313
+ const mem = getComponent(node, SimMemory);
314
+ const ai = getComponent(node, SimAi);
315
+ const events = getComponent(node, SimEvents)?.entries ?? [];
316
+
317
+ const [editPrompt, setEditPrompt] = useState<string | null>(null);
318
+ const [error, setError] = useState<string | null>(null);
319
+
320
+ const savePrompt = useCallback(() => {
321
+ if (editPrompt === null) return;
322
+ setError(null);
323
+ trpc.execute.mutate({ path: node.$path, action: 'update', data: { systemPrompt: editPrompt } })
324
+ .catch((e) => setError(errMsg(e)));
325
+ setEditPrompt(null);
326
+ }, [node.$path, editPrompt]);
327
+
328
+ return (
329
+ <div style={{ fontFamily: 'var(--mono)', fontSize: 13, maxWidth: 500 }}>
330
+ {/* Header */}
331
+ <div style={{ display: 'flex', alignItems: 'center', gap: 12, marginBottom: 16 }}>
332
+ <div style={{ width: 48, height: 48, borderRadius: '50%', background: 'var(--accent)', display: 'flex', alignItems: 'center', justifyContent: 'center', fontSize: 24 }}>
333
+ {desc?.icon ?? '?'}
334
+ </div>
335
+ <div>
336
+ <div style={{ fontSize: 16, fontWeight: 600 }}>{desc?.name ?? 'Agent'}</div>
337
+ <div style={{ color: 'var(--text-2)', fontSize: 12 }}>{desc?.description}</div>
338
+ </div>
339
+ </div>
340
+
341
+ {error && (
342
+ <div onClick={() => setError(null)} style={{ padding: '6px 12px', marginBottom: 8, background: 'var(--danger, #c44)', color: '#fff', borderRadius: 'var(--radius)', fontSize: 12, cursor: 'pointer' }}>
343
+ {error}
344
+ </div>
345
+ )}
346
+
347
+ {pos && (
348
+ <div style={{ marginBottom: 12, padding: '8px 12px', background: 'var(--surface)', borderRadius: 'var(--radius)' }}>
349
+ Position: ({pos.x}, {pos.y}) &middot; Radius: {pos.radius}
350
+ </div>
351
+ )}
352
+
353
+ {/* System Prompt */}
354
+ <div style={{ marginBottom: 12 }}>
355
+ <div style={{ fontSize: 11, color: 'var(--text-3)', textTransform: 'uppercase', letterSpacing: 0.5, marginBottom: 6, display: 'flex', alignItems: 'center', gap: 8 }}>
356
+ System Prompt
357
+ {editPrompt === null && (
358
+ <button onClick={() => setEditPrompt(ai?.systemPrompt ?? '')} style={{ ...btn, fontSize: 10, padding: '2px 8px' }}>Edit</button>
359
+ )}
360
+ </div>
361
+ {editPrompt !== null ? (
362
+ <div>
363
+ <textarea
364
+ value={editPrompt}
365
+ onChange={(e) => setEditPrompt(e.target.value)}
366
+ style={{ width: '100%', minHeight: 80, padding: 8, borderRadius: 'var(--radius)', border: '1px solid var(--border)', background: 'var(--surface)', color: 'var(--text)', fontFamily: 'var(--mono)', fontSize: 12, resize: 'vertical' }}
367
+ autoFocus
368
+ />
369
+ <div style={{ display: 'flex', gap: 6, marginTop: 6 }}>
370
+ <button onClick={savePrompt} style={{ ...btn, background: 'var(--accent)', color: '#fff' }}>Save</button>
371
+ <button onClick={() => setEditPrompt(null)} style={btn}>Cancel</button>
372
+ </div>
373
+ </div>
374
+ ) : (
375
+ <div style={{ padding: '4px 8px', fontSize: 12, color: 'var(--text-2)', whiteSpace: 'pre-wrap' }}>
376
+ {ai?.systemPrompt ?? '(none)'}
377
+ </div>
378
+ )}
379
+ </div>
380
+
381
+ {/* Event Inbox */}
382
+ <div style={{ marginBottom: 12 }}>
383
+ <div style={{ fontSize: 11, color: 'var(--text-3)', textTransform: 'uppercase', letterSpacing: 0.5, marginBottom: 6 }}>
384
+ Event Inbox ({events.length})
385
+ </div>
386
+ {events.length === 0 && <div style={{ color: 'var(--text-3)', padding: 8 }}>No events yet</div>}
387
+ {[...events].reverse().slice(0, 15).map((e, i) => (
388
+ <div key={i} style={{ padding: '3px 8px', fontSize: 11, color: 'var(--text-2)', display: 'flex', gap: 6 }}>
389
+ <span style={{ color: 'var(--text-3)', minWidth: 24 }}>R{e.round}</span>
390
+ <span style={{ color: 'var(--accent)' }}>{e.from}</span>
391
+ <span>{e.type}{e.to ? `→${e.to}` : ''}</span>
392
+ <span style={{ flex: 1, color: 'var(--text-3)' }}>
393
+ {(e.data as any)?.message ? `"${(e.data as any).message}"` : JSON.stringify(e.data).slice(0, 60)}
394
+ </span>
395
+ </div>
396
+ ))}
397
+ </div>
398
+
399
+ {/* Memory */}
400
+ <div>
401
+ <div style={{ fontSize: 11, color: 'var(--text-3)', textTransform: 'uppercase', letterSpacing: 0.5, marginBottom: 6 }}>
402
+ Memory ({mem?.entries?.length ?? 0})
403
+ </div>
404
+ {(!mem?.entries || mem.entries.length === 0) && (
405
+ <div style={{ color: 'var(--text-3)', padding: 8 }}>Empty memory</div>
406
+ )}
407
+ {mem?.entries?.map((e, i) => (
408
+ <div key={i} style={{ padding: '4px 8px', fontSize: 12, color: 'var(--text-2)', background: i === (mem.entries.length ?? 0) - 1 ? 'var(--accent-bg, rgba(99,102,241,0.08))' : 'transparent', borderRadius: 'var(--radius)' }}>
409
+ {e}
410
+ </div>
411
+ ))}
412
+ </div>
413
+ </div>
414
+ );
415
+ }
416
+
417
+ // ── ItemView ──
418
+
419
+ function ItemView() {
420
+ const node = useCurrentNode();
421
+ const desc = getComponent(node, SimDescriptive);
422
+ const pos = getComponent(node, SimPosition);
423
+
424
+ return (
425
+ <div style={{ fontFamily: 'var(--mono)', fontSize: 13, maxWidth: 400 }}>
426
+ <div style={{ display: 'flex', alignItems: 'center', gap: 12, marginBottom: 12 }}>
427
+ <div style={{ width: 40, height: 40, borderRadius: 'var(--radius)', background: 'var(--surface)', border: '1px solid var(--border)', display: 'flex', alignItems: 'center', justifyContent: 'center', fontSize: 20 }}>
428
+ {desc?.icon ?? '?'}
429
+ </div>
430
+ <div>
431
+ <div style={{ fontSize: 14, fontWeight: 600 }}>{desc?.name ?? 'Item'}</div>
432
+ <div style={{ color: 'var(--text-2)', fontSize: 12 }}>{desc?.description}</div>
433
+ </div>
434
+ </div>
435
+ {pos && (
436
+ <div style={{ padding: '8px 12px', background: 'var(--surface)', borderRadius: 'var(--radius)', fontSize: 12, color: 'var(--text-2)' }}>
437
+ Position: ({pos.x}, {pos.y}) &middot; Radius: {pos.radius}
438
+ </div>
439
+ )}
440
+ </div>
441
+ );
442
+ }
443
+
444
+ register('sim.world', 'react', WorldView as any);
445
+ register('sim.agent', 'react', AgentView as any);
446
+ register('sim.item', 'react', ItemView as any);
@@ -0,0 +1,3 @@
1
+ import './types';
2
+ import './view';
3
+ import './edit';
package/table/edit.tsx ADDED
@@ -0,0 +1,241 @@
1
+ import { type ComponentData, isComponent, type NodeData, register } from '@treenity/core';
2
+ import type { TypeSchema } from '@treenity/core/schema/types';
3
+ import { useCurrentNode } from '@treenity/react/context';
4
+ import { useChildren } from '@treenity/react/hooks';
5
+ import { useSchema } from '@treenity/react/schema-loader';
6
+ import { Button } from '@treenity/react/ui/button';
7
+ import { Checkbox } from '@treenity/react/ui/checkbox';
8
+ import { Input } from '@treenity/react/ui/input';
9
+ import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@treenity/react/ui/select';
10
+ import { useMemo, useState } from 'react';
11
+ import type { UITable } from './types';
12
+
13
+ // ── Field tree types ──
14
+
15
+ type FieldNode = {
16
+ key: string;
17
+ label: string;
18
+ type?: string;
19
+ children?: FieldNode[];
20
+ };
21
+
22
+ // ── Build field tree from schema + data sample ──
23
+
24
+ function buildFieldTree(schema: TypeSchema | null, sample: Record<string, unknown>[]): FieldNode[] {
25
+ const nodes: FieldNode[] = [];
26
+ const seen = new Set<string>();
27
+
28
+ if (schema?.properties) {
29
+ for (const [key, prop] of Object.entries(schema.properties)) {
30
+ if (key.startsWith('$')) continue;
31
+ seen.add(key);
32
+ nodes.push({ key, label: (prop as any).title || key });
33
+ }
34
+ }
35
+
36
+ for (const row of sample.slice(0, 10)) {
37
+ for (const [key, val] of Object.entries(row)) {
38
+ if (key.startsWith('$') || seen.has(key)) continue;
39
+ seen.add(key);
40
+
41
+ if (isComponent(val)) {
42
+ const children = Object.keys(val)
43
+ .filter(k => !k.startsWith('$'))
44
+ .map(k => ({ key: `${key}.${k}`, label: k }));
45
+ nodes.push({ key, label: key, type: val.$type, children });
46
+ } else {
47
+ nodes.push({ key, label: key });
48
+ }
49
+ }
50
+ }
51
+
52
+ return nodes;
53
+ }
54
+
55
+ // ── Helpers ──
56
+
57
+ function resolveDisplayType(child: NodeData, field: string): string {
58
+ if (!field) return child.$type;
59
+ const val = (child as any)[field];
60
+ return val?.$type ?? '';
61
+ }
62
+
63
+ // ── Edit View ──
64
+
65
+ function TableEditView({ value, onChange }: { value: ComponentData; onChange?: (next: ComponentData) => void }) {
66
+ const node = useCurrentNode();
67
+ const children = useChildren(node.$path, { watch: true, limit: 1000 });
68
+
69
+ const state = value as unknown as UITable;
70
+ const emit = (patch: Partial<UITable>) => {
71
+ if (!onChange) return;
72
+ onChange({ ...value, ...patch } as ComponentData);
73
+ };
74
+
75
+ // Detect types
76
+ const { types, typeMap } = useMemo(() => {
77
+ const map = new Map<string, NodeData[]>();
78
+ for (const child of children) {
79
+ const dt = resolveDisplayType(child, state.field ?? '');
80
+ if (!dt) continue;
81
+ const arr = map.get(dt) ?? [];
82
+ arr.push(child);
83
+ map.set(dt, arr);
84
+ }
85
+ return { typeMap: map, types: [...map.keys()] };
86
+ }, [children, state.field]);
87
+
88
+ const activeType = state.displayType && types.includes(state.displayType)
89
+ ? state.displayType
90
+ : types[0] ?? '';
91
+
92
+ const sample = useMemo(() => {
93
+ const matched = typeMap.get(activeType) ?? [];
94
+ return matched.slice(0, 10) as Record<string, unknown>[];
95
+ }, [typeMap, activeType]);
96
+
97
+ const schema = useSchema(activeType);
98
+ const fieldTree = useMemo(() => buildFieldTree(schema ?? null, sample), [schema, sample]);
99
+
100
+ const savedColumns = state.columns?.[activeType] ?? [];
101
+ const hasCustomColumns = savedColumns.length > 0;
102
+
103
+ const toggleField = (key: string, label: string) => {
104
+ let cols = savedColumns.length > 0 ? [...savedColumns] : fieldTree.map(f => ({ field: f.key, label: f.label, visible: true }));
105
+ const idx = cols.findIndex(c => c.field === key);
106
+ if (idx >= 0) {
107
+ cols[idx] = { ...cols[idx], visible: !cols[idx].visible };
108
+ } else {
109
+ cols.push({ field: key, label, visible: true });
110
+ }
111
+ emit({ columns: { ...state.columns, [activeType]: cols } });
112
+ };
113
+
114
+ const resetColumns = () => {
115
+ const next = { ...state.columns };
116
+ delete next[activeType];
117
+ emit({ columns: next });
118
+ };
119
+
120
+ const isFieldVisible = (key: string) => {
121
+ if (!hasCustomColumns) return true;
122
+ const col = savedColumns.find(c => c.field === key);
123
+ return col?.visible !== false;
124
+ };
125
+
126
+ const [expanded, setExpanded] = useState<Set<string>>(new Set());
127
+ const toggleExpand = (key: string) => {
128
+ setExpanded(prev => {
129
+ const next = new Set(prev);
130
+ next.has(key) ? next.delete(key) : next.add(key);
131
+ return next;
132
+ });
133
+ };
134
+
135
+ return (
136
+ <div className="flex flex-col gap-3 text-xs">
137
+ <div className="flex flex-col gap-2">
138
+ <label className="flex items-center gap-2">
139
+ <span className="w-20 text-muted-foreground">Page size</span>
140
+ <Input
141
+ type="number"
142
+ min={1}
143
+ max={1000}
144
+ className="h-7 w-20 text-xs"
145
+ value={state.pageSize ?? 25}
146
+ onChange={e => emit({ pageSize: Number(e.target.value) || 25 })}
147
+ />
148
+ </label>
149
+
150
+ <label className="flex items-center gap-2">
151
+ <span className="w-20 text-muted-foreground">Field</span>
152
+ <Input
153
+ type="text"
154
+ placeholder="(node itself)"
155
+ className="h-7 flex-1 text-xs"
156
+ value={state.field ?? ''}
157
+ onChange={e => emit({ field: e.target.value })}
158
+ />
159
+ </label>
160
+
161
+ {types.length > 1 && (
162
+ <div className="flex items-center gap-2">
163
+ <span className="w-20 text-muted-foreground">Type</span>
164
+ <Select value={activeType} onValueChange={v => emit({ displayType: v })}>
165
+ <SelectTrigger className="h-7 flex-1 text-xs">
166
+ <SelectValue />
167
+ </SelectTrigger>
168
+ <SelectContent>
169
+ {types.map(t => (
170
+ <SelectItem key={t} value={t}>{t} ({typeMap.get(t)?.length ?? 0})</SelectItem>
171
+ ))}
172
+ </SelectContent>
173
+ </Select>
174
+ </div>
175
+ )}
176
+ </div>
177
+
178
+ <div className="flex flex-col gap-1">
179
+ <div className="flex items-center justify-between">
180
+ <span className="font-medium text-muted-foreground">Columns{activeType ? ` — ${activeType}` : ''}</span>
181
+ {hasCustomColumns && (
182
+ <Button variant="ghost" size="sm" className="h-5 text-[10px]" onClick={resetColumns}>
183
+ reset
184
+ </Button>
185
+ )}
186
+ </div>
187
+
188
+ {fieldTree.length === 0 && (
189
+ <div className="py-2 text-muted-foreground">No fields detected</div>
190
+ )}
191
+
192
+ <div className="flex flex-col gap-0.5">
193
+ {fieldTree.map(field => (
194
+ <div key={field.key}>
195
+ <div className="flex items-center gap-1.5 rounded px-1 py-0.5 hover:bg-muted/50">
196
+ {field.children ? (
197
+ <Button
198
+ variant="ghost"
199
+ size="sm"
200
+ className="h-4 w-4 p-0 text-muted-foreground"
201
+ onClick={() => toggleExpand(field.key)}
202
+ >
203
+ {expanded.has(field.key) ? '▾' : '▸'}
204
+ </Button>
205
+ ) : (
206
+ <span className="w-4" />
207
+ )}
208
+ <label className="flex flex-1 cursor-pointer items-center gap-1.5">
209
+ <Checkbox
210
+ checked={isFieldVisible(field.key)}
211
+ onChange={() => toggleField(field.key, field.label)}
212
+ />
213
+ <span>{field.label}</span>
214
+ {field.type && (
215
+ <span className="text-[10px] text-muted-foreground/50">{field.type}</span>
216
+ )}
217
+ </label>
218
+ </div>
219
+
220
+ {field.children && expanded.has(field.key) && (
221
+ <div className="ml-5 flex flex-col gap-0.5">
222
+ {field.children.map(sub => (
223
+ <label key={sub.key} className="flex cursor-pointer items-center gap-1.5 rounded px-1 py-0.5 hover:bg-muted/50">
224
+ <Checkbox
225
+ checked={isFieldVisible(sub.key)}
226
+ onChange={() => toggleField(sub.key, sub.label)}
227
+ />
228
+ <span>{sub.label}</span>
229
+ </label>
230
+ ))}
231
+ </div>
232
+ )}
233
+ </div>
234
+ ))}
235
+ </div>
236
+ </div>
237
+ </div>
238
+ );
239
+ }
240
+
241
+ register('ui.table', 'react:edit', TableEditView as any);
@@ -0,0 +1 @@
1
+ import './types';