@plures/praxis 0.2.1 → 1.0.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 (486) hide show
  1. package/README.md +188 -61
  2. package/core/codegen/docs-generator.ts +808 -0
  3. package/core/codegen/index.ts +27 -0
  4. package/core/codegen/ts-generator.ts +15 -0
  5. package/core/db-adapter/index.ts +52 -0
  6. package/core/db-adapter/sync-engine.ts +450 -0
  7. package/core/logic-engine/engine.ts +12 -0
  8. package/core/logic-engine/index.ts +16 -0
  9. package/core/logic-engine/protocol.ts +16 -0
  10. package/core/logic-engine/psf-adapter.ts +269 -0
  11. package/core/logic-engine/rules.ts +16 -0
  12. package/core/schema-engine/compiler.ts +431 -0
  13. package/core/schema-engine/generator.ts +635 -0
  14. package/core/schema-engine/index.ts +18 -0
  15. package/core/schema-engine/psf.ts +664 -0
  16. package/core/schema-engine/types.ts +63 -0
  17. package/core/schema-engine/validator.ts +541 -0
  18. package/dist/core/codegen/docs-generator.d.ts +123 -0
  19. package/dist/core/codegen/docs-generator.d.ts.map +1 -0
  20. package/dist/core/codegen/docs-generator.js +674 -0
  21. package/dist/core/codegen/docs-generator.js.map +1 -0
  22. package/dist/core/codegen/index.d.ts +11 -0
  23. package/dist/core/codegen/index.d.ts.map +1 -0
  24. package/dist/core/codegen/index.js +13 -0
  25. package/dist/core/codegen/index.js.map +1 -0
  26. package/dist/core/codegen/ts-generator.d.ts +8 -0
  27. package/dist/core/codegen/ts-generator.d.ts.map +1 -0
  28. package/dist/core/codegen/ts-generator.js +8 -0
  29. package/dist/core/codegen/ts-generator.js.map +1 -0
  30. package/dist/core/db-adapter/index.d.ts +18 -0
  31. package/dist/core/db-adapter/index.d.ts.map +1 -0
  32. package/dist/core/db-adapter/index.js +23 -0
  33. package/dist/core/db-adapter/index.js.map +1 -0
  34. package/dist/core/db-adapter/sync-engine.d.ts +180 -0
  35. package/dist/core/db-adapter/sync-engine.d.ts.map +1 -0
  36. package/dist/core/db-adapter/sync-engine.js +342 -0
  37. package/dist/core/db-adapter/sync-engine.js.map +1 -0
  38. package/dist/core/logic-engine/engine.d.ts +8 -0
  39. package/dist/core/logic-engine/engine.d.ts.map +1 -0
  40. package/dist/core/logic-engine/engine.js +8 -0
  41. package/dist/core/logic-engine/engine.js.map +1 -0
  42. package/dist/core/logic-engine/index.d.ts +16 -0
  43. package/dist/core/logic-engine/index.d.ts.map +1 -0
  44. package/dist/core/logic-engine/index.js +16 -0
  45. package/dist/core/logic-engine/index.js.map +1 -0
  46. package/dist/core/logic-engine/protocol.d.ts +7 -0
  47. package/dist/core/logic-engine/protocol.d.ts.map +1 -0
  48. package/dist/core/logic-engine/protocol.js +7 -0
  49. package/dist/core/logic-engine/protocol.js.map +1 -0
  50. package/dist/core/logic-engine/psf-adapter.d.ts +88 -0
  51. package/dist/core/logic-engine/psf-adapter.d.ts.map +1 -0
  52. package/dist/core/logic-engine/psf-adapter.js +207 -0
  53. package/dist/core/logic-engine/psf-adapter.js.map +1 -0
  54. package/dist/core/logic-engine/rules.d.ts +7 -0
  55. package/dist/core/logic-engine/rules.d.ts.map +1 -0
  56. package/dist/core/logic-engine/rules.js +7 -0
  57. package/dist/core/logic-engine/rules.js.map +1 -0
  58. package/dist/core/schema-engine/compiler.d.ts +198 -0
  59. package/dist/core/schema-engine/compiler.d.ts.map +1 -0
  60. package/dist/core/schema-engine/compiler.js +262 -0
  61. package/dist/core/schema-engine/compiler.js.map +1 -0
  62. package/dist/core/schema-engine/generator.d.ts +115 -0
  63. package/dist/core/schema-engine/generator.d.ts.map +1 -0
  64. package/dist/core/schema-engine/generator.js +506 -0
  65. package/dist/core/schema-engine/generator.js.map +1 -0
  66. package/dist/core/schema-engine/index.d.ts +18 -0
  67. package/dist/core/schema-engine/index.d.ts.map +1 -0
  68. package/dist/core/schema-engine/index.js +18 -0
  69. package/dist/core/schema-engine/index.js.map +1 -0
  70. package/dist/core/schema-engine/psf.d.ts +612 -0
  71. package/dist/core/schema-engine/psf.d.ts.map +1 -0
  72. package/dist/core/schema-engine/psf.js +45 -0
  73. package/dist/core/schema-engine/psf.js.map +1 -0
  74. package/dist/core/schema-engine/types.d.ts +10 -0
  75. package/dist/core/schema-engine/types.d.ts.map +1 -0
  76. package/dist/core/schema-engine/types.js +7 -0
  77. package/dist/core/schema-engine/types.js.map +1 -0
  78. package/dist/core/schema-engine/validator.d.ts +140 -0
  79. package/dist/core/schema-engine/validator.d.ts.map +1 -0
  80. package/dist/core/schema-engine/validator.js +407 -0
  81. package/dist/core/schema-engine/validator.js.map +1 -0
  82. package/dist/src/adapters/cli.d.ts.map +1 -0
  83. package/dist/src/adapters/cli.js.map +1 -0
  84. package/dist/src/cli/commands/auth.d.ts.map +1 -0
  85. package/dist/src/cli/commands/auth.js.map +1 -0
  86. package/dist/src/cli/commands/build.d.ts +23 -0
  87. package/dist/src/cli/commands/build.d.ts.map +1 -0
  88. package/dist/src/cli/commands/build.js +162 -0
  89. package/dist/src/cli/commands/build.js.map +1 -0
  90. package/dist/src/cli/commands/canvas.d.ts +23 -0
  91. package/dist/src/cli/commands/canvas.d.ts.map +1 -0
  92. package/dist/src/cli/commands/canvas.js +215 -0
  93. package/dist/src/cli/commands/canvas.js.map +1 -0
  94. package/dist/src/cli/commands/cloud.d.ts.map +1 -0
  95. package/dist/src/cli/commands/cloud.js.map +1 -0
  96. package/dist/src/cli/commands/create.d.ts +21 -0
  97. package/dist/src/cli/commands/create.d.ts.map +1 -0
  98. package/dist/src/cli/commands/create.js +621 -0
  99. package/dist/src/cli/commands/create.js.map +1 -0
  100. package/dist/src/cli/commands/dev.d.ts +21 -0
  101. package/dist/src/cli/commands/dev.d.ts.map +1 -0
  102. package/dist/src/cli/commands/dev.js +71 -0
  103. package/dist/src/cli/commands/dev.js.map +1 -0
  104. package/dist/src/cli/commands/generate.d.ts.map +1 -0
  105. package/dist/src/cli/commands/generate.js.map +1 -0
  106. package/dist/src/cli/commands/orchestrate.d.ts +44 -0
  107. package/dist/src/cli/commands/orchestrate.d.ts.map +1 -0
  108. package/dist/src/cli/commands/orchestrate.js +150 -0
  109. package/dist/src/cli/commands/orchestrate.js.map +1 -0
  110. package/dist/{cli → src/cli}/index.d.ts.map +1 -1
  111. package/dist/{cli → src/cli}/index.js +53 -21
  112. package/dist/src/cli/index.js.map +1 -0
  113. package/dist/src/cloud/auth.d.ts.map +1 -0
  114. package/dist/src/cloud/auth.js.map +1 -0
  115. package/dist/src/cloud/billing.d.ts.map +1 -0
  116. package/dist/src/cloud/billing.js.map +1 -0
  117. package/dist/src/cloud/client.d.ts.map +1 -0
  118. package/dist/src/cloud/client.js.map +1 -0
  119. package/dist/src/cloud/index.d.ts.map +1 -0
  120. package/dist/src/cloud/index.js.map +1 -0
  121. package/dist/src/cloud/marketplace.d.ts.map +1 -0
  122. package/dist/src/cloud/marketplace.js.map +1 -0
  123. package/dist/src/cloud/provisioning.d.ts.map +1 -0
  124. package/dist/src/cloud/provisioning.js.map +1 -0
  125. package/dist/src/cloud/relay/endpoints.d.ts.map +1 -0
  126. package/dist/src/cloud/relay/endpoints.js.map +1 -0
  127. package/dist/src/cloud/relay/health/index.d.ts.map +1 -0
  128. package/dist/src/cloud/relay/health/index.js.map +1 -0
  129. package/dist/src/cloud/relay/stats/index.d.ts.map +1 -0
  130. package/dist/src/cloud/relay/stats/index.js.map +1 -0
  131. package/dist/src/cloud/relay/sync/index.d.ts.map +1 -0
  132. package/dist/src/cloud/relay/sync/index.js.map +1 -0
  133. package/dist/src/cloud/relay/usage/index.d.ts.map +1 -0
  134. package/dist/src/cloud/relay/usage/index.js.map +1 -0
  135. package/dist/src/cloud/sponsors.d.ts.map +1 -0
  136. package/dist/src/cloud/sponsors.js.map +1 -0
  137. package/dist/src/cloud/types.d.ts.map +1 -0
  138. package/dist/src/cloud/types.js.map +1 -0
  139. package/dist/src/components/index.d.ts.map +1 -0
  140. package/dist/src/components/index.js.map +1 -0
  141. package/dist/src/core/actors.d.ts.map +1 -0
  142. package/dist/src/core/actors.js.map +1 -0
  143. package/dist/src/core/component/generator.d.ts.map +1 -0
  144. package/dist/{core → src/core}/component/generator.js +45 -3
  145. package/dist/src/core/component/generator.js.map +1 -0
  146. package/dist/src/core/engine.d.ts.map +1 -0
  147. package/dist/src/core/engine.js.map +1 -0
  148. package/dist/src/core/introspection.d.ts.map +1 -0
  149. package/dist/src/core/introspection.js.map +1 -0
  150. package/dist/src/core/logic/generator.d.ts.map +1 -0
  151. package/dist/{core → src/core}/logic/generator.js +35 -4
  152. package/dist/src/core/logic/generator.js.map +1 -0
  153. package/dist/src/core/pluresdb/adapter.d.ts +72 -0
  154. package/dist/src/core/pluresdb/adapter.d.ts.map +1 -0
  155. package/dist/src/core/pluresdb/adapter.js +73 -0
  156. package/dist/src/core/pluresdb/adapter.js.map +1 -0
  157. package/dist/src/core/pluresdb/generator.d.ts.map +1 -0
  158. package/dist/{core → src/core}/pluresdb/generator.js +33 -4
  159. package/dist/src/core/pluresdb/generator.js.map +1 -0
  160. package/dist/src/core/pluresdb/index.d.ts +15 -0
  161. package/dist/src/core/pluresdb/index.d.ts.map +1 -0
  162. package/dist/src/core/pluresdb/index.js +11 -0
  163. package/dist/src/core/pluresdb/index.js.map +1 -0
  164. package/dist/src/core/pluresdb/schema-registry.d.ts +104 -0
  165. package/dist/src/core/pluresdb/schema-registry.d.ts.map +1 -0
  166. package/dist/src/core/pluresdb/schema-registry.js +130 -0
  167. package/dist/src/core/pluresdb/schema-registry.js.map +1 -0
  168. package/dist/src/core/pluresdb/store.d.ts +199 -0
  169. package/dist/src/core/pluresdb/store.d.ts.map +1 -0
  170. package/dist/src/core/pluresdb/store.js +344 -0
  171. package/dist/src/core/pluresdb/store.js.map +1 -0
  172. package/dist/src/core/protocol.d.ts.map +1 -0
  173. package/dist/src/core/protocol.js.map +1 -0
  174. package/dist/src/core/rules.d.ts.map +1 -0
  175. package/dist/src/core/rules.js.map +1 -0
  176. package/dist/src/core/schema/loader.d.ts.map +1 -0
  177. package/dist/src/core/schema/loader.js.map +1 -0
  178. package/dist/src/core/schema/normalize.d.ts.map +1 -0
  179. package/dist/src/core/schema/normalize.js.map +1 -0
  180. package/dist/src/core/schema/types.d.ts.map +1 -0
  181. package/dist/src/core/schema/types.js.map +1 -0
  182. package/dist/src/dsl/index.d.ts.map +1 -0
  183. package/dist/src/dsl/index.js.map +1 -0
  184. package/dist/src/dsl.d.ts.map +1 -0
  185. package/dist/src/dsl.js.map +1 -0
  186. package/dist/src/examples/advanced-todo/index.d.ts.map +1 -0
  187. package/dist/src/examples/advanced-todo/index.js.map +1 -0
  188. package/dist/src/examples/auth-basic/index.d.ts.map +1 -0
  189. package/dist/src/examples/auth-basic/index.js.map +1 -0
  190. package/dist/src/examples/cart/index.d.ts.map +1 -0
  191. package/dist/src/examples/cart/index.js.map +1 -0
  192. package/dist/src/examples/hero-ecommerce/index.d.ts.map +1 -0
  193. package/dist/src/examples/hero-ecommerce/index.js.map +1 -0
  194. package/dist/src/examples/svelte-counter/index.d.ts.map +1 -0
  195. package/dist/src/examples/svelte-counter/index.js.map +1 -0
  196. package/dist/src/flows.d.ts.map +1 -0
  197. package/dist/src/flows.js.map +1 -0
  198. package/dist/{index.d.ts → src/index.d.ts} +12 -2
  199. package/dist/src/index.d.ts.map +1 -0
  200. package/dist/{index.js → src/index.js} +6 -1
  201. package/dist/src/index.js.map +1 -0
  202. package/dist/src/integrations/code-canvas.d.ts +265 -0
  203. package/dist/src/integrations/code-canvas.d.ts.map +1 -0
  204. package/dist/src/integrations/code-canvas.js +451 -0
  205. package/dist/src/integrations/code-canvas.js.map +1 -0
  206. package/dist/src/integrations/pluresdb.d.ts +117 -0
  207. package/dist/src/integrations/pluresdb.d.ts.map +1 -0
  208. package/dist/src/integrations/pluresdb.js +117 -0
  209. package/dist/src/integrations/pluresdb.js.map +1 -0
  210. package/dist/src/integrations/state-docs.d.ts +191 -0
  211. package/dist/src/integrations/state-docs.d.ts.map +1 -0
  212. package/dist/src/integrations/state-docs.js +515 -0
  213. package/dist/src/integrations/state-docs.js.map +1 -0
  214. package/dist/src/integrations/svelte.d.ts.map +1 -0
  215. package/dist/src/integrations/svelte.js.map +1 -0
  216. package/dist/src/integrations/tauri.d.ts +360 -0
  217. package/dist/src/integrations/tauri.d.ts.map +1 -0
  218. package/dist/src/integrations/tauri.js +278 -0
  219. package/dist/src/integrations/tauri.js.map +1 -0
  220. package/dist/src/integrations/unum.d.ts +159 -0
  221. package/dist/src/integrations/unum.d.ts.map +1 -0
  222. package/dist/src/integrations/unum.js +240 -0
  223. package/dist/src/integrations/unum.js.map +1 -0
  224. package/dist/src/registry.d.ts.map +1 -0
  225. package/dist/src/registry.js.map +1 -0
  226. package/dist/{runtime → src/runtime}/terminal-adapter.d.ts +58 -7
  227. package/dist/src/runtime/terminal-adapter.d.ts.map +1 -0
  228. package/dist/src/runtime/terminal-adapter.js +237 -0
  229. package/dist/src/runtime/terminal-adapter.js.map +1 -0
  230. package/dist/src/step.d.ts.map +1 -0
  231. package/dist/src/step.js.map +1 -0
  232. package/dist/src/types.d.ts.map +1 -0
  233. package/dist/{cloud → src}/types.js.map +1 -1
  234. package/dist/tools/cli/commands/index.d.ts +7 -0
  235. package/dist/tools/cli/commands/index.d.ts.map +1 -0
  236. package/dist/tools/cli/commands/index.js +7 -0
  237. package/dist/tools/cli/commands/index.js.map +1 -0
  238. package/dist/tools/cli/index.d.ts +8 -0
  239. package/dist/tools/cli/index.d.ts.map +1 -0
  240. package/dist/tools/cli/index.js +9 -0
  241. package/dist/tools/cli/index.js.map +1 -0
  242. package/dist/tools/watcher/index.d.ts +105 -0
  243. package/dist/tools/watcher/index.d.ts.map +1 -0
  244. package/dist/tools/watcher/index.js +213 -0
  245. package/dist/tools/watcher/index.js.map +1 -0
  246. package/dist/ui/canvas/canvas-projection.d.ts +78 -0
  247. package/dist/ui/canvas/canvas-projection.d.ts.map +1 -0
  248. package/dist/ui/canvas/canvas-projection.js +416 -0
  249. package/dist/ui/canvas/canvas-projection.js.map +1 -0
  250. package/dist/ui/canvas/canvas-state.d.ts +200 -0
  251. package/dist/ui/canvas/canvas-state.d.ts.map +1 -0
  252. package/dist/ui/canvas/canvas-state.js +464 -0
  253. package/dist/ui/canvas/canvas-state.js.map +1 -0
  254. package/dist/ui/canvas/components/index.d.ts +95 -0
  255. package/dist/ui/canvas/components/index.d.ts.map +1 -0
  256. package/dist/ui/canvas/components/index.js +19 -0
  257. package/dist/ui/canvas/components/index.js.map +1 -0
  258. package/dist/ui/canvas/index.d.ts +32 -0
  259. package/dist/ui/canvas/index.d.ts.map +1 -0
  260. package/dist/ui/canvas/index.js +32 -0
  261. package/dist/ui/canvas/index.js.map +1 -0
  262. package/dist/ui/svelte-generator/index.d.ts +9 -0
  263. package/dist/ui/svelte-generator/index.d.ts.map +1 -0
  264. package/dist/ui/svelte-generator/index.js +11 -0
  265. package/dist/ui/svelte-generator/index.js.map +1 -0
  266. package/dist/ui/svelte-generator/psf-generator.d.ts +128 -0
  267. package/dist/ui/svelte-generator/psf-generator.d.ts.map +1 -0
  268. package/dist/ui/svelte-generator/psf-generator.js +506 -0
  269. package/dist/ui/svelte-generator/psf-generator.js.map +1 -0
  270. package/docs/README.md +155 -0
  271. package/docs/core/building-extensions.md +553 -0
  272. package/docs/core/cli-usage.md +498 -0
  273. package/docs/core/code-canvas-sync.md +468 -0
  274. package/docs/core/logic-engine.md +566 -0
  275. package/docs/core/pluresdb-integration.md +646 -0
  276. package/docs/core/schema-model.md +414 -0
  277. package/docs/core/ui-generation.md +580 -0
  278. package/docs/core/what-is-praxis.md +240 -0
  279. package/docs/tutorials/README.md +84 -0
  280. package/docs/tutorials/ecommerce-cart.md +631 -0
  281. package/docs/tutorials/first-app.md +529 -0
  282. package/docs/tutorials/form-builder.md +620 -0
  283. package/docs/tutorials/todo-pluresdb.md +589 -0
  284. package/package.json +1 -1
  285. package/src/__tests__/canvas-components.test.ts +450 -0
  286. package/src/__tests__/cli-create.test.ts +178 -0
  287. package/src/__tests__/code-canvas-integration.test.ts +277 -0
  288. package/src/__tests__/docs-generator.test.ts +181 -0
  289. package/src/__tests__/generators.test.ts +3 -2
  290. package/src/__tests__/pluresdb.test.ts +457 -0
  291. package/src/__tests__/psf-schema-engine.test.ts +450 -0
  292. package/src/__tests__/state-docs-integration.test.ts +297 -0
  293. package/src/__tests__/tauri-integration.test.ts +298 -0
  294. package/src/__tests__/terminal-node.test.ts +1 -1
  295. package/src/__tests__/unum-integration.test.ts +142 -0
  296. package/src/cli/commands/build.ts +203 -0
  297. package/src/cli/commands/canvas.ts +246 -0
  298. package/src/cli/commands/create.ts +666 -0
  299. package/src/cli/commands/dev.ts +95 -0
  300. package/src/cli/commands/orchestrate.ts +212 -0
  301. package/src/cli/index.ts +48 -21
  302. package/src/core/component/generator.ts +45 -3
  303. package/src/core/logic/generator.ts +39 -4
  304. package/src/core/pluresdb/adapter.ts +117 -0
  305. package/src/core/pluresdb/generator.ts +33 -4
  306. package/src/core/pluresdb/index.ts +37 -0
  307. package/src/core/pluresdb/schema-registry.ts +162 -0
  308. package/src/core/pluresdb/store.ts +443 -0
  309. package/src/index.ts +109 -0
  310. package/src/integrations/code-canvas.ts +717 -0
  311. package/src/integrations/pluresdb.ts +140 -29
  312. package/src/integrations/state-docs.ts +710 -0
  313. package/src/integrations/tauri.ts +638 -0
  314. package/src/integrations/unum.ts +395 -0
  315. package/src/runtime/terminal-adapter.ts +178 -23
  316. package/dist/adapters/cli.d.ts.map +0 -1
  317. package/dist/adapters/cli.js.map +0 -1
  318. package/dist/cli/commands/auth.d.ts.map +0 -1
  319. package/dist/cli/commands/auth.js.map +0 -1
  320. package/dist/cli/commands/cloud.d.ts.map +0 -1
  321. package/dist/cli/commands/cloud.js.map +0 -1
  322. package/dist/cli/commands/generate.d.ts.map +0 -1
  323. package/dist/cli/commands/generate.js.map +0 -1
  324. package/dist/cli/index.js.map +0 -1
  325. package/dist/cloud/auth.d.ts.map +0 -1
  326. package/dist/cloud/auth.js.map +0 -1
  327. package/dist/cloud/billing.d.ts.map +0 -1
  328. package/dist/cloud/billing.js.map +0 -1
  329. package/dist/cloud/client.d.ts.map +0 -1
  330. package/dist/cloud/client.js.map +0 -1
  331. package/dist/cloud/index.d.ts.map +0 -1
  332. package/dist/cloud/index.js.map +0 -1
  333. package/dist/cloud/marketplace.d.ts.map +0 -1
  334. package/dist/cloud/marketplace.js.map +0 -1
  335. package/dist/cloud/provisioning.d.ts.map +0 -1
  336. package/dist/cloud/provisioning.js.map +0 -1
  337. package/dist/cloud/relay/endpoints.d.ts.map +0 -1
  338. package/dist/cloud/relay/endpoints.js.map +0 -1
  339. package/dist/cloud/relay/health/index.d.ts.map +0 -1
  340. package/dist/cloud/relay/health/index.js.map +0 -1
  341. package/dist/cloud/relay/stats/index.d.ts.map +0 -1
  342. package/dist/cloud/relay/stats/index.js.map +0 -1
  343. package/dist/cloud/relay/sync/index.d.ts.map +0 -1
  344. package/dist/cloud/relay/sync/index.js.map +0 -1
  345. package/dist/cloud/relay/usage/index.d.ts.map +0 -1
  346. package/dist/cloud/relay/usage/index.js.map +0 -1
  347. package/dist/cloud/sponsors.d.ts.map +0 -1
  348. package/dist/cloud/sponsors.js.map +0 -1
  349. package/dist/cloud/types.d.ts.map +0 -1
  350. package/dist/components/index.d.ts.map +0 -1
  351. package/dist/components/index.js.map +0 -1
  352. package/dist/core/actors.d.ts.map +0 -1
  353. package/dist/core/actors.js.map +0 -1
  354. package/dist/core/component/generator.d.ts.map +0 -1
  355. package/dist/core/component/generator.js.map +0 -1
  356. package/dist/core/engine.d.ts.map +0 -1
  357. package/dist/core/engine.js.map +0 -1
  358. package/dist/core/introspection.d.ts.map +0 -1
  359. package/dist/core/introspection.js.map +0 -1
  360. package/dist/core/logic/generator.d.ts.map +0 -1
  361. package/dist/core/logic/generator.js.map +0 -1
  362. package/dist/core/pluresdb/generator.d.ts.map +0 -1
  363. package/dist/core/pluresdb/generator.js.map +0 -1
  364. package/dist/core/protocol.d.ts.map +0 -1
  365. package/dist/core/protocol.js.map +0 -1
  366. package/dist/core/rules.d.ts.map +0 -1
  367. package/dist/core/rules.js.map +0 -1
  368. package/dist/core/schema/loader.d.ts.map +0 -1
  369. package/dist/core/schema/loader.js.map +0 -1
  370. package/dist/core/schema/normalize.d.ts.map +0 -1
  371. package/dist/core/schema/normalize.js.map +0 -1
  372. package/dist/core/schema/types.d.ts.map +0 -1
  373. package/dist/core/schema/types.js.map +0 -1
  374. package/dist/dsl/index.d.ts.map +0 -1
  375. package/dist/dsl/index.js.map +0 -1
  376. package/dist/dsl.d.ts.map +0 -1
  377. package/dist/dsl.js.map +0 -1
  378. package/dist/examples/advanced-todo/index.d.ts.map +0 -1
  379. package/dist/examples/advanced-todo/index.js.map +0 -1
  380. package/dist/examples/auth-basic/index.d.ts.map +0 -1
  381. package/dist/examples/auth-basic/index.js.map +0 -1
  382. package/dist/examples/cart/index.d.ts.map +0 -1
  383. package/dist/examples/cart/index.js.map +0 -1
  384. package/dist/examples/hero-ecommerce/index.d.ts.map +0 -1
  385. package/dist/examples/hero-ecommerce/index.js.map +0 -1
  386. package/dist/examples/svelte-counter/index.d.ts.map +0 -1
  387. package/dist/examples/svelte-counter/index.js.map +0 -1
  388. package/dist/flows.d.ts.map +0 -1
  389. package/dist/flows.js.map +0 -1
  390. package/dist/index.d.ts.map +0 -1
  391. package/dist/index.js.map +0 -1
  392. package/dist/integrations/pluresdb.d.ts +0 -56
  393. package/dist/integrations/pluresdb.d.ts.map +0 -1
  394. package/dist/integrations/pluresdb.js +0 -46
  395. package/dist/integrations/pluresdb.js.map +0 -1
  396. package/dist/integrations/svelte.d.ts.map +0 -1
  397. package/dist/integrations/svelte.js.map +0 -1
  398. package/dist/registry.d.ts.map +0 -1
  399. package/dist/registry.js.map +0 -1
  400. package/dist/runtime/terminal-adapter.d.ts.map +0 -1
  401. package/dist/runtime/terminal-adapter.js +0 -113
  402. package/dist/runtime/terminal-adapter.js.map +0 -1
  403. package/dist/step.d.ts.map +0 -1
  404. package/dist/step.js.map +0 -1
  405. package/dist/types.d.ts.map +0 -1
  406. package/dist/types.js.map +0 -1
  407. /package/dist/{adapters → src/adapters}/cli.d.ts +0 -0
  408. /package/dist/{adapters → src/adapters}/cli.js +0 -0
  409. /package/dist/{cli → src/cli}/commands/auth.d.ts +0 -0
  410. /package/dist/{cli → src/cli}/commands/auth.js +0 -0
  411. /package/dist/{cli → src/cli}/commands/cloud.d.ts +0 -0
  412. /package/dist/{cli → src/cli}/commands/cloud.js +0 -0
  413. /package/dist/{cli → src/cli}/commands/generate.d.ts +0 -0
  414. /package/dist/{cli → src/cli}/commands/generate.js +0 -0
  415. /package/dist/{cli → src/cli}/index.d.ts +0 -0
  416. /package/dist/{cloud → src/cloud}/auth.d.ts +0 -0
  417. /package/dist/{cloud → src/cloud}/auth.js +0 -0
  418. /package/dist/{cloud → src/cloud}/billing.d.ts +0 -0
  419. /package/dist/{cloud → src/cloud}/billing.js +0 -0
  420. /package/dist/{cloud → src/cloud}/client.d.ts +0 -0
  421. /package/dist/{cloud → src/cloud}/client.js +0 -0
  422. /package/dist/{cloud → src/cloud}/index.d.ts +0 -0
  423. /package/dist/{cloud → src/cloud}/index.js +0 -0
  424. /package/dist/{cloud → src/cloud}/marketplace.d.ts +0 -0
  425. /package/dist/{cloud → src/cloud}/marketplace.js +0 -0
  426. /package/dist/{cloud → src/cloud}/provisioning.d.ts +0 -0
  427. /package/dist/{cloud → src/cloud}/provisioning.js +0 -0
  428. /package/dist/{cloud → src/cloud}/relay/endpoints.d.ts +0 -0
  429. /package/dist/{cloud → src/cloud}/relay/endpoints.js +0 -0
  430. /package/dist/{cloud → src/cloud}/relay/health/index.d.ts +0 -0
  431. /package/dist/{cloud → src/cloud}/relay/health/index.js +0 -0
  432. /package/dist/{cloud → src/cloud}/relay/stats/index.d.ts +0 -0
  433. /package/dist/{cloud → src/cloud}/relay/stats/index.js +0 -0
  434. /package/dist/{cloud → src/cloud}/relay/sync/index.d.ts +0 -0
  435. /package/dist/{cloud → src/cloud}/relay/sync/index.js +0 -0
  436. /package/dist/{cloud → src/cloud}/relay/usage/index.d.ts +0 -0
  437. /package/dist/{cloud → src/cloud}/relay/usage/index.js +0 -0
  438. /package/dist/{cloud → src/cloud}/sponsors.d.ts +0 -0
  439. /package/dist/{cloud → src/cloud}/sponsors.js +0 -0
  440. /package/dist/{cloud → src/cloud}/types.d.ts +0 -0
  441. /package/dist/{cloud → src/cloud}/types.js +0 -0
  442. /package/dist/{components → src/components}/index.d.ts +0 -0
  443. /package/dist/{components → src/components}/index.js +0 -0
  444. /package/dist/{core → src/core}/actors.d.ts +0 -0
  445. /package/dist/{core → src/core}/actors.js +0 -0
  446. /package/dist/{core → src/core}/component/generator.d.ts +0 -0
  447. /package/dist/{core → src/core}/engine.d.ts +0 -0
  448. /package/dist/{core → src/core}/engine.js +0 -0
  449. /package/dist/{core → src/core}/introspection.d.ts +0 -0
  450. /package/dist/{core → src/core}/introspection.js +0 -0
  451. /package/dist/{core → src/core}/logic/generator.d.ts +0 -0
  452. /package/dist/{core → src/core}/pluresdb/generator.d.ts +0 -0
  453. /package/dist/{core → src/core}/protocol.d.ts +0 -0
  454. /package/dist/{core → src/core}/protocol.js +0 -0
  455. /package/dist/{core → src/core}/rules.d.ts +0 -0
  456. /package/dist/{core → src/core}/rules.js +0 -0
  457. /package/dist/{core → src/core}/schema/loader.d.ts +0 -0
  458. /package/dist/{core → src/core}/schema/loader.js +0 -0
  459. /package/dist/{core → src/core}/schema/normalize.d.ts +0 -0
  460. /package/dist/{core → src/core}/schema/normalize.js +0 -0
  461. /package/dist/{core → src/core}/schema/types.d.ts +0 -0
  462. /package/dist/{core → src/core}/schema/types.js +0 -0
  463. /package/dist/{dsl → src/dsl}/index.d.ts +0 -0
  464. /package/dist/{dsl → src/dsl}/index.js +0 -0
  465. /package/dist/{dsl.d.ts → src/dsl.d.ts} +0 -0
  466. /package/dist/{dsl.js → src/dsl.js} +0 -0
  467. /package/dist/{examples → src/examples}/advanced-todo/index.d.ts +0 -0
  468. /package/dist/{examples → src/examples}/advanced-todo/index.js +0 -0
  469. /package/dist/{examples → src/examples}/auth-basic/index.d.ts +0 -0
  470. /package/dist/{examples → src/examples}/auth-basic/index.js +0 -0
  471. /package/dist/{examples → src/examples}/cart/index.d.ts +0 -0
  472. /package/dist/{examples → src/examples}/cart/index.js +0 -0
  473. /package/dist/{examples → src/examples}/hero-ecommerce/index.d.ts +0 -0
  474. /package/dist/{examples → src/examples}/hero-ecommerce/index.js +0 -0
  475. /package/dist/{examples → src/examples}/svelte-counter/index.d.ts +0 -0
  476. /package/dist/{examples → src/examples}/svelte-counter/index.js +0 -0
  477. /package/dist/{flows.d.ts → src/flows.d.ts} +0 -0
  478. /package/dist/{flows.js → src/flows.js} +0 -0
  479. /package/dist/{integrations → src/integrations}/svelte.d.ts +0 -0
  480. /package/dist/{integrations → src/integrations}/svelte.js +0 -0
  481. /package/dist/{registry.d.ts → src/registry.d.ts} +0 -0
  482. /package/dist/{registry.js → src/registry.js} +0 -0
  483. /package/dist/{step.d.ts → src/step.d.ts} +0 -0
  484. /package/dist/{step.js → src/step.js} +0 -0
  485. /package/dist/{types.d.ts → src/types.d.ts} +0 -0
  486. /package/dist/{types.js → src/types.js} +0 -0
@@ -0,0 +1,620 @@
1
+ # Form Builder Tutorial
2
+
3
+ This tutorial walks you through building a dynamic form builder application. You'll learn advanced schema composition, component generation, and complex logic flows.
4
+
5
+ **Time:** 40-50 minutes
6
+ **Level:** Intermediate
7
+ **Prerequisites:** Completed [Todo with PluresDB](./todo-pluresdb.md)
8
+
9
+ ## What You'll Build
10
+
11
+ A form builder that allows users to:
12
+ - Create new forms with a name and description
13
+ - Add various field types (text, number, select, checkbox, etc.)
14
+ - Configure field properties (label, required, validation)
15
+ - Reorder fields via drag-and-drop
16
+ - Preview the form as end-users would see it
17
+ - Collect form submissions
18
+
19
+ ## Step 1: Project Setup
20
+
21
+ ```bash
22
+ mkdir praxis-form-builder
23
+ cd praxis-form-builder
24
+ npm init -y
25
+ npm install @plures/praxis
26
+ npm install -D typescript vitest
27
+ ```
28
+
29
+ ## Step 2: Understand the Schema
30
+
31
+ The form builder schema is available at `examples/form-builder/schema.psf.json`. Let's break down its key parts:
32
+
33
+ ### Models
34
+
35
+ ```json
36
+ {
37
+ "models": [
38
+ {
39
+ "name": "Form",
40
+ "fields": [
41
+ { "name": "id", "type": "uuid" },
42
+ { "name": "name", "type": "string" },
43
+ { "name": "fields", "type": { "array": { "reference": "FormField" } } }
44
+ ]
45
+ },
46
+ {
47
+ "name": "FormField",
48
+ "fields": [
49
+ { "name": "id", "type": "string" },
50
+ { "name": "type", "type": { "enum": ["text", "number", "email", "select", "checkbox"] } },
51
+ { "name": "label", "type": "string" },
52
+ { "name": "required", "type": "boolean" },
53
+ { "name": "order", "type": "number" }
54
+ ]
55
+ }
56
+ ]
57
+ }
58
+ ```
59
+
60
+ ### Events
61
+
62
+ ```json
63
+ {
64
+ "events": [
65
+ { "tag": "CreateForm", "payload": { "name": "string" } },
66
+ { "tag": "AddField", "payload": { "formId": "string", "fieldType": "string", "label": "string" } },
67
+ { "tag": "UpdateField", "payload": { "fieldId": "string", "label": "string", "required": "boolean" } },
68
+ { "tag": "ReorderFields", "payload": { "formId": "string", "fieldOrder": "string[]" } },
69
+ { "tag": "SubmitForm", "payload": { "formId": "string", "data": "object" } }
70
+ ]
71
+ }
72
+ ```
73
+
74
+ ## Step 3: Create the Engine
75
+
76
+ Create `src/engine.ts`:
77
+
78
+ ```typescript
79
+ import {
80
+ createPraxisEngine,
81
+ PraxisRegistry,
82
+ defineFact,
83
+ defineEvent,
84
+ defineRule,
85
+ defineConstraint,
86
+ } from '@plures/praxis';
87
+
88
+ // Types
89
+ interface Form {
90
+ id: string;
91
+ name: string;
92
+ description?: string;
93
+ fields: FormField[];
94
+ createdAt: Date;
95
+ updatedAt: Date;
96
+ }
97
+
98
+ interface FormField {
99
+ id: string;
100
+ type: 'text' | 'number' | 'email' | 'select' | 'checkbox' | 'radio' | 'textarea' | 'date';
101
+ label: string;
102
+ placeholder?: string;
103
+ required: boolean;
104
+ options?: string[];
105
+ validation?: Record<string, any>;
106
+ order: number;
107
+ }
108
+
109
+ interface FormSubmission {
110
+ id: string;
111
+ formId: string;
112
+ data: Record<string, any>;
113
+ submittedAt: Date;
114
+ }
115
+
116
+ interface FormBuilderContext {
117
+ forms: Form[];
118
+ submissions: FormSubmission[];
119
+ activeFormId: string | null;
120
+ selectedFieldId: string | null;
121
+ }
122
+
123
+ // Facts
124
+ export const FormCreated = defineFact<'FormCreated', { formId: string; name: string }>('FormCreated');
125
+ export const FieldAdded = defineFact<'FieldAdded', { formId: string; fieldId: string; fieldType: string }>('FieldAdded');
126
+ export const FieldRemoved = defineFact<'FieldRemoved', { formId: string; fieldId: string }>('FieldRemoved');
127
+ export const FieldUpdated = defineFact<'FieldUpdated', { fieldId: string }>('FieldUpdated');
128
+ export const FormSubmitted = defineFact<'FormSubmitted', { formId: string; submissionId: string }>('FormSubmitted');
129
+ export const ValidationFailed = defineFact<'ValidationFailed', { formId: string; errors: Array<{ fieldId: string; message: string }> }>('ValidationFailed');
130
+
131
+ // Events
132
+ export const CREATE_FORM = defineEvent<'CREATE_FORM', { name: string; description?: string }>('CREATE_FORM');
133
+ export const ADD_FIELD = defineEvent<'ADD_FIELD', { formId: string; fieldType: string; label: string; required?: boolean }>('ADD_FIELD');
134
+ export const REMOVE_FIELD = defineEvent<'REMOVE_FIELD', { formId: string; fieldId: string }>('REMOVE_FIELD');
135
+ export const UPDATE_FIELD = defineEvent<'UPDATE_FIELD', { fieldId: string; updates: Partial<FormField> }>('UPDATE_FIELD');
136
+ export const REORDER_FIELDS = defineEvent<'REORDER_FIELDS', { formId: string; fieldOrder: string[] }>('REORDER_FIELDS');
137
+ export const SUBMIT_FORM = defineEvent<'SUBMIT_FORM', { formId: string; data: Record<string, any> }>('SUBMIT_FORM');
138
+ export const SELECT_FORM = defineEvent<'SELECT_FORM', { formId: string }>('SELECT_FORM');
139
+ export const SELECT_FIELD = defineEvent<'SELECT_FIELD', { fieldId: string | null }>('SELECT_FIELD');
140
+
141
+ // Rules
142
+ const createFormRule = defineRule<FormBuilderContext>({
143
+ id: 'form.create',
144
+ description: 'Create a new form',
145
+ impl: (state, events) => {
146
+ const event = events.find(CREATE_FORM.is);
147
+ if (!event) return [];
148
+
149
+ const now = new Date();
150
+ const formId = `form_${Date.now().toString(36)}`;
151
+
152
+ const form: Form = {
153
+ id: formId,
154
+ name: event.payload.name,
155
+ description: event.payload.description,
156
+ fields: [],
157
+ createdAt: now,
158
+ updatedAt: now,
159
+ };
160
+
161
+ state.context.forms.push(form);
162
+ state.context.activeFormId = formId;
163
+
164
+ return [FormCreated.create({ formId, name: event.payload.name })];
165
+ },
166
+ });
167
+
168
+ const addFieldRule = defineRule<FormBuilderContext>({
169
+ id: 'form.addField',
170
+ description: 'Add a field to a form',
171
+ impl: (state, events) => {
172
+ const event = events.find(ADD_FIELD.is);
173
+ if (!event) return [];
174
+
175
+ const form = state.context.forms.find(f => f.id === event.payload.formId);
176
+ if (!form) return [];
177
+
178
+ const fieldId = `field_${Date.now().toString(36)}`;
179
+ const field: FormField = {
180
+ id: fieldId,
181
+ type: event.payload.fieldType as FormField['type'],
182
+ label: event.payload.label,
183
+ required: event.payload.required ?? false,
184
+ order: form.fields.length,
185
+ };
186
+
187
+ form.fields.push(field);
188
+ form.updatedAt = new Date();
189
+ state.context.selectedFieldId = fieldId;
190
+
191
+ return [FieldAdded.create({
192
+ formId: event.payload.formId,
193
+ fieldId,
194
+ fieldType: event.payload.fieldType
195
+ })];
196
+ },
197
+ });
198
+
199
+ const removeFieldRule = defineRule<FormBuilderContext>({
200
+ id: 'form.removeField',
201
+ description: 'Remove a field from a form',
202
+ impl: (state, events) => {
203
+ const event = events.find(REMOVE_FIELD.is);
204
+ if (!event) return [];
205
+
206
+ const form = state.context.forms.find(f => f.id === event.payload.formId);
207
+ if (!form) return [];
208
+
209
+ form.fields = form.fields.filter(f => f.id !== event.payload.fieldId);
210
+ form.updatedAt = new Date();
211
+
212
+ // Reorder remaining fields
213
+ form.fields.forEach((field, index) => {
214
+ field.order = index;
215
+ });
216
+
217
+ if (state.context.selectedFieldId === event.payload.fieldId) {
218
+ state.context.selectedFieldId = null;
219
+ }
220
+
221
+ return [FieldRemoved.create({ formId: event.payload.formId, fieldId: event.payload.fieldId })];
222
+ },
223
+ });
224
+
225
+ const updateFieldRule = defineRule<FormBuilderContext>({
226
+ id: 'form.updateField',
227
+ description: 'Update field configuration',
228
+ impl: (state, events) => {
229
+ const event = events.find(UPDATE_FIELD.is);
230
+ if (!event) return [];
231
+
232
+ for (const form of state.context.forms) {
233
+ const field = form.fields.find(f => f.id === event.payload.fieldId);
234
+ if (field) {
235
+ Object.assign(field, event.payload.updates);
236
+ form.updatedAt = new Date();
237
+ return [FieldUpdated.create({ fieldId: event.payload.fieldId })];
238
+ }
239
+ }
240
+
241
+ return [];
242
+ },
243
+ });
244
+
245
+ const reorderFieldsRule = defineRule<FormBuilderContext>({
246
+ id: 'form.reorderFields',
247
+ description: 'Reorder fields in a form',
248
+ impl: (state, events) => {
249
+ const event = events.find(REORDER_FIELDS.is);
250
+ if (!event) return [];
251
+
252
+ const form = state.context.forms.find(f => f.id === event.payload.formId);
253
+ if (!form) return [];
254
+
255
+ // Create a map of field id to field
256
+ const fieldMap = new Map(form.fields.map(f => [f.id, f]));
257
+
258
+ // Reorder based on new order
259
+ form.fields = event.payload.fieldOrder
260
+ .map((id, index) => {
261
+ const field = fieldMap.get(id);
262
+ if (field) {
263
+ field.order = index;
264
+ return field;
265
+ }
266
+ return null;
267
+ })
268
+ .filter((f): f is FormField => f !== null);
269
+
270
+ form.updatedAt = new Date();
271
+
272
+ return [];
273
+ },
274
+ });
275
+
276
+ const submitFormRule = defineRule<FormBuilderContext>({
277
+ id: 'form.submit',
278
+ description: 'Handle form submission with validation',
279
+ impl: (state, events) => {
280
+ const event = events.find(SUBMIT_FORM.is);
281
+ if (!event) return [];
282
+
283
+ const form = state.context.forms.find(f => f.id === event.payload.formId);
284
+ if (!form) return [];
285
+
286
+ // Validate required fields
287
+ const errors: Array<{ fieldId: string; message: string }> = [];
288
+
289
+ for (const field of form.fields) {
290
+ if (field.required) {
291
+ const value = event.payload.data[field.id];
292
+ if (value === undefined || value === null || value === '') {
293
+ errors.push({ fieldId: field.id, message: `${field.label} is required` });
294
+ }
295
+ }
296
+
297
+ // Type-specific validation
298
+ if (field.type === 'email' && event.payload.data[field.id]) {
299
+ const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
300
+ if (!emailRegex.test(event.payload.data[field.id])) {
301
+ errors.push({ fieldId: field.id, message: 'Invalid email address' });
302
+ }
303
+ }
304
+
305
+ if (field.type === 'number' && event.payload.data[field.id]) {
306
+ if (isNaN(Number(event.payload.data[field.id]))) {
307
+ errors.push({ fieldId: field.id, message: 'Must be a number' });
308
+ }
309
+ }
310
+ }
311
+
312
+ if (errors.length > 0) {
313
+ return [ValidationFailed.create({ formId: event.payload.formId, errors })];
314
+ }
315
+
316
+ // Create submission
317
+ const submissionId = `sub_${Date.now().toString(36)}`;
318
+ const submission: FormSubmission = {
319
+ id: submissionId,
320
+ formId: event.payload.formId,
321
+ data: event.payload.data,
322
+ submittedAt: new Date(),
323
+ };
324
+
325
+ state.context.submissions.push(submission);
326
+
327
+ return [FormSubmitted.create({ formId: event.payload.formId, submissionId })];
328
+ },
329
+ });
330
+
331
+ const selectFormRule = defineRule<FormBuilderContext>({
332
+ id: 'form.select',
333
+ description: 'Select a form for editing',
334
+ impl: (state, events) => {
335
+ const event = events.find(SELECT_FORM.is);
336
+ if (!event) return [];
337
+
338
+ state.context.activeFormId = event.payload.formId;
339
+ state.context.selectedFieldId = null;
340
+
341
+ return [];
342
+ },
343
+ });
344
+
345
+ const selectFieldRule = defineRule<FormBuilderContext>({
346
+ id: 'field.select',
347
+ description: 'Select a field for editing',
348
+ impl: (state, events) => {
349
+ const event = events.find(SELECT_FIELD.is);
350
+ if (!event) return [];
351
+
352
+ state.context.selectedFieldId = event.payload.fieldId;
353
+
354
+ return [];
355
+ },
356
+ });
357
+
358
+ // Constraints
359
+ const uniqueFieldIdsConstraint = defineConstraint<FormBuilderContext>({
360
+ id: 'form.uniqueFieldIds',
361
+ description: 'All field IDs must be unique within a form',
362
+ check: (state) => {
363
+ for (const form of state.context.forms) {
364
+ const ids = form.fields.map(f => f.id);
365
+ if (ids.length !== new Set(ids).size) {
366
+ return false;
367
+ }
368
+ }
369
+ return true;
370
+ },
371
+ errorMessage: 'Duplicate field IDs detected',
372
+ severity: 'error',
373
+ });
374
+
375
+ const maxFieldsConstraint = defineConstraint<FormBuilderContext>({
376
+ id: 'form.maxFields',
377
+ description: 'Form cannot have more than 50 fields',
378
+ check: (state) => {
379
+ return state.context.forms.every(form => form.fields.length <= 50);
380
+ },
381
+ errorMessage: 'Form cannot have more than 50 fields',
382
+ severity: 'error',
383
+ });
384
+
385
+ // Registry
386
+ const registry = new PraxisRegistry<FormBuilderContext>();
387
+ registry.registerRule(createFormRule);
388
+ registry.registerRule(addFieldRule);
389
+ registry.registerRule(removeFieldRule);
390
+ registry.registerRule(updateFieldRule);
391
+ registry.registerRule(reorderFieldsRule);
392
+ registry.registerRule(submitFormRule);
393
+ registry.registerRule(selectFormRule);
394
+ registry.registerRule(selectFieldRule);
395
+ registry.registerConstraint(uniqueFieldIdsConstraint);
396
+ registry.registerConstraint(maxFieldsConstraint);
397
+
398
+ // Engine factory
399
+ export function createFormBuilderEngine() {
400
+ return createPraxisEngine({
401
+ initialContext: {
402
+ forms: [],
403
+ submissions: [],
404
+ activeFormId: null,
405
+ selectedFieldId: null,
406
+ },
407
+ registry,
408
+ enableHistory: true,
409
+ maxHistorySize: 50,
410
+ });
411
+ }
412
+
413
+ // Helper functions
414
+ export function getActiveForm(context: FormBuilderContext): Form | null {
415
+ if (!context.activeFormId) return null;
416
+ return context.forms.find(f => f.id === context.activeFormId) || null;
417
+ }
418
+
419
+ export function getSelectedField(context: FormBuilderContext): FormField | null {
420
+ if (!context.selectedFieldId) return null;
421
+ for (const form of context.forms) {
422
+ const field = form.fields.find(f => f.id === context.selectedFieldId);
423
+ if (field) return field;
424
+ }
425
+ return null;
426
+ }
427
+ ```
428
+
429
+ ## Step 4: Create the Main Application
430
+
431
+ Create `src/main.ts`:
432
+
433
+ ```typescript
434
+ import {
435
+ createFormBuilderEngine,
436
+ CREATE_FORM,
437
+ ADD_FIELD,
438
+ UPDATE_FIELD,
439
+ SUBMIT_FORM,
440
+ getActiveForm,
441
+ } from './engine';
442
+
443
+ async function main() {
444
+ console.log('šŸ› ļø Form Builder Demo\n');
445
+
446
+ const engine = createFormBuilderEngine();
447
+
448
+ // Create a contact form
449
+ console.log('Creating contact form...');
450
+ engine.dispatch([CREATE_FORM.create({
451
+ name: 'Contact Form',
452
+ description: 'Get in touch with us'
453
+ })]);
454
+
455
+ const form = getActiveForm(engine.getContext());
456
+ if (!form) throw new Error('Form not created');
457
+
458
+ console.log(`āœ… Created form: ${form.name}\n`);
459
+
460
+ // Add fields
461
+ console.log('Adding fields...');
462
+
463
+ engine.dispatch([ADD_FIELD.create({
464
+ formId: form.id,
465
+ fieldType: 'text',
466
+ label: 'Full Name',
467
+ required: true
468
+ })]);
469
+
470
+ engine.dispatch([ADD_FIELD.create({
471
+ formId: form.id,
472
+ fieldType: 'email',
473
+ label: 'Email Address',
474
+ required: true
475
+ })]);
476
+
477
+ engine.dispatch([ADD_FIELD.create({
478
+ formId: form.id,
479
+ fieldType: 'select',
480
+ label: 'Subject',
481
+ required: true
482
+ })]);
483
+
484
+ engine.dispatch([ADD_FIELD.create({
485
+ formId: form.id,
486
+ fieldType: 'textarea',
487
+ label: 'Message',
488
+ required: true
489
+ })]);
490
+
491
+ // Update the select field with options
492
+ const updatedForm = getActiveForm(engine.getContext())!;
493
+ const subjectField = updatedForm.fields.find(f => f.label === 'Subject');
494
+ if (subjectField) {
495
+ engine.dispatch([UPDATE_FIELD.create({
496
+ fieldId: subjectField.id,
497
+ updates: {
498
+ options: ['General Inquiry', 'Support', 'Feedback', 'Other'],
499
+ placeholder: 'Select a subject',
500
+ },
501
+ })]);
502
+ }
503
+
504
+ // Display form structure
505
+ const finalForm = getActiveForm(engine.getContext())!;
506
+ console.log('\nšŸ“‹ Form Structure:');
507
+ console.log('─'.repeat(50));
508
+ console.log(`Name: ${finalForm.name}`);
509
+ console.log(`Description: ${finalForm.description || 'N/A'}`);
510
+ console.log(`Fields: ${finalForm.fields.length}`);
511
+ console.log('');
512
+
513
+ finalForm.fields.forEach((field, i) => {
514
+ const required = field.required ? '*' : '';
515
+ console.log(` ${i + 1}. [${field.type}] ${field.label}${required}`);
516
+ if (field.options) {
517
+ console.log(` Options: ${field.options.join(', ')}`);
518
+ }
519
+ });
520
+ console.log('─'.repeat(50));
521
+
522
+ // Test form submission - with validation error
523
+ console.log('\nšŸ“ Testing form submission (incomplete data)...');
524
+ const result1 = engine.step([SUBMIT_FORM.create({
525
+ formId: form.id,
526
+ data: {
527
+ [finalForm.fields[0].id]: 'John Doe',
528
+ // Missing email, subject, message
529
+ },
530
+ })]);
531
+
532
+ const validationFailed = result1.state.facts.find(f => f.tag === 'ValidationFailed');
533
+ if (validationFailed) {
534
+ console.log('āŒ Validation failed:');
535
+ (validationFailed.payload as any).errors.forEach((err: any) => {
536
+ console.log(` - ${err.message}`);
537
+ });
538
+ }
539
+
540
+ // Test form submission - successful
541
+ console.log('\nšŸ“ Testing form submission (complete data)...');
542
+ const result2 = engine.step([SUBMIT_FORM.create({
543
+ formId: form.id,
544
+ data: {
545
+ [finalForm.fields[0].id]: 'John Doe',
546
+ [finalForm.fields[1].id]: 'john@example.com',
547
+ [finalForm.fields[2].id]: 'Support',
548
+ [finalForm.fields[3].id]: 'I need help with my account.',
549
+ },
550
+ })]);
551
+
552
+ const submitted = result2.state.facts.find(f => f.tag === 'FormSubmitted');
553
+ if (submitted) {
554
+ console.log('āœ… Form submitted successfully!');
555
+ console.log(` Submission ID: ${(submitted.payload as any).submissionId}`);
556
+ }
557
+
558
+ // Show submissions
559
+ console.log('\nšŸ“Š Submissions:');
560
+ const ctx = engine.getContext();
561
+ ctx.submissions.forEach((sub, i) => {
562
+ console.log(` ${i + 1}. Submitted at ${sub.submittedAt.toLocaleString()}`);
563
+ });
564
+
565
+ // Demonstrate undo
566
+ console.log('\nāŖ Demonstrating undo...');
567
+ console.log(` Before undo: ${ctx.submissions.length} submissions`);
568
+ engine.undo();
569
+ console.log(` After undo: ${engine.getContext().submissions.length} submissions`);
570
+
571
+ console.log('\nšŸŽ‰ Done!');
572
+ }
573
+
574
+ main().catch(console.error);
575
+ ```
576
+
577
+ ## Step 5: Understanding Key Patterns
578
+
579
+ ### 1. Nested State Updates
580
+
581
+ When updating nested structures like fields within a form:
582
+
583
+ ```typescript
584
+ const form = state.context.forms.find(f => f.id === formId);
585
+ if (form) {
586
+ form.fields.push(newField);
587
+ form.updatedAt = new Date();
588
+ }
589
+ ```
590
+
591
+ ### 2. Validation in Rules
592
+
593
+ The submit rule validates data and returns different facts:
594
+
595
+ ```typescript
596
+ if (errors.length > 0) {
597
+ return [ValidationFailed.create({ formId, errors })];
598
+ }
599
+ return [FormSubmitted.create({ formId, submissionId })];
600
+ ```
601
+
602
+ ### 3. History with Complex State
603
+
604
+ Undo/redo works automatically with `enableHistory: true`:
605
+
606
+ ```typescript
607
+ engine.undo(); // Reverts to previous state
608
+ engine.redo(); // Moves forward again
609
+ ```
610
+
611
+ ## Next Steps
612
+
613
+ - Complete the UI with Svelte components
614
+ - Add drag-and-drop field reordering
615
+ - Implement form templates
616
+ - Add export functionality
617
+
618
+ ---
619
+
620
+ **Next Tutorial:** [E-commerce Cart](./ecommerce-cart.md)