@rubix0270/arboris 1.0.2 → 1.0.4

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 (451) hide show
  1. package/package.json +25 -37
  2. package/run.mjs +10 -0
  3. package/dist/cli.mjs +0 -383
  4. package/manifest.json +0 -323
  5. package/prisma/skills/accessibility/SKILL.md +0 -147
  6. package/prisma/skills/agent-architecture-audit/SKILL.md +0 -257
  7. package/prisma/skills/agent-eval/SKILL.md +0 -146
  8. package/prisma/skills/agent-harness-construction/SKILL.md +0 -74
  9. package/prisma/skills/agent-introspection-debugging/SKILL.md +0 -154
  10. package/prisma/skills/agent-payment-x402/SKILL.md +0 -225
  11. package/prisma/skills/agent-self-evaluation/SKILL.md +0 -182
  12. package/prisma/skills/agent-self-evaluation/examples/high-score-example.md +0 -87
  13. package/prisma/skills/agent-self-evaluation/examples/low-score-example.md +0 -86
  14. package/prisma/skills/agent-self-evaluation/references/evaluation-criteria.md +0 -71
  15. package/prisma/skills/agent-self-evaluation/references/hook-integration.md +0 -64
  16. package/prisma/skills/agent-self-evaluation/scripts/evaluate.py +0 -408
  17. package/prisma/skills/agent-self-evaluation/templates/evaluation-report.md +0 -86
  18. package/prisma/skills/agent-sort/SKILL.md +0 -216
  19. package/prisma/skills/agentic-engineering/SKILL.md +0 -64
  20. package/prisma/skills/agentic-os/SKILL.md +0 -388
  21. package/prisma/skills/ai-first-engineering/SKILL.md +0 -52
  22. package/prisma/skills/ai-regression-testing/SKILL.md +0 -386
  23. package/prisma/skills/android-clean-architecture/SKILL.md +0 -340
  24. package/prisma/skills/angular-developer/SKILL.md +0 -155
  25. package/prisma/skills/angular-developer/references/angular-animations.md +0 -160
  26. package/prisma/skills/angular-developer/references/angular-aria.md +0 -410
  27. package/prisma/skills/angular-developer/references/cli.md +0 -86
  28. package/prisma/skills/angular-developer/references/component-harnesses.md +0 -59
  29. package/prisma/skills/angular-developer/references/component-styling.md +0 -91
  30. package/prisma/skills/angular-developer/references/components.md +0 -117
  31. package/prisma/skills/angular-developer/references/creating-services.md +0 -97
  32. package/prisma/skills/angular-developer/references/data-resolvers.md +0 -69
  33. package/prisma/skills/angular-developer/references/define-routes.md +0 -67
  34. package/prisma/skills/angular-developer/references/defining-providers.md +0 -72
  35. package/prisma/skills/angular-developer/references/di-fundamentals.md +0 -120
  36. package/prisma/skills/angular-developer/references/e2e-testing.md +0 -56
  37. package/prisma/skills/angular-developer/references/effects.md +0 -83
  38. package/prisma/skills/angular-developer/references/hierarchical-injectors.md +0 -43
  39. package/prisma/skills/angular-developer/references/host-elements.md +0 -80
  40. package/prisma/skills/angular-developer/references/injection-context.md +0 -63
  41. package/prisma/skills/angular-developer/references/inputs.md +0 -101
  42. package/prisma/skills/angular-developer/references/linked-signal.md +0 -59
  43. package/prisma/skills/angular-developer/references/loading-strategies.md +0 -61
  44. package/prisma/skills/angular-developer/references/mcp.md +0 -108
  45. package/prisma/skills/angular-developer/references/navigate-to-routes.md +0 -69
  46. package/prisma/skills/angular-developer/references/outputs.md +0 -86
  47. package/prisma/skills/angular-developer/references/reactive-forms.md +0 -122
  48. package/prisma/skills/angular-developer/references/rendering-strategies.md +0 -44
  49. package/prisma/skills/angular-developer/references/resource.md +0 -77
  50. package/prisma/skills/angular-developer/references/route-animations.md +0 -56
  51. package/prisma/skills/angular-developer/references/route-guards.md +0 -52
  52. package/prisma/skills/angular-developer/references/router-lifecycle.md +0 -45
  53. package/prisma/skills/angular-developer/references/router-testing.md +0 -87
  54. package/prisma/skills/angular-developer/references/show-routes-with-outlets.md +0 -68
  55. package/prisma/skills/angular-developer/references/signal-forms.md +0 -795
  56. package/prisma/skills/angular-developer/references/signals-overview.md +0 -94
  57. package/prisma/skills/angular-developer/references/tailwind-css.md +0 -69
  58. package/prisma/skills/angular-developer/references/template-driven-forms.md +0 -114
  59. package/prisma/skills/angular-developer/references/testing-fundamentals.md +0 -65
  60. package/prisma/skills/api-connector-builder/SKILL.md +0 -121
  61. package/prisma/skills/api-design/SKILL.md +0 -524
  62. package/prisma/skills/architecture-decision-records/SKILL.md +0 -180
  63. package/prisma/skills/article-writing/SKILL.md +0 -80
  64. package/prisma/skills/automation-audit-ops/SKILL.md +0 -143
  65. package/prisma/skills/autonomous-agent-harness/SKILL.md +0 -274
  66. package/prisma/skills/autonomous-loops/SKILL.md +0 -611
  67. package/prisma/skills/backend-patterns/SKILL.md +0 -562
  68. package/prisma/skills/benchmark/SKILL.md +0 -94
  69. package/prisma/skills/benchmark-methodology/SKILL.md +0 -190
  70. package/prisma/skills/benchmark-optimization-loop/SKILL.md +0 -70
  71. package/prisma/skills/blender-motion-state-inspection/SKILL.md +0 -165
  72. package/prisma/skills/blueprint/SKILL.md +0 -106
  73. package/prisma/skills/brand-discovery/SKILL.md +0 -145
  74. package/prisma/skills/brand-discovery/references/10_purpose-why.md +0 -40
  75. package/prisma/skills/brand-discovery/references/20_positioning.md +0 -44
  76. package/prisma/skills/brand-discovery/references/30_audience-niche.md +0 -52
  77. package/prisma/skills/brand-discovery/references/40_personality-archetype.md +0 -57
  78. package/prisma/skills/brand-discovery/references/50_voice-tone.md +0 -59
  79. package/prisma/skills/brand-discovery/references/60_narrative-story.md +0 -50
  80. package/prisma/skills/brand-discovery/references/70_founder-tension.md +0 -49
  81. package/prisma/skills/brand-discovery/references/90_SYNTHESIS.md +0 -133
  82. package/prisma/skills/brand-voice/SKILL.md +0 -98
  83. package/prisma/skills/brand-voice/references/voice-profile-schema.md +0 -55
  84. package/prisma/skills/browser-qa/SKILL.md +0 -105
  85. package/prisma/skills/bun-runtime/SKILL.md +0 -85
  86. package/prisma/skills/canary-watch/SKILL.md +0 -108
  87. package/prisma/skills/carrier-relationship-management/SKILL.md +0 -212
  88. package/prisma/skills/cisco-ios-patterns/SKILL.md +0 -164
  89. package/prisma/skills/ck/SKILL.md +0 -148
  90. package/prisma/skills/ck/commands/forget.mjs +0 -44
  91. package/prisma/skills/ck/commands/info.mjs +0 -24
  92. package/prisma/skills/ck/commands/init.mjs +0 -143
  93. package/prisma/skills/ck/commands/list.mjs +0 -40
  94. package/prisma/skills/ck/commands/migrate.mjs +0 -202
  95. package/prisma/skills/ck/commands/resume.mjs +0 -36
  96. package/prisma/skills/ck/commands/save.mjs +0 -210
  97. package/prisma/skills/ck/commands/shared.mjs +0 -387
  98. package/prisma/skills/ck/hooks/session-start.mjs +0 -224
  99. package/prisma/skills/claude-devfleet/SKILL.md +0 -112
  100. package/prisma/skills/click-path-audit/SKILL.md +0 -245
  101. package/prisma/skills/clickhouse-io/SKILL.md +0 -440
  102. package/prisma/skills/code-tour/SKILL.md +0 -254
  103. package/prisma/skills/codebase-onboarding/SKILL.md +0 -234
  104. package/prisma/skills/codehealth-mcp/SKILL.md +0 -167
  105. package/prisma/skills/coding-standards/SKILL.md +0 -551
  106. package/prisma/skills/competitive-platform-analysis/SKILL.md +0 -214
  107. package/prisma/skills/competitive-report-structure/SKILL.md +0 -162
  108. package/prisma/skills/compose-multiplatform-patterns/SKILL.md +0 -300
  109. package/prisma/skills/config-gc/SKILL.md +0 -120
  110. package/prisma/skills/configure-ecc/SKILL.md +0 -385
  111. package/prisma/skills/connections-optimizer/SKILL.md +0 -190
  112. package/prisma/skills/content-engine/SKILL.md +0 -132
  113. package/prisma/skills/content-hash-cache-pattern/SKILL.md +0 -162
  114. package/prisma/skills/context-budget/SKILL.md +0 -136
  115. package/prisma/skills/continuous-agent-loop/SKILL.md +0 -46
  116. package/prisma/skills/continuous-learning/SKILL.md +0 -132
  117. package/prisma/skills/continuous-learning/config.json +0 -18
  118. package/prisma/skills/continuous-learning/evaluate-session.sh +0 -69
  119. package/prisma/skills/continuous-learning-v2/SKILL.md +0 -361
  120. package/prisma/skills/continuous-learning-v2/agents/observer-loop.sh +0 -359
  121. package/prisma/skills/continuous-learning-v2/agents/observer.md +0 -189
  122. package/prisma/skills/continuous-learning-v2/agents/session-guardian.sh +0 -150
  123. package/prisma/skills/continuous-learning-v2/agents/start-observer.sh +0 -248
  124. package/prisma/skills/continuous-learning-v2/config.json +0 -8
  125. package/prisma/skills/continuous-learning-v2/hooks/observe.sh +0 -585
  126. package/prisma/skills/continuous-learning-v2/scripts/detect-project.sh +0 -322
  127. package/prisma/skills/continuous-learning-v2/scripts/instinct-cli.py +0 -1956
  128. package/prisma/skills/continuous-learning-v2/scripts/lib/homunculus-dir.sh +0 -31
  129. package/prisma/skills/continuous-learning-v2/scripts/migrate-homunculus.sh +0 -68
  130. package/prisma/skills/continuous-learning-v2/scripts/test_parse_instinct.py +0 -1421
  131. package/prisma/skills/cost-aware-llm-pipeline/SKILL.md +0 -184
  132. package/prisma/skills/cost-tracking/SKILL.md +0 -97
  133. package/prisma/skills/council/SKILL.md +0 -204
  134. package/prisma/skills/cpp-coding-standards/SKILL.md +0 -724
  135. package/prisma/skills/cpp-testing/SKILL.md +0 -325
  136. package/prisma/skills/crosspost/SKILL.md +0 -112
  137. package/prisma/skills/csharp-testing/SKILL.md +0 -322
  138. package/prisma/skills/customer-billing-ops/SKILL.md +0 -141
  139. package/prisma/skills/customs-trade-compliance/SKILL.md +0 -263
  140. package/prisma/skills/dart-flutter-patterns/SKILL.md +0 -564
  141. package/prisma/skills/dashboard-builder/SKILL.md +0 -109
  142. package/prisma/skills/data-scraper-agent/SKILL.md +0 -765
  143. package/prisma/skills/data-throughput-accelerator/SKILL.md +0 -73
  144. package/prisma/skills/database-migrations/SKILL.md +0 -430
  145. package/prisma/skills/deep-research/SKILL.md +0 -160
  146. package/prisma/skills/defi-amm-security/SKILL.md +0 -167
  147. package/prisma/skills/delivery-gate/SKILL.md +0 -126
  148. package/prisma/skills/delivery-gate/hooks/quality-gate.py +0 -220
  149. package/prisma/skills/deployment-patterns/SKILL.md +0 -428
  150. package/prisma/skills/design-system/SKILL.md +0 -83
  151. package/prisma/skills/django-celery/SKILL.md +0 -458
  152. package/prisma/skills/django-patterns/SKILL.md +0 -735
  153. package/prisma/skills/django-security/SKILL.md +0 -644
  154. package/prisma/skills/django-tdd/SKILL.md +0 -730
  155. package/prisma/skills/django-verification/SKILL.md +0 -470
  156. package/prisma/skills/dmux-workflows/SKILL.md +0 -192
  157. package/prisma/skills/docker-patterns/SKILL.md +0 -365
  158. package/prisma/skills/documentation-lookup/SKILL.md +0 -91
  159. package/prisma/skills/dotnet-patterns/SKILL.md +0 -322
  160. package/prisma/skills/dynamic-workflow-mode/SKILL.md +0 -124
  161. package/prisma/skills/e2e-testing/SKILL.md +0 -327
  162. package/prisma/skills/ecc-guide/SKILL.md +0 -190
  163. package/prisma/skills/ecc-recipes/SKILL.md +0 -149
  164. package/prisma/skills/ecc-tools-cost-audit/SKILL.md +0 -161
  165. package/prisma/skills/email-ops/SKILL.md +0 -122
  166. package/prisma/skills/energy-procurement/SKILL.md +0 -228
  167. package/prisma/skills/enterprise-agent-ops/SKILL.md +0 -51
  168. package/prisma/skills/error-handling/SKILL.md +0 -377
  169. package/prisma/skills/eval-harness/SKILL.md +0 -271
  170. package/prisma/skills/evm-token-decimals/SKILL.md +0 -131
  171. package/prisma/skills/exa-search/SKILL.md +0 -108
  172. package/prisma/skills/fal-ai-media/SKILL.md +0 -289
  173. package/prisma/skills/fastapi-patterns/SKILL.md +0 -514
  174. package/prisma/skills/finance-billing-ops/SKILL.md +0 -128
  175. package/prisma/skills/flox-environments/SKILL.md +0 -497
  176. package/prisma/skills/flutter-dart-code-review/SKILL.md +0 -436
  177. package/prisma/skills/foundation-models-on-device/SKILL.md +0 -243
  178. package/prisma/skills/frontend-a11y/SKILL.md +0 -446
  179. package/prisma/skills/frontend-design-direction/SKILL.md +0 -93
  180. package/prisma/skills/frontend-patterns/SKILL.md +0 -657
  181. package/prisma/skills/frontend-slides/SKILL.md +0 -185
  182. package/prisma/skills/frontend-slides/STYLE_PRESETS.md +0 -330
  183. package/prisma/skills/frontend-slides/animation-patterns.md +0 -122
  184. package/prisma/skills/frontend-slides/html-template.md +0 -419
  185. package/prisma/skills/frontend-slides/scripts/export-pdf.sh +0 -418
  186. package/prisma/skills/frontend-slides/scripts/extract-pptx.py +0 -96
  187. package/prisma/skills/frontend-slides/viewport-base.css +0 -153
  188. package/prisma/skills/fsharp-testing/SKILL.md +0 -281
  189. package/prisma/skills/gan-style-harness/SKILL.md +0 -279
  190. package/prisma/skills/gateguard/SKILL.md +0 -133
  191. package/prisma/skills/generating-python-installer/SKILL.md +0 -820
  192. package/prisma/skills/git-workflow/SKILL.md +0 -716
  193. package/prisma/skills/github-ops/SKILL.md +0 -145
  194. package/prisma/skills/golang-patterns/SKILL.md +0 -675
  195. package/prisma/skills/golang-testing/SKILL.md +0 -721
  196. package/prisma/skills/google-workspace-ops/SKILL.md +0 -96
  197. package/prisma/skills/growth-log/SKILL.md +0 -128
  198. package/prisma/skills/healthcare-cdss-patterns/SKILL.md +0 -246
  199. package/prisma/skills/healthcare-emr-patterns/SKILL.md +0 -160
  200. package/prisma/skills/healthcare-eval-harness/SKILL.md +0 -208
  201. package/prisma/skills/healthcare-phi-compliance/SKILL.md +0 -146
  202. package/prisma/skills/hermes-imports/SKILL.md +0 -89
  203. package/prisma/skills/hexagonal-architecture/SKILL.md +0 -277
  204. package/prisma/skills/hipaa-compliance/SKILL.md +0 -79
  205. package/prisma/skills/homelab-network-readiness/SKILL.md +0 -170
  206. package/prisma/skills/homelab-network-setup/SKILL.md +0 -130
  207. package/prisma/skills/homelab-pihole-dns/SKILL.md +0 -275
  208. package/prisma/skills/homelab-vlan-segmentation/SKILL.md +0 -312
  209. package/prisma/skills/homelab-wireguard-vpn/SKILL.md +0 -306
  210. package/prisma/skills/hookify-rules/SKILL.md +0 -128
  211. package/prisma/skills/inherit-legacy-style/SKILL.md +0 -157
  212. package/prisma/skills/intent-driven-development/SKILL.md +0 -360
  213. package/prisma/skills/inventory-demand-planning/SKILL.md +0 -247
  214. package/prisma/skills/investor-materials/SKILL.md +0 -97
  215. package/prisma/skills/investor-outreach/SKILL.md +0 -92
  216. package/prisma/skills/ios-icon-gen/SKILL.md +0 -158
  217. package/prisma/skills/ios-icon-gen/scripts/generate_icons.swift +0 -258
  218. package/prisma/skills/ios-icon-gen/scripts/iconify_gen.sh +0 -235
  219. package/prisma/skills/iterative-retrieval/SKILL.md +0 -212
  220. package/prisma/skills/ito-basket-compare/SKILL.md +0 -64
  221. package/prisma/skills/ito-data-atlas-agent/SKILL.md +0 -64
  222. package/prisma/skills/ito-market-intelligence/SKILL.md +0 -61
  223. package/prisma/skills/ito-trade-planner/SKILL.md +0 -68
  224. package/prisma/skills/java-coding-standards/SKILL.md +0 -384
  225. package/prisma/skills/jira-integration/SKILL.md +0 -303
  226. package/prisma/skills/jpa-patterns/SKILL.md +0 -152
  227. package/prisma/skills/knowledge-ops/SKILL.md +0 -155
  228. package/prisma/skills/kotlin-coroutines-flows/SKILL.md +0 -285
  229. package/prisma/skills/kotlin-exposed-patterns/SKILL.md +0 -720
  230. package/prisma/skills/kotlin-ktor-patterns/SKILL.md +0 -690
  231. package/prisma/skills/kotlin-patterns/SKILL.md +0 -712
  232. package/prisma/skills/kotlin-testing/SKILL.md +0 -825
  233. package/prisma/skills/kubernetes-patterns/SKILL.md +0 -756
  234. package/prisma/skills/laravel-patterns/SKILL.md +0 -416
  235. package/prisma/skills/laravel-plugin-discovery/SKILL.md +0 -230
  236. package/prisma/skills/laravel-security/SKILL.md +0 -948
  237. package/prisma/skills/laravel-tdd/SKILL.md +0 -675
  238. package/prisma/skills/laravel-verification/SKILL.md +0 -180
  239. package/prisma/skills/latency-critical-systems/SKILL.md +0 -74
  240. package/prisma/skills/lead-intelligence/SKILL.md +0 -322
  241. package/prisma/skills/lead-intelligence/agents/enrichment-agent.md +0 -85
  242. package/prisma/skills/lead-intelligence/agents/mutual-mapper.md +0 -75
  243. package/prisma/skills/lead-intelligence/agents/outreach-drafter.md +0 -98
  244. package/prisma/skills/lead-intelligence/agents/signal-scorer.md +0 -60
  245. package/prisma/skills/liquid-glass-design/SKILL.md +0 -279
  246. package/prisma/skills/llm-trading-agent-security/SKILL.md +0 -147
  247. package/prisma/skills/logistics-exception-management/SKILL.md +0 -222
  248. package/prisma/skills/loop-design-check/SKILL.md +0 -143
  249. package/prisma/skills/mailtrap-email-integration/SKILL.md +0 -77
  250. package/prisma/skills/make-interfaces-feel-better/SKILL.md +0 -152
  251. package/prisma/skills/manim-video/SKILL.md +0 -90
  252. package/prisma/skills/manim-video/assets/network_graph_scene.py +0 -52
  253. package/prisma/skills/market-research/SKILL.md +0 -76
  254. package/prisma/skills/marketing-campaign/SKILL.md +0 -114
  255. package/prisma/skills/mcp-server-patterns/SKILL.md +0 -70
  256. package/prisma/skills/messages-ops/SKILL.md +0 -105
  257. package/prisma/skills/ml-adoption-playbook/SKILL.md +0 -57
  258. package/prisma/skills/mle-workflow/SKILL.md +0 -347
  259. package/prisma/skills/motion-advanced/SKILL.md +0 -596
  260. package/prisma/skills/motion-foundations/SKILL.md +0 -299
  261. package/prisma/skills/motion-patterns/SKILL.md +0 -434
  262. package/prisma/skills/motion-ui/SKILL.md +0 -576
  263. package/prisma/skills/mysql-patterns/SKILL.md +0 -413
  264. package/prisma/skills/nanoclaw-repl/SKILL.md +0 -34
  265. package/prisma/skills/nestjs-patterns/SKILL.md +0 -231
  266. package/prisma/skills/netmiko-ssh-automation/SKILL.md +0 -174
  267. package/prisma/skills/network-bgp-diagnostics/SKILL.md +0 -168
  268. package/prisma/skills/network-config-validation/SKILL.md +0 -211
  269. package/prisma/skills/network-interface-health/SKILL.md +0 -153
  270. package/prisma/skills/nextjs-turbopack/SKILL.md +0 -58
  271. package/prisma/skills/nodejs-keccak256/SKILL.md +0 -103
  272. package/prisma/skills/nutrient-document-processing/SKILL.md +0 -168
  273. package/prisma/skills/nuxt4-patterns/SKILL.md +0 -101
  274. package/prisma/skills/openclaw-persona-forge/SKILL.md +0 -289
  275. package/prisma/skills/openclaw-persona-forge/gacha.py +0 -224
  276. package/prisma/skills/openclaw-persona-forge/gacha.sh +0 -5
  277. package/prisma/skills/openclaw-persona-forge/references/avatar-style.md +0 -124
  278. package/prisma/skills/openclaw-persona-forge/references/boundary-rules.md +0 -53
  279. package/prisma/skills/openclaw-persona-forge/references/error-handling.md +0 -53
  280. package/prisma/skills/openclaw-persona-forge/references/identity-tension.md +0 -48
  281. package/prisma/skills/openclaw-persona-forge/references/naming-system.md +0 -39
  282. package/prisma/skills/openclaw-persona-forge/references/output-template.md +0 -166
  283. package/prisma/skills/opensource-pipeline/SKILL.md +0 -256
  284. package/prisma/skills/orch-add-feature/SKILL.md +0 -45
  285. package/prisma/skills/orch-build-mvp/SKILL.md +0 -49
  286. package/prisma/skills/orch-change-feature/SKILL.md +0 -43
  287. package/prisma/skills/orch-fix-defect/SKILL.md +0 -43
  288. package/prisma/skills/orch-pipeline/SKILL.md +0 -121
  289. package/prisma/skills/orch-refine-code/SKILL.md +0 -44
  290. package/prisma/skills/parallel-execution-optimizer/SKILL.md +0 -73
  291. package/prisma/skills/perl-patterns/SKILL.md +0 -505
  292. package/prisma/skills/perl-security/SKILL.md +0 -504
  293. package/prisma/skills/perl-testing/SKILL.md +0 -476
  294. package/prisma/skills/plan-orchestrate/SKILL.md +0 -263
  295. package/prisma/skills/plankton-code-quality/SKILL.md +0 -237
  296. package/prisma/skills/postgres-patterns/SKILL.md +0 -148
  297. package/prisma/skills/prediction-market-oracle-research/SKILL.md +0 -64
  298. package/prisma/skills/prediction-market-risk-review/SKILL.md +0 -61
  299. package/prisma/skills/prisma-patterns/SKILL.md +0 -401
  300. package/prisma/skills/product-capability/SKILL.md +0 -142
  301. package/prisma/skills/product-lens/SKILL.md +0 -93
  302. package/prisma/skills/production-audit/SKILL.md +0 -207
  303. package/prisma/skills/production-scheduling/SKILL.md +0 -238
  304. package/prisma/skills/project-flow-ops/SKILL.md +0 -112
  305. package/prisma/skills/prompt-optimizer/SKILL.md +0 -398
  306. package/prisma/skills/python-patterns/SKILL.md +0 -751
  307. package/prisma/skills/python-testing/SKILL.md +0 -817
  308. package/prisma/skills/pytorch-patterns/SKILL.md +0 -397
  309. package/prisma/skills/quality-nonconformance/SKILL.md +0 -260
  310. package/prisma/skills/quarkus-patterns/SKILL.md +0 -723
  311. package/prisma/skills/quarkus-security/SKILL.md +0 -468
  312. package/prisma/skills/quarkus-tdd/SKILL.md +0 -812
  313. package/prisma/skills/quarkus-verification/SKILL.md +0 -480
  314. package/prisma/skills/ralphinho-rfc-pipeline/SKILL.md +0 -68
  315. package/prisma/skills/react-native-patterns/SKILL.md +0 -326
  316. package/prisma/skills/react-patterns/SKILL.md +0 -342
  317. package/prisma/skills/react-performance/SKILL.md +0 -575
  318. package/prisma/skills/react-testing/SKILL.md +0 -424
  319. package/prisma/skills/recsys-pipeline-architect/SKILL.md +0 -115
  320. package/prisma/skills/recursive-decision-ledger/SKILL.md +0 -80
  321. package/prisma/skills/redis-patterns/SKILL.md +0 -404
  322. package/prisma/skills/regex-vs-llm-structured-text/SKILL.md +0 -221
  323. package/prisma/skills/remotion-video-creation/SKILL.md +0 -43
  324. package/prisma/skills/remotion-video-creation/rules/3d.md +0 -86
  325. package/prisma/skills/remotion-video-creation/rules/animations.md +0 -29
  326. package/prisma/skills/remotion-video-creation/rules/assets/charts-bar-chart.tsx +0 -173
  327. package/prisma/skills/remotion-video-creation/rules/assets/text-animations-typewriter.tsx +0 -100
  328. package/prisma/skills/remotion-video-creation/rules/assets/text-animations-word-highlight.tsx +0 -108
  329. package/prisma/skills/remotion-video-creation/rules/assets.md +0 -78
  330. package/prisma/skills/remotion-video-creation/rules/audio.md +0 -172
  331. package/prisma/skills/remotion-video-creation/rules/calculate-metadata.md +0 -104
  332. package/prisma/skills/remotion-video-creation/rules/can-decode.md +0 -75
  333. package/prisma/skills/remotion-video-creation/rules/charts.md +0 -58
  334. package/prisma/skills/remotion-video-creation/rules/compositions.md +0 -146
  335. package/prisma/skills/remotion-video-creation/rules/display-captions.md +0 -126
  336. package/prisma/skills/remotion-video-creation/rules/extract-frames.md +0 -229
  337. package/prisma/skills/remotion-video-creation/rules/fonts.md +0 -152
  338. package/prisma/skills/remotion-video-creation/rules/get-audio-duration.md +0 -58
  339. package/prisma/skills/remotion-video-creation/rules/get-video-dimensions.md +0 -68
  340. package/prisma/skills/remotion-video-creation/rules/get-video-duration.md +0 -58
  341. package/prisma/skills/remotion-video-creation/rules/gifs.md +0 -138
  342. package/prisma/skills/remotion-video-creation/rules/images.md +0 -130
  343. package/prisma/skills/remotion-video-creation/rules/import-srt-captions.md +0 -67
  344. package/prisma/skills/remotion-video-creation/rules/lottie.md +0 -67
  345. package/prisma/skills/remotion-video-creation/rules/measuring-dom-nodes.md +0 -34
  346. package/prisma/skills/remotion-video-creation/rules/measuring-text.md +0 -143
  347. package/prisma/skills/remotion-video-creation/rules/sequencing.md +0 -106
  348. package/prisma/skills/remotion-video-creation/rules/tailwind.md +0 -11
  349. package/prisma/skills/remotion-video-creation/rules/text-animations.md +0 -20
  350. package/prisma/skills/remotion-video-creation/rules/timing.md +0 -179
  351. package/prisma/skills/remotion-video-creation/rules/transcribe-captions.md +0 -19
  352. package/prisma/skills/remotion-video-creation/rules/transitions.md +0 -122
  353. package/prisma/skills/remotion-video-creation/rules/trimming.md +0 -52
  354. package/prisma/skills/remotion-video-creation/rules/videos.md +0 -171
  355. package/prisma/skills/repo-scan/SKILL.md +0 -79
  356. package/prisma/skills/research-ops/SKILL.md +0 -113
  357. package/prisma/skills/returns-reverse-logistics/SKILL.md +0 -240
  358. package/prisma/skills/rules-distill/SKILL.md +0 -265
  359. package/prisma/skills/rules-distill/scripts/scan-rules.sh +0 -58
  360. package/prisma/skills/rules-distill/scripts/scan-skills.sh +0 -129
  361. package/prisma/skills/rust-patterns/SKILL.md +0 -500
  362. package/prisma/skills/rust-testing/SKILL.md +0 -501
  363. package/prisma/skills/safety-guard/SKILL.md +0 -76
  364. package/prisma/skills/santa-method/SKILL.md +0 -307
  365. package/prisma/skills/scientific-db-pubmed-database/SKILL.md +0 -176
  366. package/prisma/skills/scientific-db-uspto-database/SKILL.md +0 -178
  367. package/prisma/skills/scientific-pkg-gget/SKILL.md +0 -167
  368. package/prisma/skills/scientific-thinking-literature-review/SKILL.md +0 -193
  369. package/prisma/skills/scientific-thinking-scholar-evaluation/SKILL.md +0 -161
  370. package/prisma/skills/search-first/SKILL.md +0 -183
  371. package/prisma/skills/security-bounty-hunter/SKILL.md +0 -100
  372. package/prisma/skills/security-review/SKILL.md +0 -504
  373. package/prisma/skills/security-review/cloud-infrastructure-security.md +0 -361
  374. package/prisma/skills/security-scan/SKILL.md +0 -166
  375. package/prisma/skills/seo/SKILL.md +0 -155
  376. package/prisma/skills/skill-comply/SKILL.md +0 -59
  377. package/prisma/skills/skill-comply/fixtures/compliant_trace.jsonl +0 -5
  378. package/prisma/skills/skill-comply/fixtures/noncompliant_trace.jsonl +0 -3
  379. package/prisma/skills/skill-comply/fixtures/tdd_spec.yaml +0 -44
  380. package/prisma/skills/skill-comply/prompts/classifier.md +0 -24
  381. package/prisma/skills/skill-comply/prompts/scenario_generator.md +0 -62
  382. package/prisma/skills/skill-comply/prompts/spec_generator.md +0 -42
  383. package/prisma/skills/skill-comply/pyproject.toml +0 -15
  384. package/prisma/skills/skill-comply/scripts/__init__.py +0 -0
  385. package/prisma/skills/skill-comply/scripts/classifier.py +0 -85
  386. package/prisma/skills/skill-comply/scripts/grader.py +0 -124
  387. package/prisma/skills/skill-comply/scripts/parser.py +0 -107
  388. package/prisma/skills/skill-comply/scripts/report.py +0 -170
  389. package/prisma/skills/skill-comply/scripts/run.py +0 -127
  390. package/prisma/skills/skill-comply/scripts/runner.py +0 -194
  391. package/prisma/skills/skill-comply/scripts/scenario_generator.py +0 -70
  392. package/prisma/skills/skill-comply/scripts/spec_generator.py +0 -72
  393. package/prisma/skills/skill-comply/scripts/utils.py +0 -13
  394. package/prisma/skills/skill-comply/tests/test_grader.py +0 -197
  395. package/prisma/skills/skill-comply/tests/test_parser.py +0 -90
  396. package/prisma/skills/skill-comply/tests/test_runner.py +0 -172
  397. package/prisma/skills/skill-scout/SKILL.md +0 -141
  398. package/prisma/skills/skill-stocktake/SKILL.md +0 -195
  399. package/prisma/skills/skill-stocktake/scripts/quick-diff.sh +0 -87
  400. package/prisma/skills/skill-stocktake/scripts/save-results.sh +0 -56
  401. package/prisma/skills/skill-stocktake/scripts/scan.sh +0 -170
  402. package/prisma/skills/social-graph-ranker/SKILL.md +0 -155
  403. package/prisma/skills/social-publisher/SKILL.md +0 -130
  404. package/prisma/skills/springboot-patterns/SKILL.md +0 -315
  405. package/prisma/skills/springboot-security/SKILL.md +0 -273
  406. package/prisma/skills/springboot-tdd/SKILL.md +0 -159
  407. package/prisma/skills/springboot-verification/SKILL.md +0 -232
  408. package/prisma/skills/strategic-compact/SKILL.md +0 -136
  409. package/prisma/skills/swift-actor-persistence/SKILL.md +0 -144
  410. package/prisma/skills/swift-concurrency-6-2/SKILL.md +0 -216
  411. package/prisma/skills/swift-protocol-di-testing/SKILL.md +0 -191
  412. package/prisma/skills/swiftui-patterns/SKILL.md +0 -259
  413. package/prisma/skills/taste/SKILL.md +0 -264
  414. package/prisma/skills/taste/references/genre-taxonomy.md +0 -87
  415. package/prisma/skills/tdd-workflow/SKILL.md +0 -583
  416. package/prisma/skills/team-agent-orchestration/SKILL.md +0 -111
  417. package/prisma/skills/team-builder/SKILL.md +0 -169
  418. package/prisma/skills/terminal-ops/SKILL.md +0 -110
  419. package/prisma/skills/tinystruct-patterns/SKILL.md +0 -279
  420. package/prisma/skills/tinystruct-patterns/references/architecture.md +0 -90
  421. package/prisma/skills/tinystruct-patterns/references/data-handling.md +0 -60
  422. package/prisma/skills/tinystruct-patterns/references/database.md +0 -99
  423. package/prisma/skills/tinystruct-patterns/references/routing.md +0 -64
  424. package/prisma/skills/tinystruct-patterns/references/system-usage.md +0 -97
  425. package/prisma/skills/tinystruct-patterns/references/testing.md +0 -72
  426. package/prisma/skills/token-budget-advisor/SKILL.md +0 -134
  427. package/prisma/skills/ui-demo/SKILL.md +0 -466
  428. package/prisma/skills/ui-to-vue/SKILL.md +0 -135
  429. package/prisma/skills/uncloud/SKILL.md +0 -344
  430. package/prisma/skills/unified-notifications-ops/SKILL.md +0 -188
  431. package/prisma/skills/verification-loop/SKILL.md +0 -127
  432. package/prisma/skills/video-editing/SKILL.md +0 -311
  433. package/prisma/skills/videodb/SKILL.md +0 -375
  434. package/prisma/skills/videodb/reference/api-reference.md +0 -550
  435. package/prisma/skills/videodb/reference/capture-reference.md +0 -407
  436. package/prisma/skills/videodb/reference/capture.md +0 -101
  437. package/prisma/skills/videodb/reference/editor.md +0 -443
  438. package/prisma/skills/videodb/reference/generative.md +0 -331
  439. package/prisma/skills/videodb/reference/rtstream-reference.md +0 -564
  440. package/prisma/skills/videodb/reference/rtstream.md +0 -65
  441. package/prisma/skills/videodb/reference/search.md +0 -230
  442. package/prisma/skills/videodb/reference/streaming.md +0 -406
  443. package/prisma/skills/videodb/reference/use-cases.md +0 -118
  444. package/prisma/skills/videodb/scripts/ws_listener.py +0 -282
  445. package/prisma/skills/visa-doc-translate/README.md +0 -86
  446. package/prisma/skills/visa-doc-translate/SKILL.md +0 -117
  447. package/prisma/skills/vite-patterns/SKILL.md +0 -450
  448. package/prisma/skills/vue-patterns/SKILL.md +0 -471
  449. package/prisma/skills/windows-desktop-e2e/SKILL.md +0 -888
  450. package/prisma/skills/workspace-surface-audit/SKILL.md +0 -126
  451. package/prisma/skills/x-api/SKILL.md +0 -235
@@ -1,795 +0,0 @@
1
- # Signal Forms
2
-
3
- Signal Forms are recommended for new forms when the target Angular version supports them. They provide a reactive, type-safe, and model-driven way to manage form state using Angular Signals.
4
-
5
- When using Signal Forms, do not use `null` as a value or type of any fields.
6
-
7
- ## Imports
8
-
9
- You can import the following from `@angular/forms/signals`:
10
-
11
- ```ts
12
- import {
13
- form,
14
- FormField,
15
- submit,
16
- // Rules for field state
17
- disabled,
18
- hidden,
19
- readonly,
20
- debounce,
21
- // Schema helpers
22
- applyWhen,
23
- applyEach,
24
- schema,
25
- // Custom validation
26
- validate,
27
- validateHttp,
28
- validateStandardSchema,
29
- // Metadata
30
- metadata,
31
- } from '@angular/forms/signals';
32
- ```
33
-
34
- ## Creating a Form
35
-
36
- Use the `form()` function with a Signal model. The structure of the form is derived directly from the model.
37
-
38
- ```ts
39
- import {Component, signal} from '@angular/core';
40
- import {form, FormField} from '@angular/forms/signals';
41
-
42
- @Component({
43
- // ...
44
- imports: [FormField],
45
- })
46
- export class Example {
47
- // 1. Define your model with initial values (avoid undefined)
48
- userModel = signal({
49
- name: '', // CRITICAL: NEVER use null or undefined as initial values
50
- email: '',
51
- age: 0, // Use 0 for numbers, NOT null
52
- address: {
53
- street: '',
54
- city: '',
55
- },
56
- hobbies: [] as string[], // Use [] for arrays, NOT null
57
- });
58
-
59
- // WRONG - DO NOT DO THIS:
60
- // badModel = signal({
61
- // name: null, // ERROR: use '' instead
62
- // age: null, // ERROR: use 0 instead
63
- // items: null // ERROR: use [] instead
64
- // });
65
-
66
- // 2. Create the form
67
- userForm = form(this.userModel);
68
- }
69
- ```
70
-
71
- ## Validation
72
-
73
- Import validators from `@angular/forms/signals`.
74
-
75
- ```ts
76
- import {required, email, min, max, minLength, maxLength, pattern} from '@angular/forms/signals';
77
- ```
78
-
79
- Use them in the schema function passed to `form()`:
80
-
81
- ```ts
82
- userForm = form(this.userModel, (schemaPath) => {
83
- // Required
84
- required(schemaPath.name, {message: 'Name is required'});
85
-
86
- // Conditional required.
87
- required(schemaPath.name, {
88
- when({valueOf}) {
89
- return valueOf(schemaPath.age) > 10;
90
- },
91
- });
92
- // when is only available for required
93
- // Do NOT do this: pattern(p.name, /xxx/, {when /* ERROR */)
94
-
95
- // Email
96
- email(schemaPath.email, {message: 'Invalid email'});
97
-
98
- // Min/Max for numbers
99
- min(schemaPath.age, 18);
100
- max(schemaPath.age, 100);
101
-
102
- // MinLength/MaxLength for strings/arrays
103
- minLength(schemaPath.password, 8);
104
- maxLength(schemaPath.description, 500);
105
-
106
- // Pattern (Regex)
107
- pattern(schemaPath.zipCode, /^\d{5}$/);
108
- });
109
- ```
110
-
111
- ## FieldState vs FormField: The Parental Requirement
112
-
113
- It's important to understand the difference between **FormField** (the structure) and **FieldState** (the actual data/signals).
114
-
115
- **RULE**: You must **CALL** a field as a function to access its state signals (valid, touched, dirty, hidden, etc.).
116
-
117
- ```ts
118
- // f is a FormField (structural)
119
- const f = form(signal({cat: {name: 'pirojok-the-cat', age: 5}}));
120
-
121
- f.cat.name; // FormField: You can't get flags from here!
122
- f.cat.name.touched(); // ERROR: touched() does not exist on FormField
123
-
124
- f.cat.name(); // FieldState: Calling it gives you access to signals
125
- f.cat.name().touched(); // VALID: Accessing the signal
126
- f.cat().name.touched(); // ERROR: f.cat() is state, it doesn't have children!
127
- ```
128
-
129
- Similarly in a template:
130
-
131
- ```html
132
- <!-- WRONG: Property 'hidden' does not exist on type 'FormField' -->
133
- @if (bookingForm.hotelDetails.hidden()) { ... }
134
-
135
- <!-- RIGHT: Call it first -->
136
- @if (bookingForm.hotelDetails().hidden()) { ... }
137
- ```
138
-
139
- ## Disabled / Readonly / Hidden
140
-
141
- Control field status using rules in the schema.
142
-
143
- ```ts
144
- import {disabled, readonly, hidden} from '@angular/forms/signals';
145
-
146
- userForm = form(this.userModel, (schemaPath) => {
147
- // Conditionally disabled
148
- disabled(schemaPath.password, ({valueOf}) => !valueOf(schemaPath.createAccount));
149
-
150
- // Conditionally hidden (does NOT remove from model, just marks as hidden)
151
- hidden(schemaPath.shippingAddress, ({valueOf}) => valueOf(schemaPath.sameAsBilling));
152
-
153
- // Readonly
154
- readonly(schemaPath.username);
155
- });
156
- ```
157
-
158
- ## Binding
159
-
160
- Import `FormField` and use the `[formField]` directive.
161
-
162
- ```ts
163
- import {FormField} from '@angular/forms/signals';
164
- ```
165
-
166
- All props on state, such as `disabled`, `hidden`, `readonly` and `name` are bound automatically.
167
- Do _NOT_ bind the `name` field.
168
-
169
- **CRITICAL: FORBIDDEN ATTRIBUTES**
170
- When using `[formField]`, you MUST NOT set the following attributes in the template (either static or bound):
171
-
172
- - `min`, `max` (Use validators in the schema instead)
173
- - `value`, `[value]`, `[attr.value]` (Already handled by `[formField]`)
174
- - `[attr.min]`, `[attr.max]`
175
- - `[disabled]`, `[readonly]` (Already handled by `[formField]`)
176
-
177
- Do NOT do this: `<input min="1" [formField]>` or `<input [value]="val" [formField]>`.
178
-
179
- ```html
180
- <!-- Input -->
181
- <input [formField]="userForm.name" />
182
-
183
- <!-- Checkbox -->
184
- <input type="checkbox" [formField]="userForm.isAdmin" />
185
-
186
- <!-- Select -->
187
- <select [formField]="userForm.country">
188
- <option value="us">US</option>
189
- </select>
190
-
191
- <!-- userForm.name can NOT be nullable, because input does not accept null-->
192
- <input [formField]="userForm.name" />
193
- ```
194
-
195
- ## Reactive Forms
196
-
197
- **Do NOT import** `FormControl`, `FormGroup`, `FormArray`, or `FormBuilder` from `@angular/forms`. Signal Forms replace these concepts entirely.
198
- Signal forms does NOT have a builder.
199
-
200
- ## Accessing State
201
-
202
- Each field in the form is a function that returns its state.
203
-
204
- ```ts
205
- // Access the field by calling it
206
- const emailState = this.userForm.email();
207
-
208
- // Value (WritableSignal)
209
- const value = this.userForm().value();
210
-
211
- // Validation State (Signals)
212
- const isValid = this.userForm().valid();
213
- const isInvalid = this.userForm().invalid();
214
- const errors = this.userForm().errors(); // Array of errors
215
- const isPending = this.userForm().pending(); // Async validation pending
216
-
217
- // Interaction State (Signals)
218
- const isTouched = this.userForm().touched();
219
- const isDirty = this.userForm().dirty();
220
-
221
- // Availability State (Signals)
222
- const isDisabled = this.userForm().disabled();
223
- const isHidden = this.userForm().hidden();
224
- const isReadonly = this.userForm().readonly();
225
- ```
226
-
227
- IMPORTANT!: Make sure to call the field to get it state.
228
-
229
- ```ts
230
- form().invalid()
231
- form.field().dirty()
232
- form.field.subfield().touched()
233
- form.a.b.c.d().value()
234
- form.address.ssn().pending()
235
- form().reset()
236
-
237
- // The only exception is length:
238
- form.children.length
239
- form.length // NOTE: no parenthesis!
240
- form.client.addresses.length // No "()"
241
-
242
- @for (income of form.addresses; track $index) {/**/}
243
- ```
244
-
245
- ## Submitting
246
-
247
- Use the `submit()` function. It automatically marks all fields as touched before running the action.
248
-
249
- **CRITICAL**: The callback to `submit()` MUST be `async` and MUST return a Promise.
250
-
251
- ```ts
252
- import { submit } from '@angular/forms/signals';
253
-
254
- // CORRECT - async callback
255
- onSubmit() {
256
- submit(this.userForm, async () => {
257
- // This only runs if the form is valid
258
- await this.apiService.save(this.userModel());
259
- console.log('Saved!');
260
- });
261
- }
262
-
263
- // WRONG - missing async keyword
264
- onSubmit() {
265
- submit(this.userForm, () => { // ERROR: must be async
266
- console.log('Saved!');
267
- });
268
- }
269
- ```
270
-
271
- ## Handling Errors
272
-
273
- `field().errors()` returns the errors array of ValidationError:
274
-
275
- ```ts
276
- interface ValidationError {
277
- readonly kind: string;
278
- readonly message?: string;
279
- }
280
- ```
281
-
282
- Do _NOT_ return null from validators.
283
- When there are no errors, return undefined
284
-
285
- ### Context
286
-
287
- Functions passed to rules like `validate()`, `disabled()`, `applyWhen` take a context object. It is **CRITICAL** to understand its structure:
288
-
289
- ```ts
290
- validate(
291
- schemaPath.username,
292
- ({
293
- value, // Signal<T>: Writable current value of the field
294
- fieldTree, // FieldTree<T>: Sub-fields (if it's a group/array)
295
- state, // FieldState<T>: Access flags like state.valid(), state.dirty()
296
- valueOf, // (path) => T: Read values of OTHER fields (tracking dependencies), e.g. valueOf(schemaPath.password)
297
- stateOf, // (path) => FieldState: Access state (valid/dirty) of OTHER fields, e.g. stateOf(schemaPath.password).valid()
298
- pathKeys, // Signal<string[]>: Path from root to this field
299
- }) => {
300
- // WRONG: if (touched()) ... (touched is not in context)
301
- // RIGHT: if (state.touched()) ...
302
-
303
- if (value() === 'admin') {
304
- return {kind: 'reserved', message: 'Username admin is reserved'};
305
- }
306
- },
307
- );
308
- ```
309
-
310
- ### IMPORTANT: Paths are NOT Signals
311
-
312
- Inside the `form()` callback, `schemaPath` and its children (e.g., `schemaPath.user.name`) are **NOT** signals and are **NOT** callable.
313
-
314
- ```ts
315
- // WRONG - This will throw an error:
316
- applyWhen(p.ssn, () => p.ssn().touched(), (ssnField) => { ... });
317
-
318
- // RIGHT - Use stateOf() to get the state of a path:
319
- applyWhen(p.ssn, ({ stateOf }) => stateOf(p.ssn).touched(), (ssnField) => { ... });
320
-
321
- // RIGHT - Use valueOf() to get the value of a path:
322
- applyWhen(p.ssn, ({ valueOf }) => valueOf(p.ssn) !== '', (ssnField) => { ... });
323
- ```
324
-
325
- ### Multiple Items
326
-
327
- - Use `applyEach` for applying rules per item.
328
- - **CRITICAL**: `applyEach` callback takes ONLY ONE argument (the item path), NOT two:
329
-
330
- ```ts
331
- // CORRECT - single argument
332
- applyEach(s.items, (item) => {
333
- required(item.name);
334
- });
335
-
336
- // WRONG - do NOT pass index
337
- applyEach(s.items, (item, index) => {
338
- // ERROR: callback takes 1 argument
339
- required(item.name);
340
- });
341
- ```
342
-
343
- - In the template use `@for` to iterate over the items.
344
- - To remove an item from an array, just remove appropriate item from the array in the data.
345
- - **`select` binding**: You CAN bind to `<select [formField]="form.country">`. Ensure options have `value` attributes.
346
-
347
- ### Nested @for Loops
348
-
349
- **CRITICAL**: Angular does NOT have `$parent`. In nested loops, store outer index in a variable:
350
-
351
- ```html
352
- <!-- WRONG - $parent does not exist -->
353
- @for (item of form.items; track $index) { @for (option of item.options; track $index) {
354
- <button (click)="removeOption($parent.$index, $index)">Remove</button>
355
- <!-- ERROR -->
356
- } }
357
-
358
- <!-- CORRECT - use let to store outer index -->
359
- @for (item of form.items; track $index; let outerIndex = $index) { @for (option of item.options;
360
- track $index) {
361
- <button (click)="removeOption(outerIndex, $index)">Remove</button>
362
- } }
363
- ```
364
-
365
- ### Disabling Form Button
366
-
367
- ```html
368
- <button [disabled]="form().invalid() || form().pending()" />
369
- <!-- Or -->
370
- <button [disabled]="taxForm.invalid()" />
371
- ```
372
-
373
- Do NOT use `[disabled]` on an input. `[formField]` will do this.
374
- Do NOT use `[readonly]` on an input. `[formField]` will do this.
375
- If you need to disable or readonly a field, use `disabled()` or `readonly()` rules in the schema.
376
-
377
- ### Async Validation
378
-
379
- Do not use `validate()` for async, instead use `validateAsync()`:
380
-
381
- **CRITICAL**:
382
-
383
- 1. The `params` option MUST be a function that returns the value to validate.
384
- 2. The `onError` handler is **REQUIRED** - it is NOT optional!
385
-
386
- ```ts
387
- import {resource} from '@angular/core';
388
- import {validateAsync} from '@angular/forms/signals';
389
-
390
- userForm = form(this.userModel, (s) => {
391
- validateAsync(s.username, {
392
- // 1. MUST be a function - params takes context and returns the value
393
- params: ({value}) => value(),
394
-
395
- // 2. Create the resource - factory receives a Signal
396
- factory: (username) =>
397
- resource({
398
- params: username, // Use 'params' in resource()
399
- loader: async ({params: value}) => {
400
- await new Promise((resolve) => setTimeout(resolve, 1000));
401
- return value === 'taken';
402
- },
403
- }),
404
-
405
- // 3. Map success to errors
406
- onSuccess: (isTaken) =>
407
- isTaken ? {kind: 'taken', message: 'Username is already taken'} : undefined,
408
-
409
- // 4. Handle errors - THIS IS REQUIRED!
410
- onError: () => ({kind: 'error', message: 'Validation failed'}),
411
- });
412
- });
413
- ```
414
-
415
- **WRONG Examples:**
416
-
417
- ```ts
418
- // WRONG - params must be a function
419
- validateAsync(s.username, {
420
- params: s.username, // ERROR: must be ({ value }) => value()
421
- // ...
422
- });
423
-
424
- // WRONG - missing onError (it's required!)
425
- validateAsync(s.username, {
426
- params: ({value}) => value(),
427
- factory: (username) =>
428
- resource({
429
- /* ... */
430
- }),
431
- onSuccess: (result) => (result ? {kind: 'error'} : undefined),
432
- // ERROR: 'onError' is missing but required!
433
- });
434
- ```
435
-
436
- ### Using Resource
437
-
438
- **CRITICAL**: In Angular's `resource()`, use `params` for the input signal.
439
-
440
- ```ts
441
- // CORRECT
442
- resource({
443
- params: mySignal,
444
- loader: async ({params: value}) => {
445
- /* ... */
446
- },
447
- });
448
-
449
- // WRONG
450
- resource({
451
- request: mySignal, // ERROR: should be 'params'
452
- loader: async ({request}) => {
453
- /* ... */
454
- },
455
- });
456
- ```
457
-
458
- Use `debounce()` to delay synchronization between the UI and the model.
459
-
460
- ```ts
461
- import {debounce} from '@angular/forms/signals';
462
-
463
- userForm = form(this.userModel, (s) => {
464
- // Delay model updates by 300ms
465
- debounce(s.username, 300);
466
- });
467
- ```
468
-
469
- ### Conditional Validation
470
-
471
- ```ts
472
- form(
473
- data,
474
- (path) => {
475
- applyWhen(
476
- name,
477
- ({value}) => value() !== 'admin',
478
- (namePath) => {
479
- validate(namePath.last /* ... */);
480
- disable(namePath.last /* ... */);
481
- },
482
- );
483
- },
484
- {injector: TestBed.inject(Injector)},
485
- );
486
- ```
487
-
488
- `applyWhen` passes the path mapped to the first argument.
489
- If you need parent field, just pass it to `applyWhen`:
490
-
491
- ```ts
492
- form(
493
- data,
494
- (path) => {
495
- applyWhen(
496
- cat,
497
- ({value}) => value().name !== 'admin',
498
- (catPath) => {
499
- require(cat.catPath /* ... */);
500
- },
501
- );
502
- },
503
- {injector: TestBed.inject(Injector)},
504
- );
505
- ```
506
-
507
- ## Common Pitfalls (DO NOT DO THESE)
508
-
509
- | Error Scenario | WRONG (Common Mistake) | RIGHT (Correct Way) |
510
- | :--------------------- | :-------------------------------------------- | :---------------------------------------------------------- |
511
- | **Accessing Flags** | `form.field.valid()` | `form.field().valid()` |
512
- | **Accessing value** | `form.field.value()` | `form.field().value()` |
513
- | **Setting value** | `form.field.set(x)` | Update model signal: `this.model.update(...)` |
514
- | **Form root flags** | `form.invalid()` | `form().invalid()` |
515
- | **Double-calling** | `form.field()()` | `form.field().value()` |
516
- | **Rules Context** | `({ touched }) => touched()` | `({ state }) => state.touched()` |
517
- | **Calling Paths** | `applyWhen(p.foo, () => p.foo() === 'x')` | `applyWhen(p.foo, ({ valueOf }) => valueOf(p.foo) === 'x')` |
518
- | **applyWhen args** | `applyWhen(condition, () => {...})` | `applyWhen(path, condition, schemaFn)` - needs 3 args |
519
- | **Array length** | `form.items().length` | `form.items.length` (structural) |
520
- | **Multi-select array** | `<select [formField]="form.tags">` (string[]) | Use checkboxes for array fields |
521
- | **readonly attribute** | `<input readonly [formField]>` | Use `readonly()` rule in schema |
522
- | **min/max attributes** | `<input min="1" max="10">` | Use `min()` and `max()` rules in schema |
523
- | **value binding** | `<input [value]="val">` | Do NOT use `[value]` with `[formField]` |
524
- | **when option** | `pattern(p.x, /.../, {when: ...})` | `when` only works with `required()` |
525
- | **Submit callback** | `submit(form, () => { ... })` | `submit(form, async () => { ... })` |
526
- | **Async params** | `params: s.field` | `params: ({ value }) => value()` |
527
- | **Async onError** | Omitting `onError` | `onError` is REQUIRED in `validateAsync` |
528
- | **resource() API** | `request: signal` | `params: signal` |
529
- | **applyEach args** | `applyEach(s.items, (item, index) => ...)` | `applyEach(s.items, (item) => ...)` |
530
- | **Nested @for** | `$parent.$index` | Use `let outerIndex = $index` |
531
- | **FormState import** | `import { FormState }` | `FormState` does not exist, use `FieldState` |
532
- | **Null in model** | `signal({ name: null })` | `signal({ name: '' })` or `signal({ age: 0 })` |
533
- | **Validate syntax** | `validate(s.field, { value } => ...)` | `validate(s.field, ({ value }) => ...)` |
534
- | **Checkbox Array** | `[formField]="form.tags"` (string[]) | Checkboxes ONLY bind to `boolean` |
535
-
536
- ## Big Form Example
537
-
538
- ### `src/app/app.ts`
539
-
540
- ```ts
541
- import {Component, signal, ChangeDetectionStrategy} from '@angular/core';
542
- import {
543
- form,
544
- FormField,
545
- submit,
546
- required,
547
- email,
548
- min,
549
- hidden,
550
- applyEach,
551
- validate,
552
- } from '@angular/forms/signals';
553
-
554
- @Component({
555
- selector: 'app-root',
556
- standalone: true,
557
- imports: [FormField],
558
- templateUrl: './app.html',
559
- changeDetection: ChangeDetectionStrategy.OnPush,
560
- })
561
- export class App {
562
- model = signal({
563
- personalInfo: {
564
- firstName: '',
565
- lastName: '',
566
- email: '',
567
- age: 0,
568
- },
569
- tripDetails: {
570
- destination: 'Mars',
571
- launchDate: '',
572
- },
573
- package: {
574
- tier: 'economy',
575
- extras: [] as string[],
576
- },
577
- companions: [] as Array<{name: string; relation: string}>,
578
- });
579
-
580
- bookingForm = form(this.model, (s) => {
581
- required(s.personalInfo.firstName, {message: 'First name is required'});
582
- required(s.personalInfo.lastName, {message: 'Last name is required'});
583
- required(s.personalInfo.email, {message: 'Email is required'});
584
- email(s.personalInfo.email, {message: 'Invalid email address'});
585
- required(s.personalInfo.age, {message: 'Age is required'});
586
- min(s.personalInfo.age, 18, {message: 'Must be at least 18'});
587
-
588
- required(s.tripDetails.destination);
589
- required(s.tripDetails.launchDate);
590
- validate(s.tripDetails.launchDate, ({value}) => {
591
- const date = new Date(value());
592
- if (isNaN(date.getTime())) return undefined;
593
- const today = new Date();
594
- if (date < today) {
595
- return {kind: 'pastData', message: 'Launch date must be in the future'};
596
- }
597
- return undefined;
598
- });
599
-
600
- // valueOf is used to access values of other fields in rules
601
- hidden(s.package.extras, ({valueOf}) => valueOf(s.package.tier) === 'economy');
602
-
603
- applyEach(s.companions, (companion) => {
604
- required(companion.name, {message: 'Companion name required'});
605
- required(companion.relation, {message: 'Relation required'});
606
- });
607
- });
608
-
609
- addCompanion() {
610
- this.model.update((m) => ({
611
- ...m,
612
- companions: [...m.companions, {name: '', relation: ''}],
613
- }));
614
- }
615
-
616
- removeCompanion(index: number) {
617
- this.model.update((m) => ({
618
- ...m,
619
- companions: m.companions.filter((_, i) => i !== index),
620
- }));
621
- }
622
-
623
- onSubmit() {
624
- // CRITICAL: submit callback MUST be async
625
- submit(this.bookingForm, async () => {
626
- console.log('Booking Confirmed:', this.model());
627
- // If you need to do async work:
628
- // await this.apiService.save(this.model());
629
- });
630
- }
631
- }
632
- ```
633
-
634
- ### `src/app/app.html`
635
-
636
- ```html
637
- <form (submit)="onSubmit(); $event.preventDefault()">
638
- <h1>Interstellar Booking</h1>
639
-
640
- <section>
641
- <h2>Personal Info</h2>
642
-
643
- <label>
644
- First Name
645
- <input [formField]="bookingForm.personalInfo.firstName" />
646
- @if (bookingForm.personalInfo.firstName().touched() &&
647
- bookingForm.personalInfo.firstName().errors().length) {
648
- <span>{{ bookingForm.personalInfo.firstName().errors()[0].message }}</span>
649
- }
650
- </label>
651
-
652
- <label>
653
- Last Name
654
- <input [formField]="bookingForm.personalInfo.lastName" />
655
- @if (bookingForm.personalInfo.lastName().touched() &&
656
- bookingForm.personalInfo.lastName().errors().length) {
657
- <span>{{ bookingForm.personalInfo.lastName().errors()[0].message }}</span>
658
- }
659
- </label>
660
-
661
- <label>
662
- Email
663
- <input type="email" [formField]="bookingForm.personalInfo.email" />
664
- @if (bookingForm.personalInfo.email().touched() &&
665
- bookingForm.personalInfo.email().errors().length) {
666
- <span>{{ bookingForm.personalInfo.email().errors()[0].message }}</span>
667
- }
668
- </label>
669
-
670
- <label>
671
- Age
672
- <input type="number" [formField]="bookingForm.personalInfo.age" />
673
- @if (bookingForm.personalInfo.age().touched() &&
674
- bookingForm.personalInfo.age().errors().length) {
675
- <span>{{ bookingForm.personalInfo.age().errors()[0].message }}</span>
676
- }
677
- </label>
678
- </section>
679
-
680
- <section>
681
- <h2>Trip Details</h2>
682
-
683
- <label>
684
- Destination
685
- <select [formField]="bookingForm.tripDetails.destination">
686
- <option value="Mars">Mars</option>
687
- <option value="Moon">Moon</option>
688
- <option value="Titan">Titan</option>
689
- </select>
690
- </label>
691
-
692
- <label>
693
- Launch Date
694
- <input type="date" [formField]="bookingForm.tripDetails.launchDate" />
695
- @if (bookingForm.tripDetails.launchDate().touched() &&
696
- bookingForm.tripDetails.launchDate().errors().length) {
697
- <span>{{ bookingForm.tripDetails.launchDate().errors()[0].message }}</span>
698
- }
699
- </label>
700
- </section>
701
-
702
- <section>
703
- <h2>Package</h2>
704
-
705
- <label>
706
- <input type="radio" value="economy" [formField]="bookingForm.package.tier" />
707
- Economy
708
- </label>
709
- <label>
710
- <input type="radio" value="business" [formField]="bookingForm.package.tier" />
711
- Business
712
- </label>
713
- <label>
714
- <input type="radio" value="first" [formField]="bookingForm.package.tier" />
715
- First Class
716
- </label>
717
-
718
- @if (!bookingForm.package.extras().hidden()) {
719
- <div>
720
- <h3>Extras</h3>
721
- <!-- Multi-select for arrays must use select multiple -->
722
- <select multiple [formField]="bookingForm.package.extras">
723
- <option value="wifi">WiFi</option>
724
- <option value="gym">Gym</option>
725
- </select>
726
- </div>
727
- }
728
- </section>
729
-
730
- <section>
731
- <h2>Companions</h2>
732
- <button type="button" (click)="addCompanion()">Add Companion</button>
733
-
734
- @for (companion of bookingForm.companions; track $index) {
735
- <div>
736
- <input [formField]="companion.name" placeholder="Name" />
737
- @if (companion.name().touched() && companion.name().errors().length) {
738
- <span>{{ companion.name().errors()[0].message }}</span>
739
- }
740
-
741
- <input [formField]="companion.relation" placeholder="Relation" />
742
- @if (companion.relation().touched() && companion.relation().errors().length) {
743
- <span>{{ companion.relation().errors()[0].message }}</span>
744
- }
745
-
746
- <button type="button" (click)="removeCompanion($index)">Remove</button>
747
- </div>
748
- }
749
- </section>
750
-
751
- <button [disabled]="bookingForm().invalid()">Submit</button>
752
- </form>
753
- ```
754
-
755
- ## Recovering from Build Errors
756
-
757
- If you encounter build errors, here are the most common fixes:
758
-
759
- ### `Property 'value' does not exist on type 'FieldTree'`
760
-
761
- **Problem**: Accessing `.value()` directly on a field without calling it first.
762
-
763
- ```ts
764
- // WRONG
765
- const val = this.form.field.value();
766
- // RIGHT
767
- const val = this.form.field().value();
768
- ```
769
-
770
- ### `Property 'set' does not exist on type 'FieldTree'`
771
-
772
- **Problem**: Trying to set values on the form tree. Signal Forms are model-driven.
773
-
774
- ```ts
775
- // WRONG
776
- this.form.address.street.set('Main St');
777
- // RIGHT - update the model signal instead
778
- this.model.update((m) => ({...m, address: {...m.address, street: 'Main St'}}));
779
- ```
780
-
781
- ### `Type 'string[]' is not assignable to type 'string'`
782
-
783
- **Problem**: Binding `[formField]` to an array field with a single-value `<select>`.
784
-
785
- ```html
786
- <!-- WRONG - assignees is string[], select expects string -->
787
- <select [formField]="form.assignees">
788
- ...
789
- </select>
790
-
791
- <!-- RIGHT - Use select multiple for array fields -->
792
- <select multiple [formField]="form.assignees">
793
- <option value="us">US</option>
794
- </select>
795
- ```