@heytherevibin/skillforge 0.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +16 -0
- package/CODE_OF_CONDUCT.md +34 -0
- package/CONTRIBUTING.md +38 -0
- package/LICENSE +21 -0
- package/README.md +337 -0
- package/RELEASING.md +93 -0
- package/SECURITY.md +31 -0
- package/STRATEGY.md +26 -0
- package/bin/cli.js +547 -0
- package/lib/packs.js +184 -0
- package/package.json +38 -0
- package/python/app/__init__.py +0 -0
- package/python/app/__pycache__/__init__.cpython-312.pyc +0 -0
- package/python/app/__pycache__/auth.cpython-312.pyc +0 -0
- package/python/app/__pycache__/main.cpython-312.pyc +0 -0
- package/python/app/auth.py +63 -0
- package/python/app/cli.py +78 -0
- package/python/app/db_paths.py +26 -0
- package/python/app/events_cli.py +175 -0
- package/python/app/main.py +647 -0
- package/python/app/materialize.py +138 -0
- package/python/app/mcp_server.py +610 -0
- package/python/app/route_cli.py +117 -0
- package/python/requirements-dev.txt +1 -0
- package/python/requirements.txt +7 -0
- package/python/tests/test_db_paths.py +41 -0
- package/skills/accessibility/SKILL.md +145 -0
- package/skills/agent-architecture-audit/SKILL.md +256 -0
- package/skills/agent-eval/SKILL.md +144 -0
- package/skills/agent-harness-construction/SKILL.md +72 -0
- package/skills/agent-introspection-debugging/SKILL.md +152 -0
- package/skills/agent-payment-x402/SKILL.md +224 -0
- package/skills/agent-sort/SKILL.md +214 -0
- package/skills/agentic-engineering/SKILL.md +62 -0
- package/skills/agentic-os/SKILL.md +386 -0
- package/skills/ai-first-engineering/SKILL.md +50 -0
- package/skills/ai-regression-testing/SKILL.md +384 -0
- package/skills/android-clean-architecture/SKILL.md +338 -0
- package/skills/angular-developer/SKILL.md +153 -0
- package/skills/angular-developer/references/angular-animations.md +160 -0
- package/skills/angular-developer/references/angular-aria.md +410 -0
- package/skills/angular-developer/references/cli.md +86 -0
- package/skills/angular-developer/references/component-harnesses.md +59 -0
- package/skills/angular-developer/references/component-styling.md +91 -0
- package/skills/angular-developer/references/components.md +117 -0
- package/skills/angular-developer/references/creating-services.md +97 -0
- package/skills/angular-developer/references/data-resolvers.md +69 -0
- package/skills/angular-developer/references/define-routes.md +67 -0
- package/skills/angular-developer/references/defining-providers.md +72 -0
- package/skills/angular-developer/references/di-fundamentals.md +120 -0
- package/skills/angular-developer/references/e2e-testing.md +56 -0
- package/skills/angular-developer/references/effects.md +83 -0
- package/skills/angular-developer/references/hierarchical-injectors.md +43 -0
- package/skills/angular-developer/references/host-elements.md +80 -0
- package/skills/angular-developer/references/injection-context.md +63 -0
- package/skills/angular-developer/references/inputs.md +101 -0
- package/skills/angular-developer/references/linked-signal.md +59 -0
- package/skills/angular-developer/references/loading-strategies.md +61 -0
- package/skills/angular-developer/references/mcp.md +108 -0
- package/skills/angular-developer/references/navigate-to-routes.md +69 -0
- package/skills/angular-developer/references/outputs.md +86 -0
- package/skills/angular-developer/references/reactive-forms.md +122 -0
- package/skills/angular-developer/references/rendering-strategies.md +44 -0
- package/skills/angular-developer/references/resource.md +77 -0
- package/skills/angular-developer/references/route-animations.md +56 -0
- package/skills/angular-developer/references/route-guards.md +52 -0
- package/skills/angular-developer/references/router-lifecycle.md +45 -0
- package/skills/angular-developer/references/router-testing.md +87 -0
- package/skills/angular-developer/references/show-routes-with-outlets.md +68 -0
- package/skills/angular-developer/references/signal-forms.md +795 -0
- package/skills/angular-developer/references/signals-overview.md +94 -0
- package/skills/angular-developer/references/tailwind-css.md +69 -0
- package/skills/angular-developer/references/template-driven-forms.md +114 -0
- package/skills/angular-developer/references/testing-fundamentals.md +65 -0
- package/skills/api-connector-builder/SKILL.md +120 -0
- package/skills/api-design/SKILL.md +522 -0
- package/skills/architecture-decision-records/SKILL.md +178 -0
- package/skills/article-writing/SKILL.md +78 -0
- package/skills/automation-audit-ops/SKILL.md +141 -0
- package/skills/autonomous-agent-harness/SKILL.md +272 -0
- package/skills/autonomous-loops/SKILL.md +609 -0
- package/skills/backend-patterns/SKILL.md +560 -0
- package/skills/benchmark/SKILL.md +92 -0
- package/skills/blueprint/SKILL.md +104 -0
- package/skills/browser-qa/SKILL.md +86 -0
- package/skills/bun-runtime/SKILL.md +83 -0
- package/skills/canary-watch/SKILL.md +98 -0
- package/skills/carrier-relationship-management/SKILL.md +211 -0
- package/skills/cisco-ios-patterns/SKILL.md +163 -0
- package/skills/ck/SKILL.md +147 -0
- package/skills/ck/commands/forget.mjs +44 -0
- package/skills/ck/commands/info.mjs +24 -0
- package/skills/ck/commands/init.mjs +143 -0
- package/skills/ck/commands/list.mjs +40 -0
- package/skills/ck/commands/migrate.mjs +202 -0
- package/skills/ck/commands/resume.mjs +36 -0
- package/skills/ck/commands/save.mjs +210 -0
- package/skills/ck/commands/shared.mjs +387 -0
- package/skills/ck/hooks/session-start.mjs +224 -0
- package/skills/claude-devfleet/SKILL.md +103 -0
- package/skills/click-path-audit/SKILL.md +244 -0
- package/skills/clickhouse-io/SKILL.md +438 -0
- package/skills/code-tour/SKILL.md +235 -0
- package/skills/codebase-onboarding/SKILL.md +232 -0
- package/skills/coding-standards/SKILL.md +548 -0
- package/skills/compose-multiplatform-patterns/SKILL.md +298 -0
- package/skills/connections-optimizer/SKILL.md +188 -0
- package/skills/content-engine/SKILL.md +126 -0
- package/skills/content-hash-cache-pattern/SKILL.md +160 -0
- package/skills/context-budget/SKILL.md +134 -0
- package/skills/continuous-agent-loop/SKILL.md +44 -0
- package/skills/continuous-learning/SKILL.md +129 -0
- package/skills/continuous-learning/config.json +18 -0
- package/skills/continuous-learning/evaluate-session.sh +69 -0
- package/skills/continuous-learning-v2/SKILL.md +358 -0
- package/skills/continuous-learning-v2/agents/observer-loop.sh +322 -0
- package/skills/continuous-learning-v2/agents/observer.md +198 -0
- package/skills/continuous-learning-v2/agents/session-guardian.sh +150 -0
- package/skills/continuous-learning-v2/agents/start-observer.sh +248 -0
- package/skills/continuous-learning-v2/config.json +8 -0
- package/skills/continuous-learning-v2/hooks/observe.sh +476 -0
- package/skills/continuous-learning-v2/scripts/detect-project.sh +288 -0
- package/skills/continuous-learning-v2/scripts/instinct-cli.py +1519 -0
- package/skills/continuous-learning-v2/scripts/lib/homunculus-dir.sh +31 -0
- package/skills/continuous-learning-v2/scripts/migrate-homunculus.sh +62 -0
- package/skills/continuous-learning-v2/scripts/test_parse_instinct.py +1018 -0
- package/skills/cost-aware-llm-pipeline/SKILL.md +182 -0
- package/skills/cost-tracking/SKILL.md +147 -0
- package/skills/council/SKILL.md +202 -0
- package/skills/cpp-coding-standards/SKILL.md +722 -0
- package/skills/cpp-testing/SKILL.md +323 -0
- package/skills/crosspost/SKILL.md +110 -0
- package/skills/csharp-testing/SKILL.md +320 -0
- package/skills/customer-billing-ops/SKILL.md +139 -0
- package/skills/customs-trade-compliance/SKILL.md +262 -0
- package/skills/dart-flutter-patterns/SKILL.md +562 -0
- package/skills/dashboard-builder/SKILL.md +108 -0
- package/skills/data-scraper-agent/SKILL.md +764 -0
- package/skills/database-migrations/SKILL.md +428 -0
- package/skills/deep-research/SKILL.md +158 -0
- package/skills/defi-amm-security/SKILL.md +166 -0
- package/skills/deployment-patterns/SKILL.md +426 -0
- package/skills/design-system/SKILL.md +81 -0
- package/skills/django-celery/SKILL.md +456 -0
- package/skills/django-patterns/SKILL.md +733 -0
- package/skills/django-security/SKILL.md +592 -0
- package/skills/django-tdd/SKILL.md +728 -0
- package/skills/django-verification/SKILL.md +468 -0
- package/skills/dmux-workflows/SKILL.md +190 -0
- package/skills/docker-patterns/SKILL.md +363 -0
- package/skills/documentation-lookup/SKILL.md +89 -0
- package/skills/dotnet-patterns/SKILL.md +320 -0
- package/skills/e2e-testing/SKILL.md +325 -0
- package/skills/email-ops/SKILL.md +120 -0
- package/skills/energy-procurement/SKILL.md +227 -0
- package/skills/enterprise-agent-ops/SKILL.md +49 -0
- package/skills/error-handling/SKILL.md +375 -0
- package/skills/eval-harness/SKILL.md +269 -0
- package/skills/evm-token-decimals/SKILL.md +130 -0
- package/skills/exa-search/SKILL.md +106 -0
- package/skills/fal-ai-media/SKILL.md +287 -0
- package/skills/fastapi-patterns/SKILL.md +327 -0
- package/skills/finance-billing-ops/SKILL.md +126 -0
- package/skills/flox-environments/SKILL.md +496 -0
- package/skills/flutter-dart-code-review/SKILL.md +434 -0
- package/skills/foundation-models-on-device/SKILL.md +243 -0
- package/skills/frontend-design-direction/SKILL.md +92 -0
- package/skills/frontend-patterns/SKILL.md +641 -0
- package/skills/frontend-slides/SKILL.md +183 -0
- package/skills/frontend-slides/STYLE_PRESETS.md +330 -0
- package/skills/frontend-slides/animation-patterns.md +122 -0
- package/skills/frontend-slides/html-template.md +419 -0
- package/skills/frontend-slides/scripts/export-pdf.sh +418 -0
- package/skills/frontend-slides/scripts/extract-pptx.py +96 -0
- package/skills/frontend-slides/viewport-base.css +153 -0
- package/skills/fsharp-testing/SKILL.md +279 -0
- package/skills/gan-style-harness/SKILL.md +278 -0
- package/skills/gateguard/SKILL.md +125 -0
- package/skills/git-workflow/SKILL.md +714 -0
- package/skills/github-ops/SKILL.md +143 -0
- package/skills/golang-patterns/SKILL.md +673 -0
- package/skills/golang-testing/SKILL.md +719 -0
- package/skills/google-workspace-ops/SKILL.md +94 -0
- package/skills/healthcare-cdss-patterns/SKILL.md +245 -0
- package/skills/healthcare-emr-patterns/SKILL.md +159 -0
- package/skills/healthcare-eval-harness/SKILL.md +207 -0
- package/skills/healthcare-phi-compliance/SKILL.md +145 -0
- package/skills/hermes-imports/SKILL.md +87 -0
- package/skills/hexagonal-architecture/SKILL.md +275 -0
- package/skills/hipaa-compliance/SKILL.md +78 -0
- package/skills/homelab-network-readiness/SKILL.md +169 -0
- package/skills/homelab-network-setup/SKILL.md +129 -0
- package/skills/homelab-pihole-dns/SKILL.md +274 -0
- package/skills/homelab-vlan-segmentation/SKILL.md +311 -0
- package/skills/homelab-wireguard-vpn/SKILL.md +305 -0
- package/skills/hookify-rules/SKILL.md +128 -0
- package/skills/inventory-demand-planning/SKILL.md +246 -0
- package/skills/investor-materials/SKILL.md +95 -0
- package/skills/investor-outreach/SKILL.md +90 -0
- package/skills/ios-icon-gen/SKILL.md +157 -0
- package/skills/ios-icon-gen/scripts/generate_icons.swift +258 -0
- package/skills/ios-icon-gen/scripts/iconify_gen.sh +235 -0
- package/skills/iterative-retrieval/SKILL.md +209 -0
- package/skills/java-coding-standards/SKILL.md +382 -0
- package/skills/jira-integration/SKILL.md +292 -0
- package/skills/jpa-patterns/SKILL.md +150 -0
- package/skills/knowledge-ops/SKILL.md +153 -0
- package/skills/kotlin-coroutines-flows/SKILL.md +283 -0
- package/skills/kotlin-exposed-patterns/SKILL.md +718 -0
- package/skills/kotlin-ktor-patterns/SKILL.md +688 -0
- package/skills/kotlin-patterns/SKILL.md +710 -0
- package/skills/kotlin-testing/SKILL.md +823 -0
- package/skills/laravel-patterns/SKILL.md +414 -0
- package/skills/laravel-plugin-discovery/SKILL.md +228 -0
- package/skills/laravel-security/SKILL.md +284 -0
- package/skills/laravel-tdd/SKILL.md +282 -0
- package/skills/laravel-verification/SKILL.md +178 -0
- package/skills/lead-intelligence/SKILL.md +320 -0
- package/skills/lead-intelligence/agents/enrichment-agent.md +85 -0
- package/skills/lead-intelligence/agents/mutual-mapper.md +75 -0
- package/skills/lead-intelligence/agents/outreach-drafter.md +98 -0
- package/skills/lead-intelligence/agents/signal-scorer.md +60 -0
- package/skills/liquid-glass-design/SKILL.md +279 -0
- package/skills/llm-trading-agent-security/SKILL.md +146 -0
- package/skills/logistics-exception-management/SKILL.md +221 -0
- package/skills/make-interfaces-feel-better/SKILL.md +151 -0
- package/skills/manim-video/SKILL.md +88 -0
- package/skills/manim-video/assets/network_graph_scene.py +52 -0
- package/skills/market-research/SKILL.md +74 -0
- package/skills/mcp-server-patterns/SKILL.md +68 -0
- package/skills/messages-ops/SKILL.md +103 -0
- package/skills/mle-workflow/SKILL.md +345 -0
- package/skills/motion-advanced/SKILL.md +596 -0
- package/skills/motion-foundations/SKILL.md +299 -0
- package/skills/motion-patterns/SKILL.md +435 -0
- package/skills/motion-ui/SKILL.md +574 -0
- package/skills/mysql-patterns/SKILL.md +411 -0
- package/skills/nanoclaw-repl/SKILL.md +32 -0
- package/skills/nestjs-patterns/SKILL.md +229 -0
- package/skills/netmiko-ssh-automation/SKILL.md +173 -0
- package/skills/network-bgp-diagnostics/SKILL.md +167 -0
- package/skills/network-config-validation/SKILL.md +210 -0
- package/skills/network-interface-health/SKILL.md +152 -0
- package/skills/nextjs-turbopack/SKILL.md +43 -0
- package/skills/nodejs-keccak256/SKILL.md +102 -0
- package/skills/nutrient-document-processing/SKILL.md +166 -0
- package/skills/nuxt4-patterns/SKILL.md +99 -0
- package/skills/openclaw-persona-forge/SKILL.md +288 -0
- package/skills/openclaw-persona-forge/gacha.py +224 -0
- package/skills/openclaw-persona-forge/gacha.sh +5 -0
- package/skills/openclaw-persona-forge/references/avatar-style.md +124 -0
- package/skills/openclaw-persona-forge/references/boundary-rules.md +53 -0
- package/skills/openclaw-persona-forge/references/error-handling.md +53 -0
- package/skills/openclaw-persona-forge/references/identity-tension.md +48 -0
- package/skills/openclaw-persona-forge/references/naming-system.md +39 -0
- package/skills/openclaw-persona-forge/references/output-template.md +166 -0
- package/skills/opensource-pipeline/SKILL.md +254 -0
- package/skills/perl-patterns/SKILL.md +503 -0
- package/skills/perl-security/SKILL.md +502 -0
- package/skills/perl-testing/SKILL.md +474 -0
- package/skills/plan-orchestrate/SKILL.md +253 -0
- package/skills/plankton-code-quality/SKILL.md +236 -0
- package/skills/postgres-patterns/SKILL.md +146 -0
- package/skills/product-capability/SKILL.md +140 -0
- package/skills/product-lens/SKILL.md +91 -0
- package/skills/production-audit/SKILL.md +206 -0
- package/skills/production-scheduling/SKILL.md +237 -0
- package/skills/project-flow-ops/SKILL.md +110 -0
- package/skills/prompt-optimizer/SKILL.md +398 -0
- package/skills/python-patterns/SKILL.md +749 -0
- package/skills/python-testing/SKILL.md +815 -0
- package/skills/pytorch-patterns/SKILL.md +395 -0
- package/skills/quality-nonconformance/SKILL.md +259 -0
- package/skills/quarkus-patterns/SKILL.md +721 -0
- package/skills/quarkus-security/SKILL.md +466 -0
- package/skills/quarkus-tdd/SKILL.md +810 -0
- package/skills/quarkus-verification/SKILL.md +478 -0
- package/skills/ralphinho-rfc-pipeline/SKILL.md +66 -0
- package/skills/redis-patterns/SKILL.md +402 -0
- package/skills/regex-vs-llm-structured-text/SKILL.md +219 -0
- package/skills/remotion-video-creation/SKILL.md +43 -0
- package/skills/remotion-video-creation/rules/3d.md +86 -0
- package/skills/remotion-video-creation/rules/animations.md +29 -0
- package/skills/remotion-video-creation/rules/assets/charts-bar-chart.tsx +173 -0
- package/skills/remotion-video-creation/rules/assets/text-animations-typewriter.tsx +100 -0
- package/skills/remotion-video-creation/rules/assets/text-animations-word-highlight.tsx +108 -0
- package/skills/remotion-video-creation/rules/assets.md +78 -0
- package/skills/remotion-video-creation/rules/audio.md +172 -0
- package/skills/remotion-video-creation/rules/calculate-metadata.md +104 -0
- package/skills/remotion-video-creation/rules/can-decode.md +75 -0
- package/skills/remotion-video-creation/rules/charts.md +58 -0
- package/skills/remotion-video-creation/rules/compositions.md +146 -0
- package/skills/remotion-video-creation/rules/display-captions.md +126 -0
- package/skills/remotion-video-creation/rules/extract-frames.md +229 -0
- package/skills/remotion-video-creation/rules/fonts.md +152 -0
- package/skills/remotion-video-creation/rules/get-audio-duration.md +58 -0
- package/skills/remotion-video-creation/rules/get-video-dimensions.md +68 -0
- package/skills/remotion-video-creation/rules/get-video-duration.md +58 -0
- package/skills/remotion-video-creation/rules/gifs.md +138 -0
- package/skills/remotion-video-creation/rules/images.md +130 -0
- package/skills/remotion-video-creation/rules/import-srt-captions.md +67 -0
- package/skills/remotion-video-creation/rules/lottie.md +67 -0
- package/skills/remotion-video-creation/rules/measuring-dom-nodes.md +34 -0
- package/skills/remotion-video-creation/rules/measuring-text.md +143 -0
- package/skills/remotion-video-creation/rules/sequencing.md +106 -0
- package/skills/remotion-video-creation/rules/tailwind.md +11 -0
- package/skills/remotion-video-creation/rules/text-animations.md +20 -0
- package/skills/remotion-video-creation/rules/timing.md +179 -0
- package/skills/remotion-video-creation/rules/transcribe-captions.md +19 -0
- package/skills/remotion-video-creation/rules/transitions.md +122 -0
- package/skills/remotion-video-creation/rules/trimming.md +52 -0
- package/skills/remotion-video-creation/rules/videos.md +171 -0
- package/skills/repo-scan/SKILL.md +78 -0
- package/skills/research-ops/SKILL.md +111 -0
- package/skills/returns-reverse-logistics/SKILL.md +239 -0
- package/skills/rules-distill/SKILL.md +263 -0
- package/skills/rules-distill/scripts/scan-rules.sh +58 -0
- package/skills/rules-distill/scripts/scan-skills.sh +129 -0
- package/skills/rust-patterns/SKILL.md +498 -0
- package/skills/rust-testing/SKILL.md +499 -0
- package/skills/safety-guard/SKILL.md +74 -0
- package/skills/santa-method/SKILL.md +306 -0
- package/skills/scientific-db-pubmed-database/SKILL.md +175 -0
- package/skills/scientific-db-uspto-database/SKILL.md +177 -0
- package/skills/scientific-pkg-gget/SKILL.md +166 -0
- package/skills/scientific-thinking-literature-review/SKILL.md +192 -0
- package/skills/scientific-thinking-scholar-evaluation/SKILL.md +160 -0
- package/skills/search-first/SKILL.md +181 -0
- package/skills/security-bounty-hunter/SKILL.md +99 -0
- package/skills/security-review/SKILL.md +502 -0
- package/skills/security-review/cloud-infrastructure-security.md +361 -0
- package/skills/seo/SKILL.md +153 -0
- package/skills/skill-comply/SKILL.md +57 -0
- package/skills/skill-comply/fixtures/compliant_trace.jsonl +5 -0
- package/skills/skill-comply/fixtures/noncompliant_trace.jsonl +3 -0
- package/skills/skill-comply/fixtures/tdd_spec.yaml +44 -0
- package/skills/skill-comply/prompts/classifier.md +24 -0
- package/skills/skill-comply/prompts/scenario_generator.md +62 -0
- package/skills/skill-comply/prompts/spec_generator.md +42 -0
- package/skills/skill-comply/pyproject.toml +15 -0
- package/skills/skill-comply/scripts/__init__.py +0 -0
- package/skills/skill-comply/scripts/classifier.py +85 -0
- package/skills/skill-comply/scripts/grader.py +124 -0
- package/skills/skill-comply/scripts/parser.py +107 -0
- package/skills/skill-comply/scripts/report.py +170 -0
- package/skills/skill-comply/scripts/run.py +127 -0
- package/skills/skill-comply/scripts/runner.py +186 -0
- package/skills/skill-comply/scripts/scenario_generator.py +70 -0
- package/skills/skill-comply/scripts/spec_generator.py +72 -0
- package/skills/skill-comply/scripts/utils.py +13 -0
- package/skills/skill-comply/tests/test_grader.py +197 -0
- package/skills/skill-comply/tests/test_parser.py +90 -0
- package/skills/skill-comply/tests/test_runner.py +172 -0
- package/skills/skill-scout/SKILL.md +139 -0
- package/skills/skill-stocktake/SKILL.md +193 -0
- package/skills/skill-stocktake/scripts/quick-diff.sh +87 -0
- package/skills/skill-stocktake/scripts/save-results.sh +56 -0
- package/skills/skill-stocktake/scripts/scan.sh +170 -0
- package/skills/social-graph-ranker/SKILL.md +153 -0
- package/skills/springboot-patterns/SKILL.md +313 -0
- package/skills/springboot-security/SKILL.md +271 -0
- package/skills/springboot-tdd/SKILL.md +157 -0
- package/skills/springboot-verification/SKILL.md +230 -0
- package/skills/strategic-compact/SKILL.md +129 -0
- package/skills/strategic-compact/suggest-compact.sh +54 -0
- package/skills/swift-actor-persistence/SKILL.md +142 -0
- package/skills/swift-concurrency-6-2/SKILL.md +216 -0
- package/skills/swift-protocol-di-testing/SKILL.md +189 -0
- package/skills/swiftui-patterns/SKILL.md +259 -0
- package/skills/tdd-workflow/SKILL.md +462 -0
- package/skills/team-builder/SKILL.md +166 -0
- package/skills/terminal-ops/SKILL.md +108 -0
- package/skills/tinystruct-patterns/SKILL.md +130 -0
- package/skills/tinystruct-patterns/references/architecture.md +77 -0
- package/skills/tinystruct-patterns/references/data-handling.md +35 -0
- package/skills/tinystruct-patterns/references/routing.md +57 -0
- package/skills/tinystruct-patterns/references/system-usage.md +74 -0
- package/skills/tinystruct-patterns/references/testing.md +59 -0
- package/skills/token-budget-advisor/SKILL.md +133 -0
- package/skills/ui-demo/SKILL.md +464 -0
- package/skills/ui-to-vue/SKILL.md +134 -0
- package/skills/unified-notifications-ops/SKILL.md +186 -0
- package/skills/verification-loop/SKILL.md +125 -0
- package/skills/video-editing/SKILL.md +309 -0
- package/skills/videodb/SKILL.md +373 -0
- package/skills/videodb/reference/api-reference.md +550 -0
- package/skills/videodb/reference/capture-reference.md +407 -0
- package/skills/videodb/reference/capture.md +101 -0
- package/skills/videodb/reference/editor.md +443 -0
- package/skills/videodb/reference/generative.md +331 -0
- package/skills/videodb/reference/rtstream-reference.md +564 -0
- package/skills/videodb/reference/rtstream.md +65 -0
- package/skills/videodb/reference/search.md +230 -0
- package/skills/videodb/reference/streaming.md +406 -0
- package/skills/videodb/reference/use-cases.md +118 -0
- package/skills/videodb/scripts/ws_listener.py +282 -0
- package/skills/visa-doc-translate/README.md +86 -0
- package/skills/visa-doc-translate/SKILL.md +117 -0
- package/skills/vite-patterns/SKILL.md +448 -0
- package/skills/windows-desktop-e2e/SKILL.md +787 -0
- package/skills/workspace-surface-audit/SKILL.md +124 -0
- package/skills/x-api/SKILL.md +233 -0
|
@@ -0,0 +1,795 @@
|
|
|
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
|
+
```
|