@pilotiq/pilotiq 0.7.2 → 0.8.0

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 (367) hide show
  1. package/.turbo/turbo-build.log +2 -2
  2. package/CHANGELOG.md +142 -0
  3. package/CLAUDE.md +59 -3
  4. package/dist/Pilotiq.d.ts +83 -0
  5. package/dist/Pilotiq.d.ts.map +1 -1
  6. package/dist/Pilotiq.js +39 -0
  7. package/dist/Pilotiq.js.map +1 -1
  8. package/dist/actions/Action.d.ts +27 -99
  9. package/dist/actions/Action.d.ts.map +1 -1
  10. package/dist/actions/Action.js +52 -754
  11. package/dist/actions/Action.js.map +1 -1
  12. package/dist/actions/bulkFactories.d.ts +46 -0
  13. package/dist/actions/bulkFactories.d.ts.map +1 -0
  14. package/dist/actions/bulkFactories.js +144 -0
  15. package/dist/actions/bulkFactories.js.map +1 -0
  16. package/dist/actions/crudFactories.d.ts +94 -0
  17. package/dist/actions/crudFactories.d.ts.map +1 -0
  18. package/dist/actions/crudFactories.js +209 -0
  19. package/dist/actions/crudFactories.js.map +1 -0
  20. package/dist/actions/factoryHelpers.d.ts +108 -0
  21. package/dist/actions/factoryHelpers.d.ts.map +1 -0
  22. package/dist/actions/factoryHelpers.js +138 -0
  23. package/dist/actions/factoryHelpers.js.map +1 -0
  24. package/dist/actions/m2mFactories.d.ts +47 -0
  25. package/dist/actions/m2mFactories.d.ts.map +1 -0
  26. package/dist/actions/m2mFactories.js +173 -0
  27. package/dist/actions/m2mFactories.js.map +1 -0
  28. package/dist/actions/relationFactories.d.ts +93 -0
  29. package/dist/actions/relationFactories.d.ts.map +1 -0
  30. package/dist/actions/relationFactories.js +321 -0
  31. package/dist/actions/relationFactories.js.map +1 -0
  32. package/dist/elements/dispatchForm.js +1 -1
  33. package/dist/elements/dispatchForm.js.map +1 -1
  34. package/dist/elements/dispatchTable.js +1 -1
  35. package/dist/elements/dispatchTable.js.map +1 -1
  36. package/dist/fields/Field.d.ts +31 -0
  37. package/dist/fields/Field.d.ts.map +1 -1
  38. package/dist/fields/Field.js +25 -0
  39. package/dist/fields/Field.js.map +1 -1
  40. package/dist/pageData/breadcrumbs.d.ts +42 -0
  41. package/dist/pageData/breadcrumbs.d.ts.map +1 -0
  42. package/dist/pageData/breadcrumbs.js +172 -0
  43. package/dist/pageData/breadcrumbs.js.map +1 -0
  44. package/dist/pageData/forms.d.ts +137 -0
  45. package/dist/pageData/forms.d.ts.map +1 -0
  46. package/dist/pageData/forms.js +427 -0
  47. package/dist/pageData/forms.js.map +1 -0
  48. package/dist/pageData/helpers.d.ts +239 -0
  49. package/dist/pageData/helpers.d.ts.map +1 -0
  50. package/dist/pageData/helpers.js +703 -0
  51. package/dist/pageData/helpers.js.map +1 -0
  52. package/dist/pageData/misc.d.ts +76 -0
  53. package/dist/pageData/misc.d.ts.map +1 -0
  54. package/dist/pageData/misc.js +263 -0
  55. package/dist/pageData/misc.js.map +1 -0
  56. package/dist/pageData/navigation.d.ts +292 -0
  57. package/dist/pageData/navigation.d.ts.map +1 -0
  58. package/dist/pageData/navigation.js +591 -0
  59. package/dist/pageData/navigation.js.map +1 -0
  60. package/dist/pageData/relationPages.d.ts +172 -0
  61. package/dist/pageData/relationPages.d.ts.map +1 -0
  62. package/dist/pageData/relationPages.js +867 -0
  63. package/dist/pageData/relationPages.js.map +1 -0
  64. package/dist/pageData/relationTabs.d.ts +65 -0
  65. package/dist/pageData/relationTabs.d.ts.map +1 -0
  66. package/dist/pageData/relationTabs.js +258 -0
  67. package/dist/pageData/relationTabs.js.map +1 -0
  68. package/dist/pageData/resourcePages.d.ts +48 -0
  69. package/dist/pageData/resourcePages.d.ts.map +1 -0
  70. package/dist/pageData/resourcePages.js +504 -0
  71. package/dist/pageData/resourcePages.js.map +1 -0
  72. package/dist/pageData.d.ts +12 -792
  73. package/dist/pageData.d.ts.map +1 -1
  74. package/dist/pageData.js +24 -3797
  75. package/dist/pageData.js.map +1 -1
  76. package/dist/react/AppShell.d.ts +8 -0
  77. package/dist/react/AppShell.d.ts.map +1 -1
  78. package/dist/react/AppShell.js +11 -1
  79. package/dist/react/AppShell.js.map +1 -1
  80. package/dist/react/CollabExtensionFactoryRegistry.d.ts +47 -0
  81. package/dist/react/CollabExtensionFactoryRegistry.d.ts.map +1 -0
  82. package/dist/react/CollabExtensionFactoryRegistry.js +14 -0
  83. package/dist/react/CollabExtensionFactoryRegistry.js.map +1 -0
  84. package/dist/react/CollabRoomContext.d.ts +37 -0
  85. package/dist/react/CollabRoomContext.d.ts.map +1 -0
  86. package/dist/react/CollabRoomContext.js +12 -0
  87. package/dist/react/CollabRoomContext.js.map +1 -0
  88. package/dist/react/FormCollabBindingRegistry.d.ts +62 -0
  89. package/dist/react/FormCollabBindingRegistry.d.ts.map +1 -0
  90. package/dist/react/FormCollabBindingRegistry.js +14 -0
  91. package/dist/react/FormCollabBindingRegistry.js.map +1 -0
  92. package/dist/react/RecordWrapperGate.d.ts +25 -0
  93. package/dist/react/RecordWrapperGate.d.ts.map +1 -0
  94. package/dist/react/RecordWrapperGate.js +30 -0
  95. package/dist/react/RecordWrapperGate.js.map +1 -0
  96. package/dist/react/RecordWrapperRegistry.d.ts +31 -0
  97. package/dist/react/RecordWrapperRegistry.d.ts.map +1 -0
  98. package/dist/react/RecordWrapperRegistry.js +15 -0
  99. package/dist/react/RecordWrapperRegistry.js.map +1 -0
  100. package/dist/react/SchemaRenderer.d.ts +17 -23
  101. package/dist/react/SchemaRenderer.d.ts.map +1 -1
  102. package/dist/react/SchemaRenderer.js +71 -3647
  103. package/dist/react/SchemaRenderer.js.map +1 -1
  104. package/dist/react/component-slots.d.ts +103 -0
  105. package/dist/react/component-slots.d.ts.map +1 -0
  106. package/dist/react/component-slots.js +18 -0
  107. package/dist/react/component-slots.js.map +1 -0
  108. package/dist/react/fields/BuilderInput.d.ts.map +1 -1
  109. package/dist/react/fields/BuilderInput.js +21 -117
  110. package/dist/react/fields/BuilderInput.js.map +1 -1
  111. package/dist/react/fields/MarkdownInput.d.ts.map +1 -1
  112. package/dist/react/fields/MarkdownInput.js +1 -3
  113. package/dist/react/fields/MarkdownInput.js.map +1 -1
  114. package/dist/react/fields/RepeaterInput.d.ts.map +1 -1
  115. package/dist/react/fields/RepeaterInput.js +22 -127
  116. package/dist/react/fields/RepeaterInput.js.map +1 -1
  117. package/dist/react/fields/rowState.d.ts +40 -0
  118. package/dist/react/fields/rowState.d.ts.map +1 -0
  119. package/dist/react/fields/rowState.js +60 -0
  120. package/dist/react/fields/rowState.js.map +1 -0
  121. package/dist/react/fields/useRowReorderDnd.d.ts +28 -0
  122. package/dist/react/fields/useRowReorderDnd.d.ts.map +1 -0
  123. package/dist/react/fields/useRowReorderDnd.js +51 -0
  124. package/dist/react/fields/useRowReorderDnd.js.map +1 -0
  125. package/dist/react/index.d.ts +9 -0
  126. package/dist/react/index.d.ts.map +1 -1
  127. package/dist/react/index.js +8 -0
  128. package/dist/react/index.js.map +1 -1
  129. package/dist/react/layouts/SidebarLayout.d.ts +1 -1
  130. package/dist/react/layouts/SidebarLayout.d.ts.map +1 -1
  131. package/dist/react/layouts/SidebarLayout.js +10 -2
  132. package/dist/react/layouts/SidebarLayout.js.map +1 -1
  133. package/dist/react/layouts/TopbarLayout.d.ts +1 -1
  134. package/dist/react/layouts/TopbarLayout.d.ts.map +1 -1
  135. package/dist/react/layouts/TopbarLayout.js +19 -11
  136. package/dist/react/layouts/TopbarLayout.js.map +1 -1
  137. package/dist/react/parseRecordEditUrl.d.ts +29 -0
  138. package/dist/react/parseRecordEditUrl.d.ts.map +1 -0
  139. package/dist/react/parseRecordEditUrl.js +25 -0
  140. package/dist/react/parseRecordEditUrl.js.map +1 -0
  141. package/dist/react/persistedState.d.ts +19 -0
  142. package/dist/react/persistedState.d.ts.map +1 -0
  143. package/dist/react/persistedState.js +51 -0
  144. package/dist/react/persistedState.js.map +1 -0
  145. package/dist/react/schemaRenderer/AlertRenderer.d.ts +12 -0
  146. package/dist/react/schemaRenderer/AlertRenderer.d.ts.map +1 -0
  147. package/dist/react/schemaRenderer/AlertRenderer.js +61 -0
  148. package/dist/react/schemaRenderer/AlertRenderer.js.map +1 -0
  149. package/dist/react/schemaRenderer/EntryRenderer.d.ts +13 -0
  150. package/dist/react/schemaRenderer/EntryRenderer.d.ts.map +1 -0
  151. package/dist/react/schemaRenderer/EntryRenderer.js +277 -0
  152. package/dist/react/schemaRenderer/EntryRenderer.js.map +1 -0
  153. package/dist/react/schemaRenderer/SectionRenderer.d.ts +16 -0
  154. package/dist/react/schemaRenderer/SectionRenderer.d.ts.map +1 -0
  155. package/dist/react/schemaRenderer/SectionRenderer.js +62 -0
  156. package/dist/react/schemaRenderer/SectionRenderer.js.map +1 -0
  157. package/dist/react/schemaRenderer/SimpleElements.d.ts +25 -0
  158. package/dist/react/schemaRenderer/SimpleElements.d.ts.map +1 -0
  159. package/dist/react/schemaRenderer/SimpleElements.js +147 -0
  160. package/dist/react/schemaRenderer/SimpleElements.js.map +1 -0
  161. package/dist/react/schemaRenderer/TabsRenderer.d.ts +17 -0
  162. package/dist/react/schemaRenderer/TabsRenderer.d.ts.map +1 -0
  163. package/dist/react/schemaRenderer/TabsRenderer.js +31 -0
  164. package/dist/react/schemaRenderer/TabsRenderer.js.map +1 -0
  165. package/dist/react/schemaRenderer/WizardRenderer.d.ts +34 -0
  166. package/dist/react/schemaRenderer/WizardRenderer.d.ts.map +1 -0
  167. package/dist/react/schemaRenderer/WizardRenderer.js +208 -0
  168. package/dist/react/schemaRenderer/WizardRenderer.js.map +1 -0
  169. package/dist/react/schemaRenderer/action/ActionGroupTrigger.d.ts +21 -0
  170. package/dist/react/schemaRenderer/action/ActionGroupTrigger.d.ts.map +1 -0
  171. package/dist/react/schemaRenderer/action/ActionGroupTrigger.js +82 -0
  172. package/dist/react/schemaRenderer/action/ActionGroupTrigger.js.map +1 -0
  173. package/dist/react/schemaRenderer/action/ActionModalDialog.d.ts +30 -0
  174. package/dist/react/schemaRenderer/action/ActionModalDialog.d.ts.map +1 -0
  175. package/dist/react/schemaRenderer/action/ActionModalDialog.js +182 -0
  176. package/dist/react/schemaRenderer/action/ActionModalDialog.js.map +1 -0
  177. package/dist/react/schemaRenderer/action/ConfirmActionDialog.d.ts +17 -0
  178. package/dist/react/schemaRenderer/action/ConfirmActionDialog.d.ts.map +1 -0
  179. package/dist/react/schemaRenderer/action/ConfirmActionDialog.js +19 -0
  180. package/dist/react/schemaRenderer/action/ConfirmActionDialog.js.map +1 -0
  181. package/dist/react/schemaRenderer/action/HandlerActionButton.d.ts +16 -0
  182. package/dist/react/schemaRenderer/action/HandlerActionButton.d.ts.map +1 -0
  183. package/dist/react/schemaRenderer/action/HandlerActionButton.js +16 -0
  184. package/dist/react/schemaRenderer/action/HandlerActionButton.js.map +1 -0
  185. package/dist/react/schemaRenderer/action/MethodActionButton.d.ts +22 -0
  186. package/dist/react/schemaRenderer/action/MethodActionButton.d.ts.map +1 -0
  187. package/dist/react/schemaRenderer/action/MethodActionButton.js +26 -0
  188. package/dist/react/schemaRenderer/action/MethodActionButton.js.map +1 -0
  189. package/dist/react/schemaRenderer/action/buttons.d.ts +18 -0
  190. package/dist/react/schemaRenderer/action/buttons.d.ts.map +1 -0
  191. package/dist/react/schemaRenderer/action/buttons.js +74 -0
  192. package/dist/react/schemaRenderer/action/buttons.js.map +1 -0
  193. package/dist/react/schemaRenderer/action/helpers.d.ts +26 -0
  194. package/dist/react/schemaRenderer/action/helpers.d.ts.map +1 -0
  195. package/dist/react/schemaRenderer/action/helpers.js +126 -0
  196. package/dist/react/schemaRenderer/action/helpers.js.map +1 -0
  197. package/dist/react/schemaRenderer/action/renderAction.d.ts +21 -0
  198. package/dist/react/schemaRenderer/action/renderAction.d.ts.map +1 -0
  199. package/dist/react/schemaRenderer/action/renderAction.js +102 -0
  200. package/dist/react/schemaRenderer/action/renderAction.js.map +1 -0
  201. package/dist/react/schemaRenderer/columnFormat.d.ts +10 -0
  202. package/dist/react/schemaRenderer/columnFormat.d.ts.map +1 -0
  203. package/dist/react/schemaRenderer/columnFormat.js +76 -0
  204. package/dist/react/schemaRenderer/columnFormat.js.map +1 -0
  205. package/dist/react/schemaRenderer/constants.d.ts +8 -0
  206. package/dist/react/schemaRenderer/constants.d.ts.map +1 -0
  207. package/dist/react/schemaRenderer/constants.js +45 -0
  208. package/dist/react/schemaRenderer/constants.js.map +1 -0
  209. package/dist/react/schemaRenderer/form/FormRenderer.d.ts +29 -0
  210. package/dist/react/schemaRenderer/form/FormRenderer.d.ts.map +1 -0
  211. package/dist/react/schemaRenderer/form/FormRenderer.js +152 -0
  212. package/dist/react/schemaRenderer/form/FormRenderer.js.map +1 -0
  213. package/dist/react/schemaRenderer/form/renderField.d.ts +6 -0
  214. package/dist/react/schemaRenderer/form/renderField.d.ts.map +1 -0
  215. package/dist/react/schemaRenderer/form/renderField.js +239 -0
  216. package/dist/react/schemaRenderer/form/renderField.js.map +1 -0
  217. package/dist/react/schemaRenderer/helpers.d.ts +32 -0
  218. package/dist/react/schemaRenderer/helpers.d.ts.map +1 -0
  219. package/dist/react/schemaRenderer/helpers.js +52 -0
  220. package/dist/react/schemaRenderer/helpers.js.map +1 -0
  221. package/dist/react/schemaRenderer/table/CardsLayoutBody.d.ts +60 -0
  222. package/dist/react/schemaRenderer/table/CardsLayoutBody.d.ts.map +1 -0
  223. package/dist/react/schemaRenderer/table/CardsLayoutBody.js +189 -0
  224. package/dist/react/schemaRenderer/table/CardsLayoutBody.js.map +1 -0
  225. package/dist/react/schemaRenderer/table/TableRenderer.d.ts +29 -0
  226. package/dist/react/schemaRenderer/table/TableRenderer.d.ts.map +1 -0
  227. package/dist/react/schemaRenderer/table/TableRenderer.js +85 -0
  228. package/dist/react/schemaRenderer/table/TableRenderer.js.map +1 -0
  229. package/dist/react/schemaRenderer/table/TableRendererBody.d.ts +18 -0
  230. package/dist/react/schemaRenderer/table/TableRendererBody.d.ts.map +1 -0
  231. package/dist/react/schemaRenderer/table/TableRendererBody.js +555 -0
  232. package/dist/react/schemaRenderer/table/TableRendererBody.js.map +1 -0
  233. package/dist/react/schemaRenderer/table/filters.d.ts +263 -0
  234. package/dist/react/schemaRenderer/table/filters.d.ts.map +1 -0
  235. package/dist/react/schemaRenderer/table/filters.js +497 -0
  236. package/dist/react/schemaRenderer/table/filters.js.map +1 -0
  237. package/dist/react/schemaRenderer/table/formatCell.d.ts +11 -0
  238. package/dist/react/schemaRenderer/table/formatCell.d.ts.map +1 -0
  239. package/dist/react/schemaRenderer/table/formatCell.js +172 -0
  240. package/dist/react/schemaRenderer/table/formatCell.js.map +1 -0
  241. package/dist/react/schemaRenderer/table/links.d.ts +42 -0
  242. package/dist/react/schemaRenderer/table/links.d.ts.map +1 -0
  243. package/dist/react/schemaRenderer/table/links.js +55 -0
  244. package/dist/react/schemaRenderer/table/links.js.map +1 -0
  245. package/dist/react/schemaRenderer/table/renderRowActions.d.ts +13 -0
  246. package/dist/react/schemaRenderer/table/renderRowActions.d.ts.map +1 -0
  247. package/dist/react/schemaRenderer/table/renderRowActions.js +25 -0
  248. package/dist/react/schemaRenderer/table/renderRowActions.js.map +1 -0
  249. package/dist/react/schemaRenderer/table/url.d.ts +41 -0
  250. package/dist/react/schemaRenderer/table/url.d.ts.map +1 -0
  251. package/dist/react/schemaRenderer/table/url.js +114 -0
  252. package/dist/react/schemaRenderer/table/url.js.map +1 -0
  253. package/dist/routes/globals.d.ts +13 -0
  254. package/dist/routes/globals.d.ts.map +1 -0
  255. package/dist/routes/globals.js +131 -0
  256. package/dist/routes/globals.js.map +1 -0
  257. package/dist/routes/helpers.d.ts +217 -0
  258. package/dist/routes/helpers.d.ts.map +1 -0
  259. package/dist/routes/helpers.js +498 -0
  260. package/dist/routes/helpers.js.map +1 -0
  261. package/dist/routes/pages.d.ts +15 -0
  262. package/dist/routes/pages.d.ts.map +1 -0
  263. package/dist/routes/pages.js +145 -0
  264. package/dist/routes/pages.js.map +1 -0
  265. package/dist/routes/panel.d.ts +19 -0
  266. package/dist/routes/panel.d.ts.map +1 -0
  267. package/dist/routes/panel.js +191 -0
  268. package/dist/routes/panel.js.map +1 -0
  269. package/dist/routes/relations.d.ts +21 -0
  270. package/dist/routes/relations.d.ts.map +1 -0
  271. package/dist/routes/relations.js +1239 -0
  272. package/dist/routes/relations.js.map +1 -0
  273. package/dist/routes/resources.d.ts +28 -0
  274. package/dist/routes/resources.d.ts.map +1 -0
  275. package/dist/routes/resources.js +741 -0
  276. package/dist/routes/resources.js.map +1 -0
  277. package/dist/routes/theme.d.ts +12 -0
  278. package/dist/routes/theme.d.ts.map +1 -0
  279. package/dist/routes/theme.js +82 -0
  280. package/dist/routes/theme.js.map +1 -0
  281. package/dist/routes.d.ts.map +1 -1
  282. package/dist/routes.js +64 -3078
  283. package/dist/routes.js.map +1 -1
  284. package/dist/vite.d.ts +1 -0
  285. package/dist/vite.d.ts.map +1 -1
  286. package/dist/vite.js +26 -5
  287. package/dist/vite.js.map +1 -1
  288. package/package.json +2 -1
  289. package/src/Pilotiq.ts +95 -0
  290. package/src/actions/Action.ts +79 -723
  291. package/src/actions/bulkFactories.ts +168 -0
  292. package/src/actions/crudFactories.ts +220 -0
  293. package/src/actions/factoryHelpers.ts +177 -0
  294. package/src/actions/m2mFactories.ts +193 -0
  295. package/src/actions/relationFactories.ts +372 -0
  296. package/src/elements/dispatchForm.ts +1 -1
  297. package/src/elements/dispatchTable.ts +1 -1
  298. package/src/fields/Field.ts +39 -0
  299. package/src/pageData/breadcrumbs.ts +288 -0
  300. package/src/pageData/forms.ts +578 -0
  301. package/src/pageData/helpers.ts +764 -0
  302. package/src/pageData/misc.ts +347 -0
  303. package/src/pageData/navigation.ts +779 -0
  304. package/src/pageData/relationPages.ts +1246 -0
  305. package/src/pageData/relationTabs.ts +286 -0
  306. package/src/pageData/resourcePages.ts +593 -0
  307. package/src/pageData.ts +122 -4731
  308. package/src/react/AppShell.tsx +27 -1
  309. package/src/react/CollabExtensionFactoryRegistry.ts +55 -0
  310. package/src/react/CollabRoomContext.ts +42 -0
  311. package/src/react/FormCollabBindingRegistry.ts +72 -0
  312. package/src/react/RecordWrapperGate.tsx +40 -0
  313. package/src/react/RecordWrapperRegistry.ts +39 -0
  314. package/src/react/SchemaRenderer.tsx +230 -6479
  315. package/src/react/component-slots.test.ts +103 -0
  316. package/src/react/component-slots.ts +116 -0
  317. package/src/react/fields/BuilderInput.tsx +29 -117
  318. package/src/react/fields/MarkdownInput.tsx +0 -1
  319. package/src/react/fields/RepeaterInput.tsx +29 -130
  320. package/src/react/fields/rowState.ts +106 -0
  321. package/src/react/fields/useRowReorderDnd.ts +78 -0
  322. package/src/react/index.ts +38 -0
  323. package/src/react/layouts/SidebarLayout.tsx +39 -28
  324. package/src/react/layouts/TopbarLayout.tsx +70 -57
  325. package/src/react/parseRecordEditUrl.test.ts +75 -0
  326. package/src/react/parseRecordEditUrl.ts +55 -0
  327. package/src/react/persistedState.ts +40 -0
  328. package/src/react/schemaRenderer/AlertRenderer.tsx +112 -0
  329. package/src/react/schemaRenderer/EntryRenderer.tsx +501 -0
  330. package/src/react/schemaRenderer/SectionRenderer.tsx +120 -0
  331. package/src/react/schemaRenderer/SimpleElements.tsx +306 -0
  332. package/src/react/schemaRenderer/TabsRenderer.tsx +62 -0
  333. package/src/react/schemaRenderer/WizardRenderer.tsx +338 -0
  334. package/src/react/schemaRenderer/action/ActionGroupTrigger.tsx +177 -0
  335. package/src/react/schemaRenderer/action/ActionModalDialog.tsx +273 -0
  336. package/src/react/schemaRenderer/action/ConfirmActionDialog.tsx +61 -0
  337. package/src/react/schemaRenderer/action/HandlerActionButton.tsx +43 -0
  338. package/src/react/schemaRenderer/action/MethodActionButton.tsx +64 -0
  339. package/src/react/schemaRenderer/action/buttons.tsx +99 -0
  340. package/src/react/schemaRenderer/action/helpers.ts +140 -0
  341. package/src/react/schemaRenderer/action/renderAction.tsx +245 -0
  342. package/src/react/schemaRenderer/columnFormat.ts +65 -0
  343. package/src/react/schemaRenderer/constants.ts +50 -0
  344. package/src/react/schemaRenderer/form/FormRenderer.tsx +233 -0
  345. package/src/react/schemaRenderer/form/renderField.tsx +511 -0
  346. package/src/react/schemaRenderer/helpers.tsx +81 -0
  347. package/src/react/schemaRenderer/table/CardsLayoutBody.tsx +308 -0
  348. package/src/react/schemaRenderer/table/TableRenderer.tsx +123 -0
  349. package/src/react/schemaRenderer/table/TableRendererBody.tsx +974 -0
  350. package/src/react/schemaRenderer/table/filters.tsx +1233 -0
  351. package/src/react/schemaRenderer/table/formatCell.tsx +264 -0
  352. package/src/react/schemaRenderer/table/links.tsx +112 -0
  353. package/src/react/schemaRenderer/table/renderRowActions.tsx +52 -0
  354. package/src/react/schemaRenderer/table/url.tsx +143 -0
  355. package/src/routes/globals.ts +154 -0
  356. package/src/routes/helpers.ts +668 -0
  357. package/src/routes/pages.ts +173 -0
  358. package/src/routes/panel.ts +204 -0
  359. package/src/routes/relations.ts +1219 -0
  360. package/src/routes/resources.ts +786 -0
  361. package/src/routes/theme.ts +109 -0
  362. package/src/routes.test.ts +1 -1
  363. package/src/routes.ts +64 -3176
  364. package/src/schema/TableWidget.test.ts +2 -2
  365. package/src/theme/migrate.test.ts +178 -0
  366. package/src/vite.test.ts +184 -0
  367. package/src/vite.ts +26 -4
@@ -0,0 +1,338 @@
1
+ import React, { useEffect, useState } from 'react'
2
+ import type { ElementMeta } from '../../schema/Element.js'
3
+ import { useFormState } from '../FormStateContext.js'
4
+ import { readStoredString, writeStoredString } from '../persistedState.js'
5
+ import { layoutClasses, resolveIcon, withTooltip } from './helpers.js'
6
+ import { actionButtonClass, renderActionBadge, renderActionIcon } from './action/buttons.js'
7
+
8
+ // ─── Wizard (Plan #8) ───────────────────────────────────────
9
+
10
+ /**
11
+ * Deps injected from the top-level dispatch so the wizard body can
12
+ * recurse into the main element renderer without creating a cycle
13
+ * with SchemaRenderer.tsx.
14
+ */
15
+ export interface WizardRendererDeps {
16
+ renderElement: (el: ElementMeta, index: number) => React.ReactNode
17
+ }
18
+
19
+ /**
20
+ * Resolve the initial active step for `WizardRenderer`. Priority:
21
+ * 1. URL `?<queryKey>=N` (1-based — wizards expose human-friendly indexes
22
+ * when `Wizard.persistStepInQueryString()` is enabled).
23
+ * 2. `localStorage[<storageKey>]` (0-based, set by the persist effect).
24
+ * 3. `startOnStep` configured on the Wizard.
25
+ *
26
+ * SSR-safe: returns `startOnStep` when `window` is undefined.
27
+ */
28
+ function readInitialWizardStep(
29
+ total: number,
30
+ startOnStep: number,
31
+ storageKey: string | undefined,
32
+ queryKey: string | undefined,
33
+ ): number {
34
+ if (typeof window === 'undefined') return startOnStep
35
+ if (queryKey) {
36
+ try {
37
+ const raw = new URL(window.location.href).searchParams.get(queryKey)
38
+ if (raw !== null && raw !== '') {
39
+ const n = Number(raw) - 1
40
+ if (Number.isFinite(n) && n >= 0 && n < total) return n
41
+ }
42
+ } catch { /* ignore */ }
43
+ }
44
+ if (storageKey) {
45
+ const stored = readStoredString(storageKey)
46
+ if (stored !== null) {
47
+ const n = Number(stored)
48
+ if (Number.isFinite(n) && n >= 0 && n < total) return n
49
+ }
50
+ }
51
+ return startOnStep
52
+ }
53
+
54
+ /**
55
+ * Multi-step form layout. Tracks active step in `useState`, optionally
56
+ * persisted to localStorage and/or the URL query string. On Next click,
57
+ * POSTs `{ step, values }` to the form's `wizardUrl` (stamped by the
58
+ * route handler when the form has a Wizard descendant). 200 → advance;
59
+ * 422 → stamp inline errors; absent `wizardUrl` → advance immediately
60
+ * (no validation).
61
+ *
62
+ * Inactive steps render hidden (display:none) rather than unmounted so
63
+ * controlled inputs preserve their values across step transitions and
64
+ * cross-step `$get` works on the resolved meta.
65
+ *
66
+ * Nav buttons honor `Wizard.submitAction() / nextAction() / previousAction()`
67
+ * — chrome (label / icon / color / size / outlined / iconOnly / tooltip /
68
+ * disabled rules) carries through to the rendered button while the click
69
+ * behavior stays hardwired (advance / recede / submit-form). Bare wizards
70
+ * keep the built-in defaults.
71
+ */
72
+ export function WizardRenderer({
73
+ el,
74
+ index,
75
+ deps,
76
+ }: {
77
+ el: ElementMeta
78
+ index: number
79
+ deps: WizardRendererDeps
80
+ }) {
81
+ const { renderElement } = deps
82
+ const formState = useFormState()
83
+ const formId = formState?.formMeta['formId'] ? String(formState.formMeta['formId']) : undefined
84
+ const wizardUrl = formState?.formMeta['wizardUrl'] ? String(formState.formMeta['wizardUrl']) : undefined
85
+
86
+ const steps = (el.children ?? []).filter(c => c.type === 'step')
87
+ const skippable = Boolean(el['skippable'])
88
+ const startOnStep = Math.max(0, Math.min(Math.max(0, steps.length - 1), Number(el['startOnStep'] ?? 0)))
89
+ const persist = el['persist'] !== false
90
+ const storageKey = persist && formId ? `pilotiq.wizard.${formId}.step` : undefined
91
+ const queryKey = typeof el['persistStepInQueryString'] === 'string'
92
+ ? String(el['persistStepInQueryString'])
93
+ : undefined
94
+
95
+ const submitActionMeta = el['submitAction'] as ElementMeta | undefined
96
+ const nextActionMeta = el['nextAction'] as ElementMeta | undefined
97
+ const previousActionMeta = el['previousAction'] as ElementMeta | undefined
98
+
99
+ // Initial-step resolution priority: URL (?<key>=N, 1-based) > localStorage >
100
+ // startOnStep. URL wins on first paint so deep links land on the right step
101
+ // before localStorage can override. Lazy initializer — resolution runs once.
102
+ const [active, setActive] = useState(() => readInitialWizardStep(steps.length, startOnStep, storageKey, queryKey))
103
+ const [advancing, setAdvancing] = useState(false)
104
+ const [advanceError, setAdvanceError] = useState<string | null>(null)
105
+
106
+ // Persist active step changes to localStorage (when enabled).
107
+ useEffect(() => {
108
+ if (!storageKey) return
109
+ writeStoredString(storageKey, String(active))
110
+ }, [storageKey, active])
111
+
112
+ // Mirror active step to the URL via replaceState — purely client-side state
113
+ // sync, no SPA re-fetch. 1-based externally; cleared when on the first step
114
+ // so bare URLs don't grow ?step=1 noise.
115
+ useEffect(() => {
116
+ if (!queryKey) return
117
+ if (typeof window === 'undefined') return
118
+ try {
119
+ const url = new URL(window.location.href)
120
+ if (active === 0) url.searchParams.delete(queryKey)
121
+ else url.searchParams.set(queryKey, String(active + 1))
122
+ window.history.replaceState(window.history.state, '', url.toString())
123
+ } catch { /* ignore */ }
124
+ }, [queryKey, active])
125
+
126
+ if (steps.length === 0) {
127
+ return (
128
+ <div key={index} className="rounded-lg border border-dashed p-8 text-center text-sm text-muted-foreground">
129
+ No steps configured.
130
+ </div>
131
+ )
132
+ }
133
+
134
+ const isLast = active === steps.length - 1
135
+ const isFirst = active === 0
136
+
137
+ const advance = async (target: number) => {
138
+ setAdvanceError(null)
139
+ if (!wizardUrl) {
140
+ setActive(target)
141
+ return
142
+ }
143
+ setAdvancing(true)
144
+ try {
145
+ const values = formState?.values ?? {}
146
+ // Validate intermediate steps in order when jumping ahead.
147
+ const path = target > active
148
+ ? Array.from({ length: target - active }, (_, k) => active + k)
149
+ : [active] // jumping back is unconstrained
150
+ let landed = active
151
+ for (const stepIdx of path) {
152
+ const res = await fetch(wizardUrl, {
153
+ method: 'POST',
154
+ headers: { 'Content-Type': 'application/json', 'Accept': 'application/json' },
155
+ body: JSON.stringify({ step: stepIdx, values }),
156
+ })
157
+ if (res.status === 422) {
158
+ const data = await res.json().catch(() => ({}))
159
+ const errors = (data?.errors ?? {}) as Record<string, string[]>
160
+ if (formState?.applyErrors) formState.applyErrors(errors)
161
+ landed = stepIdx
162
+ setAdvanceError('Please fix the highlighted fields.')
163
+ break
164
+ }
165
+ if (!res.ok) {
166
+ setAdvanceError('Step validation failed.')
167
+ break
168
+ }
169
+ landed = stepIdx + 1
170
+ }
171
+ setActive(target > active ? landed : target)
172
+ } catch {
173
+ setAdvanceError('Step validation failed.')
174
+ } finally {
175
+ setAdvancing(false)
176
+ }
177
+ }
178
+
179
+ return (
180
+ <div key={index} className={`flex flex-col gap-6 ${layoutClasses(el)}`.trim()}>
181
+ {/* Step indicator */}
182
+ <ol className="flex items-center gap-3 overflow-x-auto" aria-label="Wizard progress">
183
+ {steps.map((s, i) => {
184
+ const Icon = resolveIcon(s['icon'] ? String(s['icon']) : undefined)
185
+ const reachable = skippable || i <= active
186
+ const isActive = i === active
187
+ const isDone = i < active
188
+ return (
189
+ <li key={i} className="flex items-center gap-2 shrink-0">
190
+ <button
191
+ type="button"
192
+ disabled={!reachable || advancing}
193
+ onClick={() => reachable && advance(i)}
194
+ className={[
195
+ 'flex items-center gap-2 rounded-md border px-3 py-1.5 text-sm transition',
196
+ isActive ? 'border-primary bg-primary/10 text-foreground'
197
+ : isDone ? 'border-border text-muted-foreground hover:bg-muted'
198
+ : 'border-border text-muted-foreground',
199
+ reachable ? 'cursor-pointer' : 'opacity-50 cursor-not-allowed',
200
+ ].join(' ')}
201
+ aria-current={isActive ? 'step' : undefined}
202
+ >
203
+ <span className={[
204
+ 'flex size-5 items-center justify-center rounded-full text-[11px] font-semibold',
205
+ isActive ? 'bg-primary text-primary-foreground'
206
+ : isDone ? 'bg-muted-foreground/20 text-foreground'
207
+ : 'bg-muted text-muted-foreground',
208
+ ].join(' ')}>
209
+ {Icon ? <Icon className="size-3" aria-hidden="true" /> : i + 1}
210
+ </span>
211
+ <span className="font-medium">{String(s['label'] ?? `Step ${i + 1}`)}</span>
212
+ </button>
213
+ {i < steps.length - 1 && <span className="h-px w-6 bg-border" aria-hidden="true" />}
214
+ </li>
215
+ )
216
+ })}
217
+ </ol>
218
+
219
+ {/* Active step heading */}
220
+ {Boolean(steps[active]?.['description']) && (
221
+ <p className="text-sm text-muted-foreground">{String(steps[active]!['description'])}</p>
222
+ )}
223
+
224
+ {/* Step content. Inactive steps render hidden so controlled inputs survive. */}
225
+ {steps.map((s, i) => (
226
+ <div
227
+ key={i}
228
+ className={i === active ? 'flex flex-col gap-4' : 'hidden'}
229
+ aria-hidden={i === active ? undefined : true}
230
+ >
231
+ {(s.children ?? []).map((c, ci) => renderElement(c, ci))}
232
+ </div>
233
+ ))}
234
+
235
+ {advanceError && (
236
+ <p className="text-sm text-destructive" role="alert">{advanceError}</p>
237
+ )}
238
+
239
+ {/* Step nav. The Form's Save button (in Heading actions or page chrome)
240
+ stays as-is by default — only the final step's submit goes through
241
+ the form. `Wizard.submitAction()` opts into a wizard-owned submit
242
+ button on the final step (use this when the wizard is the entire
243
+ form and there's no page chrome). `nextAction() / previousAction()`
244
+ customize the chrome of the built-in Back / Next buttons. */}
245
+ <div className="flex items-center justify-between gap-2">
246
+ <WizardNavButton
247
+ actionMeta={previousActionMeta}
248
+ fallbackLabel="Back"
249
+ disabled={isFirst || advancing}
250
+ onClick={() => advance(active - 1)}
251
+ />
252
+ {isLast
253
+ ? (submitActionMeta
254
+ ? <WizardNavButton
255
+ actionMeta={submitActionMeta}
256
+ fallbackLabel="Submit"
257
+ type="submit"
258
+ disabled={advancing}
259
+ />
260
+ : <span className="text-xs text-muted-foreground">Submit the form to finish.</span>)
261
+ : <WizardNavButton
262
+ actionMeta={nextActionMeta}
263
+ fallbackLabel={advancing ? 'Validating…' : 'Next'}
264
+ disabled={advancing}
265
+ onClick={() => advance(active + 1)}
266
+ />
267
+ }
268
+ </div>
269
+ </div>
270
+ )
271
+ }
272
+
273
+ /**
274
+ * Renders one wizard nav slot (Back / Next / Submit). Falls back to plain
275
+ * built-in chrome (border button for Back, primary button for Next/Submit)
276
+ * when no `actionMeta` is supplied; otherwise reads the resolved Action's
277
+ * chrome (`label / icon / color / size / outlined / iconOnly / tooltip /
278
+ * disabled`) and applies it to a button whose click is hardwired by the
279
+ * surrounding wizard. `type="submit"` lets the Submit slot trigger the
280
+ * surrounding form's onSubmit dispatcher (no `onClick` needed).
281
+ *
282
+ * Hidden actions (`.visible(false)` resolved-away) drop the slot entirely
283
+ * — the resolver returns `undefined` for hidden Action elements, which
284
+ * arrives here as `actionMeta == null` so we fall through to the default
285
+ * chrome. Use `Wizard.skippable()` semantics to hide nav buttons when
286
+ * appropriate; for permanent removal subclass the wizard.
287
+ */
288
+ function WizardNavButton({
289
+ actionMeta,
290
+ fallbackLabel,
291
+ type = 'button',
292
+ disabled,
293
+ onClick,
294
+ }: {
295
+ actionMeta: ElementMeta | undefined
296
+ fallbackLabel: string
297
+ type?: 'button' | 'submit'
298
+ disabled?: boolean
299
+ onClick?: () => void
300
+ }) {
301
+ // Bare default — keep historical chrome for back-compat (un-customized
302
+ // wizards look identical to before this change).
303
+ if (!actionMeta) {
304
+ const isPrimary = type === 'submit' || fallbackLabel !== 'Back'
305
+ return (
306
+ <button
307
+ type={type}
308
+ disabled={disabled}
309
+ onClick={onClick}
310
+ className={isPrimary
311
+ ? 'rounded-md bg-primary px-3 py-1.5 text-sm font-medium text-primary-foreground hover:opacity-90 disabled:opacity-50 disabled:cursor-not-allowed'
312
+ : 'rounded-md border border-border px-3 py-1.5 text-sm font-medium text-foreground hover:bg-muted disabled:opacity-50 disabled:cursor-not-allowed'}
313
+ >
314
+ {fallbackLabel}
315
+ </button>
316
+ )
317
+ }
318
+
319
+ const ownDisabled = Boolean(actionMeta['disabled'])
320
+ const label = String(actionMeta['label'] ?? fallbackLabel)
321
+ const tooltip = actionMeta['tooltip'] ? String(actionMeta['tooltip']) : undefined
322
+ const iconOnly = Boolean(actionMeta['iconOnly'])
323
+ const className = actionButtonClass(actionMeta, {})
324
+ const node = (
325
+ <button
326
+ type={type}
327
+ disabled={disabled || ownDisabled}
328
+ onClick={onClick}
329
+ className={`${className} disabled:opacity-50 disabled:cursor-not-allowed`}
330
+ aria-label={iconOnly ? label : undefined}
331
+ >
332
+ {renderActionIcon(actionMeta)}
333
+ {!iconOnly && <span>{label}</span>}
334
+ {renderActionBadge(actionMeta)}
335
+ </button>
336
+ )
337
+ return <>{withTooltip(node, tooltip)}</>
338
+ }
@@ -0,0 +1,177 @@
1
+ import React, { useState } from 'react'
2
+ import type { ElementMeta } from '../../../schema/Element.js'
3
+ import {
4
+ Dialog, DialogContent, DialogDescription, DialogFooter, DialogHeader, DialogTitle,
5
+ } from '../../ui/dialog.js'
6
+ import {
7
+ DropdownMenu, DropdownMenuContent, DropdownMenuItem, DropdownMenuTrigger,
8
+ } from '../../ui/dropdown-menu.js'
9
+ import { useNavigate } from '../../navigate.js'
10
+ import { useToast } from '../../Toaster.js'
11
+ import { withTooltip } from '../helpers.js'
12
+ import { actionButtonClass } from './buttons.js'
13
+ import { dispatchHandlerAction, dispatchMethodAction } from './helpers.js'
14
+ import { ActionModalDialog } from './ActionModalDialog.js'
15
+
16
+ /**
17
+ * Trigger button + dropdown menu for an `ActionGroup` meta. Reuses the
18
+ * action button styling helpers so a group's chrome (color/size/outlined/
19
+ * tooltip/iconButton) matches a regular Action. Each child Action
20
+ * dispatches via the same logic as `renderAction` — link/method/handler/
21
+ * confirm/modal — but routed through a `pending` state so the dropdown
22
+ * closes before any dialog opens (shadcn pattern: one popup at a time).
23
+ *
24
+ * `renderFormChild` + `renderElement` are injected through to
25
+ * `ActionModalDialog` for modal-form bodies. They originate in
26
+ * `SchemaRenderer.tsx` (form / dispatch layers).
27
+ */
28
+ export function ActionGroupTrigger({
29
+ el,
30
+ ids = [],
31
+ renderFormChild,
32
+ renderElement,
33
+ }: {
34
+ el: ElementMeta
35
+ ids?: string[]
36
+ renderFormChild: (child: ElementMeta, index: number, values: Record<string, unknown>, errors: Record<string, string[]>) => React.ReactNode
37
+ renderElement: (el: ElementMeta, index: number) => React.ReactNode
38
+ }) {
39
+ const [pending, setPending] = useState<ElementMeta | null>(null)
40
+ const navigate = useNavigate()
41
+ const { notify } = useToast()
42
+
43
+ const name = String(el['name'] ?? '')
44
+ const label = String(el['label'] ?? name)
45
+ const tooltip = el['tooltip'] as string | undefined
46
+ const iconOnly = Boolean(el['iconOnly'])
47
+ const isDisabled = Boolean(el['disabled'])
48
+ const childActions = (el.children ?? []).filter(c => c.type === 'action')
49
+
50
+ const className = actionButtonClass(el, {}) + (isDisabled ? ' opacity-50 cursor-not-allowed pointer-events-none' : '')
51
+ const ariaLabel = iconOnly ? label : undefined
52
+
53
+ // Direct-dispatch path mirrors renderAction's branches but skipping
54
+ // confirm/modal (those queue into `pending` so the dropdown can close).
55
+ const dispatch = (action: ElementMeta): void => {
56
+ const href = action['href'] as string | undefined
57
+ const method = action['method'] as 'post' | 'put' | 'patch' | 'delete' | undefined
58
+ const actionUrl = action['action'] as string | undefined
59
+ const dispatchUrl = action['dispatchUrl'] as string | undefined
60
+ if (href) {
61
+ navigate(href)
62
+ return
63
+ }
64
+ if (method && actionUrl) {
65
+ void dispatchMethodAction(actionUrl, method, navigate, notify)
66
+ return
67
+ }
68
+ if (dispatchUrl) {
69
+ void dispatchHandlerAction(dispatchUrl, ids, navigate, notify)
70
+ return
71
+ }
72
+ }
73
+
74
+ const onItemClick = (action: ElementMeta): void => {
75
+ if (action['modal'] || action['confirm']) {
76
+ setPending(action)
77
+ return
78
+ }
79
+ dispatch(action)
80
+ }
81
+
82
+ const pendingHandler = pending && pending['dispatchUrl']
83
+ const pendingConfirmOnly = pending && !pendingHandler && (pending['confirm'] as { title?: string; message: string } | undefined)
84
+ const pendingConfirm = pendingConfirmOnly || (pending?.['confirm'] as { title?: string; message: string } | undefined)
85
+
86
+ return (
87
+ <>
88
+ <DropdownMenu>
89
+ <DropdownMenuTrigger
90
+ render={(props) => withTooltip(
91
+ <button
92
+ {...props}
93
+ type="button"
94
+ className={className}
95
+ data-action-group-name={name}
96
+ aria-label={ariaLabel}
97
+ >
98
+ {iconOnly ? null : <span>{label}</span>}
99
+ </button>,
100
+ tooltip,
101
+ ) as React.ReactElement}
102
+ />
103
+ <DropdownMenuContent align="end">
104
+ {childActions.map((a, i) => {
105
+ const itemLabel = String(a['label'] ?? a['name'] ?? '')
106
+ const destructive = Boolean(a['destructive'])
107
+ const itemDisabled = Boolean(a['disabled'])
108
+ return (
109
+ <DropdownMenuItem
110
+ key={i}
111
+ destructive={destructive}
112
+ disabled={itemDisabled}
113
+ onClick={() => { if (!itemDisabled) onItemClick(a) }}
114
+ >
115
+ {itemLabel}
116
+ </DropdownMenuItem>
117
+ )
118
+ })}
119
+ </DropdownMenuContent>
120
+ </DropdownMenu>
121
+
122
+ {/* Modal / handler-style pending — fetch+JSON dispatch via ActionModalDialog. */}
123
+ {pendingHandler && pending && (
124
+ <ActionModalDialog
125
+ meta={pending}
126
+ ids={ids}
127
+ open={true}
128
+ onOpenChange={(o) => { if (!o) setPending(null) }}
129
+ renderFormChild={renderFormChild}
130
+ renderElement={renderElement}
131
+ />
132
+ )}
133
+
134
+ {/* Form-method confirm — fetch+JSON dispatch via dispatchMethodAction; SPA-navigates on success. */}
135
+ <Dialog
136
+ open={Boolean(pendingConfirmOnly)}
137
+ onOpenChange={(o) => { if (!o) setPending(null) }}
138
+ >
139
+ <DialogContent>
140
+ {pendingConfirmOnly && pendingConfirm && (
141
+ <>
142
+ <DialogHeader>
143
+ <DialogTitle>{pendingConfirm.title ?? 'Are you sure?'}</DialogTitle>
144
+ <DialogDescription>{pendingConfirm.message}</DialogDescription>
145
+ </DialogHeader>
146
+ <DialogFooter>
147
+ <button
148
+ type="button"
149
+ onClick={() => setPending(null)}
150
+ className="inline-flex items-center justify-center rounded-md border border-input bg-background px-3 h-9 text-sm font-medium hover:bg-accent hover:text-accent-foreground"
151
+ >
152
+ Cancel
153
+ </button>
154
+ <button
155
+ type="button"
156
+ autoFocus
157
+ onClick={() => {
158
+ const action = pending
159
+ setPending(null)
160
+ if (action) dispatch(action)
161
+ }}
162
+ className={
163
+ pending && pending['destructive']
164
+ ? 'inline-flex items-center justify-center rounded-md bg-destructive px-3 h-9 text-sm font-medium text-destructive-foreground hover:bg-destructive/90'
165
+ : 'inline-flex items-center justify-center rounded-md bg-primary px-3 h-9 text-sm font-medium text-primary-foreground hover:bg-primary/90'
166
+ }
167
+ >
168
+ {pending && pending['destructive'] ? 'Delete' : 'Confirm'}
169
+ </button>
170
+ </DialogFooter>
171
+ </>
172
+ )}
173
+ </DialogContent>
174
+ </Dialog>
175
+ </>
176
+ )
177
+ }