@polymorphism-tech/morph-spec 4.6.0 → 4.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (239) hide show
  1. package/README.md +414 -700
  2. package/docs/ARCHITECTURE.md +331 -0
  3. package/docs/CHEATSHEET.md +221 -0
  4. package/docs/COMMAND-FLOWS.md +368 -0
  5. package/docs/QUICKSTART.md +212 -0
  6. package/docs/examples/order-management/contracts.cs +84 -0
  7. package/docs/examples/order-management/proposal.md +24 -0
  8. package/docs/examples/order-management/spec.md +162 -0
  9. package/docs/plans/2026-02-23-ddd-architecture-refactor.md +1153 -0
  10. package/docs/plans/2026-02-23-ddd-nextsteps.md +682 -0
  11. package/docs/plans/2026-02-23-infra-architect-refactor.md +437 -0
  12. package/docs/plans/2026-02-23-nextjs-code-review-design.md +156 -0
  13. package/docs/plans/2026-02-23-nextjs-code-review-impl.md +1254 -0
  14. package/docs/plans/2026-02-23-nextjs-standards-design.md +149 -0
  15. package/docs/plans/2026-02-23-nextjs-standards-impl.md +1846 -0
  16. package/framework/agents/README.md +14 -14
  17. package/framework/agents/architecture/standards-architect.md +159 -159
  18. package/framework/agents/frontend/nextjs-expert.md +87 -127
  19. package/framework/agents/infrastructure/azure-architect.md +147 -147
  20. package/framework/agents/infrastructure/infra-architect.md +45 -0
  21. package/framework/agents.json +1145 -278
  22. package/framework/rules/frontend-standards.md +0 -3
  23. package/framework/rules/nextjs-standards.md +17 -0
  24. package/framework/skills/level-0-meta/code-review-nextjs/SKILL.md +147 -0
  25. package/framework/skills/level-0-meta/code-review-nextjs/references/review-example-nextjs.md +254 -0
  26. package/framework/skills/level-0-meta/tool-usage-guide/SKILL.md +3 -3
  27. package/framework/skills/level-1-workflows/phase-design/SKILL.md +45 -9
  28. package/framework/skills/level-1-workflows/phase-tasks/SKILL.md +38 -0
  29. package/framework/standards/STANDARDS.json +121 -0
  30. package/framework/standards/architecture/ddd/bounded-contexts.md +105 -0
  31. package/framework/standards/architecture/ddd/complexity-levels.md +108 -0
  32. package/framework/standards/architecture/ddd/ubiquitous-language.md +58 -0
  33. package/framework/standards/frontend/nextjs/app-router.md +123 -0
  34. package/framework/standards/frontend/nextjs/components.md +132 -0
  35. package/framework/standards/frontend/nextjs/data-fetching.md +126 -0
  36. package/framework/standards/frontend/nextjs/forms.md +128 -0
  37. package/framework/standards/frontend/nextjs/naming-conventions.md +67 -0
  38. package/framework/standards/frontend/nextjs/project-structure.md +102 -0
  39. package/framework/standards/frontend/nextjs/state-management.md +72 -0
  40. package/framework/standards/frontend/nextjs/testing.md +111 -0
  41. package/framework/templates/REGISTRY.json +538 -142
  42. package/framework/templates/code/dotnet/contracts/contracts-level1.cs +69 -0
  43. package/framework/templates/code/dotnet/contracts/contracts-level2.cs +86 -0
  44. package/framework/templates/code/dotnet/contracts/contracts-level3.cs +41 -0
  45. package/framework/templates/docs/spec.md +49 -0
  46. package/framework/templates/frontend/nextjs/Dockerfile.nextjs.hbs +43 -0
  47. package/framework/templates/frontend/nextjs/client-component.tsx.hbs +26 -0
  48. package/framework/templates/frontend/nextjs/env.mjs.hbs +32 -0
  49. package/framework/templates/frontend/nextjs/feature-form.tsx.hbs +56 -0
  50. package/framework/templates/frontend/nextjs/page.tsx.hbs +22 -0
  51. package/framework/templates/frontend/nextjs/tsconfig.json.hbs +26 -0
  52. package/framework/templates/frontend/nextjs/use-feature.ts.hbs +54 -0
  53. package/framework/templates/project-structure/dotnet-ddd.md +70 -0
  54. package/framework/workflows/docs/enforcement-pipeline.md +2 -1
  55. package/package.json +1 -1
  56. package/scripts/scan-nextjs.mjs +169 -0
  57. package/src/commands/project/doctor.js +52 -1
  58. package/src/commands/project/init.js +15 -1
  59. package/src/commands/project/update.js +6 -1
  60. package/src/lib/standards/standards-context-injector.js +5 -0
  61. package/src/lib/validators/nextjs/index.js +6 -0
  62. package/src/lib/validators/nextjs/next-component-validator.js +181 -0
  63. package/src/lib/validators/validation-runner.js +5 -0
  64. package/src/utils/agents-installer.js +14 -2
  65. package/.morph/.morphversion +0 -5
  66. package/.morph/analytics/threads-log.jsonl +0 -6
  67. package/.morph/config/config.json +0 -8
  68. package/.morph/framework/agents.json +0 -948
  69. package/.morph/framework/standards/STANDARDS.json +0 -812
  70. package/.morph/framework/standards/ai-agents/blazor-ui.md +0 -364
  71. package/.morph/framework/standards/ai-agents/production.md +0 -415
  72. package/.morph/framework/standards/ai-agents/setup.md +0 -418
  73. package/.morph/framework/standards/ai-agents/team-orchestration.md +0 -479
  74. package/.morph/framework/standards/ai-agents/workflows.md +0 -354
  75. package/.morph/framework/standards/architecture/ddd/aggregates.md +0 -120
  76. package/.morph/framework/standards/architecture/ddd/entities.md +0 -99
  77. package/.morph/framework/standards/architecture/ddd/value-objects.md +0 -124
  78. package/.morph/framework/standards/backend/api/minimal-api.md +0 -494
  79. package/.morph/framework/standards/backend/api/rest.md +0 -492
  80. package/.morph/framework/standards/backend/api/validation.md +0 -88
  81. package/.morph/framework/standards/backend/authentication/passkeys.md +0 -428
  82. package/.morph/framework/standards/backend/database/ef-core.md +0 -199
  83. package/.morph/framework/standards/backend/database/migrations.md +0 -393
  84. package/.morph/framework/standards/backend/database/postgresql/database.md +0 -352
  85. package/.morph/framework/standards/backend/database/repository-patterns.md +0 -528
  86. package/.morph/framework/standards/backend/database/vector-search-rag.md +0 -541
  87. package/.morph/framework/standards/backend/dotnet/async.md +0 -366
  88. package/.morph/framework/standards/backend/dotnet/core.md +0 -117
  89. package/.morph/framework/standards/backend/dotnet/di.md +0 -439
  90. package/.morph/framework/standards/backend/dotnet/program-cs-checklist.md +0 -92
  91. package/.morph/framework/standards/backend/integrations/asaas/asaas-api.md +0 -216
  92. package/.morph/framework/standards/backend/integrations/clerk/clerk-auth.md +0 -290
  93. package/.morph/framework/standards/backend/integrations/hangfire/hangfire-jobs.md +0 -350
  94. package/.morph/framework/standards/backend/integrations/resend/resend-email.md +0 -385
  95. package/.morph/framework/standards/context/analytics.md +0 -96
  96. package/.morph/framework/standards/context/bundles.md +0 -110
  97. package/.morph/framework/standards/context/priming.md +0 -78
  98. package/.morph/framework/standards/core/architecture.md +0 -185
  99. package/.morph/framework/standards/core/coding.md +0 -214
  100. package/.morph/framework/standards/core/git-branching-strategy.md +0 -403
  101. package/.morph/framework/standards/core/git.md +0 -185
  102. package/.morph/framework/standards/core/testing.md +0 -295
  103. package/.morph/framework/standards/data/nosql/blob-storage.md +0 -102
  104. package/.morph/framework/standards/data/nosql/cache/redis.md +0 -97
  105. package/.morph/framework/standards/data/nosql/cosmos-db.md +0 -118
  106. package/.morph/framework/standards/data/vector-search/azure-ai-search.md +0 -121
  107. package/.morph/framework/standards/data/vector-search/rag-chunking.md +0 -104
  108. package/.morph/framework/standards/frontend/blazor/design-checklist.md +0 -222
  109. package/.morph/framework/standards/frontend/blazor/fluent-ui-setup.md +0 -595
  110. package/.morph/framework/standards/frontend/blazor/fluent-ui.md +0 -137
  111. package/.morph/framework/standards/frontend/blazor/html-conversion.md +0 -184
  112. package/.morph/framework/standards/frontend/blazor/lifecycle.md +0 -195
  113. package/.morph/framework/standards/frontend/blazor/pitfalls.md +0 -198
  114. package/.morph/framework/standards/frontend/blazor/state.md +0 -191
  115. package/.morph/framework/standards/frontend/design-system/animations.md +0 -151
  116. package/.morph/framework/standards/frontend/design-system/naming.md +0 -64
  117. package/.morph/framework/standards/frontend/nextjs/nextjs-patterns.md +0 -215
  118. package/.morph/framework/standards/infrastructure/azure/azure.md +0 -624
  119. package/.morph/framework/standards/infrastructure/azure/bicep/bicep-patterns.md +0 -422
  120. package/.morph/framework/standards/infrastructure/azure/devops/azure-devops-setup.md +0 -516
  121. package/.morph/framework/standards/infrastructure/azure/devops/local-development.md +0 -520
  122. package/.morph/framework/standards/infrastructure/azure/services/functions.md +0 -486
  123. package/.morph/framework/standards/infrastructure/azure/services/service-bus.md +0 -459
  124. package/.morph/framework/standards/infrastructure/azure/services/storage.md +0 -407
  125. package/.morph/framework/standards/infrastructure/docker/easypanel-deploy.md +0 -196
  126. package/.morph/framework/standards/infrastructure/supabase/mcp-setup.md +0 -252
  127. package/.morph/framework/standards/infrastructure/supabase/supabase-auth.md +0 -176
  128. package/.morph/framework/standards/infrastructure/supabase/supabase-pgvector.md +0 -169
  129. package/.morph/framework/standards/infrastructure/supabase/supabase-rls.md +0 -184
  130. package/.morph/framework/standards/infrastructure/supabase/supabase-storage.md +0 -153
  131. package/.morph/framework/standards/integration/api/graphql.md +0 -91
  132. package/.morph/framework/standards/integration/api/grpc.md +0 -114
  133. package/.morph/framework/standards/integration/api/rest-design.md +0 -95
  134. package/.morph/framework/standards/integration/event-driven/cqrs.md +0 -101
  135. package/.morph/framework/standards/integration/event-driven/event-sourcing.md +0 -124
  136. package/.morph/framework/standards/integration/event-driven/service-bus.md +0 -95
  137. package/.morph/framework/standards/integration/mcp/mcp-tools.md +0 -384
  138. package/.morph/framework/standards/observability/logging.md +0 -131
  139. package/.morph/framework/standards/observability/metrics.md +0 -121
  140. package/.morph/framework/standards/observability/monitoring.md +0 -114
  141. package/.morph/framework/standards/observability/tracing.md +0 -132
  142. package/.morph/framework/standards/workflows/parallel-execution.md +0 -112
  143. package/.morph/framework/standards/workflows/thread-management.md +0 -113
  144. package/.morph/framework/templates/.idea/morph-templates.xml +0 -92
  145. package/.morph/framework/templates/.vscode/morph-templates.code-snippets +0 -186
  146. package/.morph/framework/templates/IDE-SNIPPETS.md +0 -266
  147. package/.morph/framework/templates/README.md +0 -814
  148. package/.morph/framework/templates/REGISTRY.json +0 -1492
  149. package/.morph/framework/templates/code/dotnet/backend/repository.cs +0 -141
  150. package/.morph/framework/templates/code/dotnet/backend/service.cs +0 -139
  151. package/.morph/framework/templates/code/dotnet/contracts/Commands.cs +0 -74
  152. package/.morph/framework/templates/code/dotnet/contracts/Entities.cs +0 -25
  153. package/.morph/framework/templates/code/dotnet/contracts/Queries.cs +0 -74
  154. package/.morph/framework/templates/code/dotnet/contracts/README.md +0 -74
  155. package/.morph/framework/templates/code/dotnet/contracts/api-contracts.cs +0 -173
  156. package/.morph/framework/templates/code/dotnet/contracts/contracts.cs +0 -217
  157. package/.morph/framework/templates/code/dotnet/contracts/contracts.cs.hbs +0 -172
  158. package/.morph/framework/templates/code/dotnet/database/migration.cs +0 -83
  159. package/.morph/framework/templates/code/dotnet/frontend/component.razor +0 -239
  160. package/.morph/framework/templates/code/dotnet/jobs/agent.cs +0 -163
  161. package/.morph/framework/templates/code/dotnet/jobs/job.cs +0 -171
  162. package/.morph/framework/templates/code/dotnet/test.cs +0 -239
  163. package/.morph/framework/templates/code/sql/rls-policy.sql +0 -57
  164. package/.morph/framework/templates/code/sql/supabase-migration.sql +0 -100
  165. package/.morph/framework/templates/code/sql/supabase-migration.template.sql +0 -113
  166. package/.morph/framework/templates/code/typescript/contracts.ts +0 -168
  167. package/.morph/framework/templates/context/CONTEXT-FEATURE.md +0 -276
  168. package/.morph/framework/templates/context/CONTEXT.md +0 -181
  169. package/.morph/framework/templates/docs/clarifications.md +0 -253
  170. package/.morph/framework/templates/docs/onboarding.md +0 -123
  171. package/.morph/framework/templates/docs/proposal.md +0 -182
  172. package/.morph/framework/templates/docs/schema-analysis.md +0 -119
  173. package/.morph/framework/templates/docs/spec.md +0 -149
  174. package/.morph/framework/templates/docs/ui-components.md +0 -124
  175. package/.morph/framework/templates/docs/ui-design-system.md +0 -76
  176. package/.morph/framework/templates/docs/ui-flows.md +0 -167
  177. package/.morph/framework/templates/docs/ui-mockups.md +0 -98
  178. package/.morph/framework/templates/docs/user-stories.md +0 -34
  179. package/.morph/framework/templates/examples/design-system-examples.md +0 -357
  180. package/.morph/framework/templates/examples/spec-examples.md +0 -90
  181. package/.morph/framework/templates/feature/decisions.md +0 -187
  182. package/.morph/framework/templates/feature/recap.md +0 -146
  183. package/.morph/framework/templates/feature/tasks.md +0 -199
  184. package/.morph/framework/templates/infrastructure/azure/Dockerfile.example +0 -82
  185. package/.morph/framework/templates/infrastructure/azure/README.md +0 -286
  186. package/.morph/framework/templates/infrastructure/azure/app-insights.bicep +0 -63
  187. package/.morph/framework/templates/infrastructure/azure/app-service.bicep +0 -164
  188. package/.morph/framework/templates/infrastructure/azure/container-app-env.bicep +0 -49
  189. package/.morph/framework/templates/infrastructure/azure/container-app.bicep +0 -156
  190. package/.morph/framework/templates/infrastructure/azure/deploy-checklist.md +0 -426
  191. package/.morph/framework/templates/infrastructure/azure/deploy.ps1 +0 -229
  192. package/.morph/framework/templates/infrastructure/azure/deploy.sh +0 -208
  193. package/.morph/framework/templates/infrastructure/azure/key-vault.bicep +0 -91
  194. package/.morph/framework/templates/infrastructure/azure/main.bicep +0 -189
  195. package/.morph/framework/templates/infrastructure/azure/parameters.dev.json +0 -29
  196. package/.morph/framework/templates/infrastructure/azure/parameters.prod.json +0 -29
  197. package/.morph/framework/templates/infrastructure/azure/parameters.staging.json +0 -29
  198. package/.morph/framework/templates/infrastructure/azure/sql-database.bicep +0 -103
  199. package/.morph/framework/templates/infrastructure/azure/storage.bicep +0 -106
  200. package/.morph/framework/templates/infrastructure/docker/Dockerfile.template +0 -58
  201. package/.morph/framework/templates/infrastructure/docker/docker-compose.template.yml +0 -67
  202. package/.morph/framework/templates/infrastructure/docker/dockerfile-api.dockerfile +0 -38
  203. package/.morph/framework/templates/infrastructure/docker/dockerfile-web.dockerfile +0 -48
  204. package/.morph/framework/templates/infrastructure/docker/easypanel.template.json +0 -54
  205. package/.morph/framework/templates/infrastructure/github/README.md +0 -593
  206. package/.morph/framework/templates/infrastructure/github/actions/azure-auth/action.yml.hbs +0 -22
  207. package/.morph/framework/templates/infrastructure/github/actions/docker-build-push/action.yml.hbs +0 -45
  208. package/.morph/framework/templates/infrastructure/github/actions/health-check/action.yml.hbs +0 -27
  209. package/.morph/framework/templates/infrastructure/github/workflows/deploy-azure-app-service.yml.hbs +0 -61
  210. package/.morph/framework/templates/infrastructure/github/workflows/deploy-easypanel.yml.hbs +0 -31
  211. package/.morph/framework/templates/infrastructure/github/workflows/docker-build-push.yml.hbs +0 -59
  212. package/.morph/framework/templates/infrastructure/github/workflows/dotnet-build.yml.hbs +0 -39
  213. package/.morph/framework/templates/integrations/asaas-client.cs +0 -387
  214. package/.morph/framework/templates/integrations/asaas-webhook.cs +0 -351
  215. package/.morph/framework/templates/integrations/azure-identity-config.cs +0 -288
  216. package/.morph/framework/templates/integrations/clerk-config.cs +0 -258
  217. package/.morph/framework/templates/meta-prompts/fusion/fusion-agent.md +0 -76
  218. package/.morph/framework/templates/meta-prompts/fusion/fusion-aggregator.md +0 -100
  219. package/.morph/framework/templates/meta-prompts/hops/hop-retry.md +0 -78
  220. package/.morph/framework/templates/meta-prompts/hops/hop-validation.md +0 -97
  221. package/.morph/framework/templates/meta-prompts/hops/hop-wrapper.md +0 -36
  222. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-coordinator.md +0 -113
  223. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-worker.md +0 -80
  224. package/.morph/framework/templates/meta-prompts/squad-leaders/backend-squad.md +0 -90
  225. package/.morph/framework/templates/meta-prompts/squad-leaders/frontend-squad.md +0 -126
  226. package/.morph/framework/templates/meta-prompts/squad-leaders/squad-leader.md +0 -43
  227. package/.morph/framework/templates/meta-prompts/validators/checkpoint-validator.md +0 -107
  228. package/.morph/framework/templates/meta-prompts/validators/pre-commit-validator.md +0 -95
  229. package/.morph/framework/templates/saas/subscription.cs +0 -347
  230. package/.morph/framework/templates/saas/tenant.cs +0 -338
  231. package/.morph/framework/templates/state.template.json +0 -17
  232. package/.morph/framework/templates/ui/FluentDesignTheme.cs +0 -149
  233. package/.morph/framework/templates/ui/MudTheme.cs +0 -281
  234. package/.morph/framework/templates/ui/design-system.css +0 -226
  235. package/.morph/logs/tool-failures.log +0 -7
  236. package/.morph/memory/pre-compact-2026-02-23T15-43-03-521Z.json +0 -16
  237. package/.morph/state.json +0 -48
  238. package/framework/templates/code/dotnet/contracts/contracts.cs +0 -217
  239. package/framework/templates/code/dotnet/contracts/contracts.cs.hbs +0 -172
@@ -1,137 +0,0 @@
1
- # Fluent UI Blazor - Reference Guide
2
-
3
- > **Scope:** blazor-azure
4
- > **Layer:** 2 (on keyword)
5
- > **Keywords:** blazor, fluent, fluentui, components, datagrid, dialog
6
- > **Load When:** fluent ui keywords detected
7
-
8
- ## Quick Reference
9
-
10
- | Topic | Correct | Incorrect |
11
- |-------|---------|-----------|
12
- | **Icon Sizes** | Size20, Size24, Size28, Size32, Size48 | Size12, Size14, Size16 (don't exist) |
13
- | **Icon Namespace** | `@using Icons = Microsoft.FluentUI.AspNetCore.Components.Icons` | `@using ...Icons` (without alias) |
14
- | **Toast Methods** | `ToastService.ShowSuccess("msg")` | `await ToastService.ShowSuccessAsync("msg")` |
15
- | **Dialog Open** | `_dialog.Show()` | `await _dialog.ShowAsync()` |
16
- | **Dialog Close** | `await _dialog.CloseAsync()` | `await _dialog.HideAsync()` |
17
- | **Dialog Visibility** | `Hidden="true"` (default hidden) | Without Hidden (visible on load!) |
18
-
19
- ## Setup
20
-
21
- ```razor
22
- @* _Imports.razor *@
23
- @using Microsoft.FluentUI.AspNetCore.Components
24
- @using Icons = Microsoft.FluentUI.AspNetCore.Components.Icons
25
- ```
26
-
27
- ```csharp
28
- // Program.cs
29
- builder.Services.AddFluentUIComponents();
30
- ```
31
-
32
- ```razor
33
- @* MainLayout.razor *@
34
- <FluentToastProvider />
35
- <FluentDialogProvider />
36
- <main>@Body</main>
37
- ```
38
-
39
- ## Icons
40
-
41
- | Size | Use For |
42
- |------|---------|
43
- | Size20 | Inputs, badges, inline |
44
- | Size24 | Buttons, lists, menus |
45
- | Size28 | Medium-large elements |
46
- | Size32 | Headers, cards |
47
- | Size48 | Illustrations, empty states |
48
-
49
- ```razor
50
- <FluentIcon Value="@(new Icons.Regular.Size24.Home())" />
51
- <FluentIcon Value="@(new Icons.Filled.Size20.CheckmarkCircle())" Color="Color.Accent" />
52
-
53
- <!-- For sizes < 20, use CSS -->
54
- <FluentIcon Value="@(new Icons.Filled.Size20.Star())" Style="width: 14px; height: 14px;" />
55
- ```
56
-
57
- **Missing Icons — Alternatives:**
58
-
59
- | Missing | Use Instead |
60
- |---------|-------------|
61
- | Compass | CompassNorthwest |
62
- | Instagram | Camera |
63
- | WhatsApp | Chat |
64
- | Facebook/Twitter/LinkedIn | Share / Person |
65
-
66
- ## Toast Service (Synchronous!)
67
-
68
- ```csharp
69
- @inject IToastService ToastService
70
-
71
- // Methods are SYNC, not async
72
- ToastService.ShowSuccess("Saved!");
73
- ToastService.ShowError("Error: " + ex.Message);
74
- ToastService.ShowWarning("Warning");
75
- ToastService.ShowInfo("Processing...");
76
- ```
77
-
78
- ## FluentDialog
79
-
80
- ```razor
81
- <FluentButton OnClick="() => _dialog.Show()">Open</FluentButton>
82
-
83
- <FluentDialog @ref="_dialog" Hidden="true" Modal="true" PreventDismissOnOverlayClick="true">
84
- <FluentDialogHeader>
85
- <FluentLabel Typo="Typography.H4">Confirm</FluentLabel>
86
- </FluentDialogHeader>
87
- <FluentDialogBody><p>Continue?</p></FluentDialogBody>
88
- <FluentDialogFooter>
89
- <FluentButton Appearance="Appearance.Neutral" OnClick="() => _dialog.CloseAsync()">Cancel</FluentButton>
90
- <FluentButton Appearance="Appearance.Accent" OnClick="Confirm">OK</FluentButton>
91
- </FluentDialogFooter>
92
- </FluentDialog>
93
-
94
- @code {
95
- private FluentDialog _dialog = default!;
96
- private async Task Confirm() { /* logic */ await _dialog.CloseAsync(); }
97
- }
98
- ```
99
-
100
- > **Custom modals:** See `html-to-blazor.md` for custom modal pattern when design doesn't follow Fluent.
101
-
102
- ## FluentDataGrid
103
-
104
- ```razor
105
- <FluentDataGrid Items="@_orders" Virtualize="true" ItemSize="50">
106
- <PropertyColumn Property="@(o => o.OrderNumber)" Title="Order" Sortable="true" />
107
- <PropertyColumn Property="@(o => o.Total)" Title="Total" Format="C" />
108
- <TemplateColumn Title="Status">
109
- <FluentBadge Appearance="@GetBadge(context.Status)">@context.Status</FluentBadge>
110
- </TemplateColumn>
111
- <TemplateColumn Title="Actions">
112
- <FluentButton Appearance="Appearance.Lightweight" OnClick="@(() => View(context))">View</FluentButton>
113
- </TemplateColumn>
114
- </FluentDataGrid>
115
- ```
116
-
117
- ## Common Errors
118
-
119
- | Error | Cause | Fix |
120
- |-------|-------|-----|
121
- | "Size16 does not exist" | Invalid icon size | Use Size20, 24, 28, 32, 48 |
122
- | "'Regular' not in context" | Missing icon alias | `@using Icons = ...Icons` |
123
- | Dialog visible on page load | Missing `Hidden="true"` | Add `Hidden="true"` |
124
- | "ShowSuccessAsync not found" | Async method doesn't exist | Use sync `ShowSuccess()` |
125
-
126
- ## Checklist
127
-
128
- - [ ] `AddFluentUIComponents()` in Program.cs
129
- - [ ] `FluentToastProvider` + `FluentDialogProvider` in layout
130
- - [ ] Icons use alias `Icons =` and valid sizes (20-48)
131
- - [ ] Toast methods are synchronous
132
- - [ ] Dialogs have `Hidden="true"`
133
- - [ ] DataGrid uses `Virtualize` for large lists
134
-
135
- ---
136
-
137
- *MORPH-SPEC by Polymorphism Tech*
@@ -1,184 +0,0 @@
1
- # Guia de Conversao HTML para Blazor
2
-
3
- > **Scope:** blazor-azure
4
- > **Layer:** 2 (on keyword)
5
- > **Keywords:** blazor, html, conversion, prototype, migrate
6
- > **Load When:** html conversion keywords detected
7
-
8
- ## Element Mapping
9
-
10
- ### Basic Elements
11
-
12
- | HTML | Blazor (Fluent UI) |
13
- |------|---------------------|
14
- | `<button class="btn-primary">` | `<FluentButton Appearance="Appearance.Accent">` |
15
- | `<button class="btn-secondary">` | `<FluentButton Appearance="Appearance.Neutral">` |
16
- | `<input type="text">` | `<FluentTextField @bind-Value="Model.Field">` |
17
- | `<input type="email">` | `<FluentTextField Type="TextFieldType.Email">` |
18
- | `<input type="password">` | `<FluentTextField Type="TextFieldType.Password">` |
19
- | `<textarea>` | `<FluentTextArea @bind-Value="Model.Field">` |
20
- | `<select>` | `<FluentSelect @bind-Value="Model.Field">` |
21
- | `<input type="checkbox">` | `<FluentCheckbox @bind-Value="Model.Flag">` |
22
- | `<input type="radio">` | `<FluentRadioGroup @bind-Value="Model.Option">` |
23
-
24
- ### Containers & Layout
25
-
26
- | HTML | Blazor |
27
- |------|--------|
28
- | `<div class="card">` | `<FluentCard>` |
29
- | `<div class="flex">` | `<FluentStack Orientation="Horizontal">` |
30
- | `<div class="flex-col">` | `<FluentStack Orientation="Vertical">` |
31
- | `<div class="grid">` | `<FluentGrid>` |
32
- | `<div class="modal">` | Custom HTML or `<FluentDialog>` |
33
-
34
- ### Feedback & Data
35
-
36
- | HTML | Blazor |
37
- |------|--------|
38
- | `<div class="alert-success">` | `<FluentMessageBar Intent="MessageIntent.Success">` |
39
- | `<div class="alert-error">` | `<FluentMessageBar Intent="MessageIntent.Error">` |
40
- | `<div class="toast">` | `ToastService.ShowSuccess("msg")` |
41
- | `<div class="spinner">` | `<FluentProgress />` |
42
- | `<table>` | `<FluentDataGrid Items="@items">` |
43
- | `<th>` | `<PropertyColumn Property="@(x => x.Prop)">` |
44
- | `<td>` with actions | `<TemplateColumn>` |
45
-
46
- ---
47
-
48
- ## Page Structure
49
-
50
- ```razor
51
- @page "/pagename"
52
- @rendermode InteractiveServer
53
- @inject IMyService _service
54
-
55
- <PageTitle>App - Pagina</PageTitle>
56
-
57
- <div class="page-layout">
58
- <header class="page-header-nav"><!-- Nav --></header>
59
- <div class="page-content container">
60
- <h1>Titulo</h1>
61
- <!-- Content -->
62
- </div>
63
- </div>
64
-
65
- @code {
66
- private bool _isLoading;
67
- private string? _error;
68
- protected override async Task OnInitializedAsync() => await LoadDataAsync();
69
- private async Task HandleClick() { }
70
- private string FormatValue(decimal value) => value.ToString("C");
71
- }
72
- ```
73
-
74
- ---
75
-
76
- ## Event Conversion
77
-
78
- | HTML Event | Blazor Pattern |
79
- |------------|---------------|
80
- | `onclick="fn()"` | `@onclick="HandleClick"` (sync) or `@onclick="HandleClickAsync"` (async) |
81
- | `onchange="fn(val)"` | `@bind-Value="_val"` (two-way) or `@onchange="HandleChange"` (one-way) |
82
- | `<form onsubmit>` | `<EditForm Model="@_model" OnValidSubmit="HandleSubmit">` with `<DataAnnotationsValidator />` |
83
-
84
- ### Form Example
85
-
86
- ```razor
87
- <EditForm Model="@_model" OnValidSubmit="HandleSubmit">
88
- <DataAnnotationsValidator />
89
- <FluentTextField @bind-Value="_model.Email" Label="Email" Required />
90
- <FluentButton Type="ButtonType.Submit" Appearance="Appearance.Accent">Enviar</FluentButton>
91
- </EditForm>
92
-
93
- @code {
94
- private MyModel _model = new();
95
- private async Task HandleSubmit() => await _service.SaveAsync(_model);
96
- public class MyModel { [Required][EmailAddress] public string Email { get; set; } = ""; }
97
- }
98
- ```
99
-
100
- ---
101
-
102
- ## Modal Conversion
103
-
104
- **Use custom HTML** when: custom design, backdrop blur, complex animations.
105
- **Use `<FluentDialog>`** for simple confirmations.
106
-
107
- ### Custom Modal
108
-
109
- ```razor
110
- @if (!_modalHidden)
111
- {
112
- <div class="modal-overlay" @onclick="CloseModal">
113
- <div class="modal-container animate-scaleIn" @onclick:stopPropagation="true">
114
- <div class="modal-header">
115
- <h3>Titulo</h3>
116
- <button class="modal-close" @onclick="CloseModal">
117
- <FluentIcon Value="@(new Icons.Regular.Size20.Dismiss())" />
118
- </button>
119
- </div>
120
- <div class="modal-body">Conteudo...</div>
121
- <div class="modal-footer">
122
- <FluentButton Appearance="Appearance.Neutral" @onclick="CloseModal">Cancelar</FluentButton>
123
- <FluentButton Appearance="Appearance.Accent" @onclick="Confirm">Confirmar</FluentButton>
124
- </div>
125
- </div>
126
- </div>
127
- }
128
- @code {
129
- private bool _modalHidden = true;
130
- private void OpenModal() => _modalHidden = false;
131
- private void CloseModal() => _modalHidden = true;
132
- private async Task Confirm() { /* logic */ CloseModal(); }
133
- }
134
- ```
135
-
136
- ---
137
-
138
- ## Style Interpolation
139
-
140
- ```razor
141
- <!-- Static: same as HTML -->
142
- <div style="color: red; margin-top: 20px;">
143
-
144
- <!-- Dynamic: ALWAYS use @($"...") -->
145
- <div style="@($"color: {_color}; margin-top: {_margin}px;")">
146
-
147
- <!-- WRONG: causes RZ9986 -->
148
- <FluentStack Style="animation-delay: @(0.05 * index)s;">
149
- <!-- CORRECT -->
150
- <FluentStack Style="@($"animation-delay: {0.05 * index}s;")">
151
- ```
152
-
153
- ## Loops & Conditionals
154
-
155
- ```razor
156
- <!-- Loop -->
157
- @foreach (var item in _items) { <li>@item.Name</li> }
158
-
159
- <!-- Loop with stagger animation -->
160
- @for (int i = 0; i < _items.Count; i++)
161
- {
162
- var index = i; // capture for closure
163
- <li class="animate-slideInUp" style="@($"animation-delay: {0.05 * index}s;")">@_items[index].Name</li>
164
- }
165
-
166
- <!-- Conditional rendering (replaces display:none toggle) -->
167
- @if (_isLoading) { <FluentProgress /> }
168
- else if (_error != null) { <FluentMessageBar Intent="MessageIntent.Error">@_error</FluentMessageBar> }
169
- else { <FluentDataGrid Items="@_items"><!-- columns --></FluentDataGrid> }
170
- ```
171
-
172
- ## Checklist
173
-
174
- - [ ] Page structure: `@page`, `@rendermode`, `@inject`
175
- - [ ] HTML elements mapped to Fluent UI components
176
- - [ ] Events: `onclick` → `@onclick`, forms → `EditForm`
177
- - [ ] Modals: custom vs `FluentDialog` decided
178
- - [ ] Style interpolation uses `@($"...")`
179
- - [ ] Loops use `var index = i` for closures
180
- - [ ] CSS classes validated with `morph-spec validate-css`
181
-
182
- ---
183
-
184
- *MORPH-SPEC by Polymorphism Tech*
@@ -1,195 +0,0 @@
1
- # Blazor Lifecycle: Pitfalls & Solutions
2
-
3
- > **Scope:** blazor-azure
4
- > **Layer:** 2 (on keyword)
5
- > **Keywords:** blazor, lifecycle, oninit, onafter, onparametersset, dispose
6
- > **Load When:** blazor work
7
-
8
- ## Lifecycle Flow
9
-
10
- ```
11
- 1. SetParametersAsync
12
- 2. OnInitialized(Async) ← No JS available (prerendering)
13
- 3. OnParametersSet(Async)
14
- 4. BuildRenderTree (Render)
15
- 5. OnAfterRender(Async) ← JS available (firstRender=true)
16
- 6. [User Interaction] → 3→4→5 (firstRender=false)
17
- 7. Dispose (component removed)
18
- ```
19
-
20
- ## Decision Matrix
21
-
22
- | Need to... | Use Method | Note |
23
- |------------|------------|------|
24
- | Load data from API/DB | `OnInitializedAsync` | No JS available |
25
- | Call JavaScript / DOM | `OnAfterRenderAsync(firstRender)` | firstRender guard |
26
- | React to parameter changes | `OnParametersSetAsync` | Called on every change |
27
- | Clean up resources | `Dispose` | Timers, subscriptions |
28
- | Update UI after async op | `StateHasChanged()` | Manual re-render |
29
-
30
- ## Common Pitfalls
31
-
32
- | Problem | Error | Solution |
33
- |---------|-------|----------|
34
- | JSRuntime in OnInitialized | "statically rendered" | Move to `OnAfterRenderAsync(firstRender)` |
35
- | Layout with @rendermode | "Cannot pass Body with rendermode" | Remove `@rendermode` from layouts |
36
- | HttpClient no BaseAddress | "Invalid request URI" | Use `NavigationManager.BaseUri` |
37
- | UI not updating after await | Stale UI | Call `StateHasChanged()` after async |
38
- | File upload > 512KB | "exceeds maximum size" | `OpenReadStream(maxAllowedSize: size)` |
39
- | Dispose not called | Memory leaks | Implement `IDisposable` |
40
- | Cascading param null | NullRef during init | Check null in `OnParametersSet()` |
41
- | Infinite render loop | Browser freezes | Never `StateHasChanged()` in property setters |
42
-
43
- ---
44
-
45
- ## Essential Patterns
46
-
47
- ### JSRuntime (OnAfterRender only)
48
-
49
- ```csharp
50
- @inject IJSRuntime JSRuntime
51
- @code {
52
- private bool _initialized;
53
- protected override async Task OnAfterRenderAsync(bool firstRender)
54
- {
55
- if (firstRender && !_initialized)
56
- {
57
- await JSRuntime.InvokeVoidAsync("initializeChart", "myChart");
58
- _initialized = true;
59
- StateHasChanged();
60
- }
61
- }
62
- }
63
- ```
64
-
65
- ### Data Loading
66
-
67
- ```csharp
68
- @code {
69
- private List<Item> Items { get; set; } = new();
70
- private bool Loading = true;
71
- private string? Error;
72
-
73
- protected override async Task OnInitializedAsync()
74
- {
75
- try { Items = await HttpClient.GetFromJsonAsync<List<Item>>("/api/items"); }
76
- catch (Exception ex) { Error = ex.Message; }
77
- finally { Loading = false; }
78
- }
79
- }
80
- ```
81
-
82
- ### Parameter Change
83
-
84
- ```csharp
85
- @code {
86
- [Parameter] public int ProductId { get; set; }
87
- private Product? Product;
88
-
89
- protected override async Task OnParametersSetAsync()
90
- {
91
- if (ProductId > 0)
92
- Product = await HttpClient.GetFromJsonAsync<Product>($"/api/products/{ProductId}");
93
- }
94
- }
95
- ```
96
-
97
- ### Dispose Pattern
98
-
99
- ```csharp
100
- @implements IDisposable
101
- @code {
102
- private System.Timers.Timer? _timer;
103
- protected override void OnInitialized()
104
- {
105
- _timer = new System.Timers.Timer(1000);
106
- _timer.Elapsed += (s, e) => InvokeAsync(StateHasChanged);
107
- _timer.Start();
108
- }
109
- public void Dispose() { _timer?.Stop(); _timer?.Dispose(); }
110
- }
111
- ```
112
-
113
- ### File Upload
114
-
115
- ```csharp
116
- @code {
117
- private const long MaxFileSize = 10 * 1024 * 1024;
118
- private async Task HandleFile(InputFileChangeEventArgs e)
119
- {
120
- if (e.File.Size > MaxFileSize) { Error = "File > 10MB"; return; }
121
- using var stream = e.File.OpenReadStream(maxAllowedSize: MaxFileSize);
122
- var buffer = new byte[e.File.Size];
123
- await stream.ReadAsync(buffer);
124
- }
125
- }
126
- ```
127
-
128
- ### Infinite Render Loop
129
-
130
- ```csharp
131
- // WRONG - StateHasChanged in property setter = infinite loop
132
- private int Counter { get => _c; set { _c = value; StateHasChanged(); } }
133
-
134
- // CORRECT - StateHasChanged only in event handlers
135
- private int _c;
136
- private void Increment() { _c++; StateHasChanged(); }
137
- ```
138
-
139
- ---
140
-
141
- ## Performance
142
-
143
- | Pattern | Code |
144
- |---------|------|
145
- | Prevent unnecessary renders | `protected override bool ShouldRender() => _dataChanged;` |
146
- | List items with key | `@foreach (var item in Items) { <ItemComponent @key="item.Id" Item="@item" /> }` |
147
- | Debounce input | `_debounce = new Timer(500); _debounce.Elapsed += async (s, a) => await Search(value);` |
148
-
149
- ---
150
-
151
- ## Checklist
152
-
153
- - [ ] No JSRuntime in `OnInitializedAsync`
154
- - [ ] `StateHasChanged()` after async operations
155
- - [ ] File uploads specify `maxAllowedSize`
156
- - [ ] Dispose pattern for subscriptions/timers
157
- - [ ] No `@rendermode` on layouts
158
- - [ ] Cascading parameters null-checked
159
-
160
- ---
161
-
162
- *MORPH-SPEC by Polymorphism Tech*
163
-
164
- ---
165
-
166
- ## Code Review Checklist
167
-
168
- Use this checklist during code review for Blazor component lifecycle:
169
-
170
- ### OnInitializedAsync vs OnAfterRenderAsync
171
-
172
- - [ ] **JSRuntime calls are in `OnAfterRenderAsync`, not `OnInitializedAsync`?**
173
- - [ ] **Database/API calls are in `OnInitializedAsync`, not `OnAfterRenderAsync`?**
174
- - [ ] **`firstRender` check is used to prevent duplicate JS calls?**
175
-
176
- ### StateHasChanged
177
-
178
- - [ ] **`StateHasChanged()` called after async operations that update UI?**
179
- - [ ] **No excessive `StateHasChanged()` calls in loops?**
180
-
181
- ### IDisposable / IAsyncDisposable
182
-
183
- - [ ] **Component implements `IDisposable` if it has timers, event handlers, or CTS?**
184
- - [ ] **`CancellationTokenSource` is cancelled and disposed?**
185
- - [ ] **Event handlers are unsubscribed in `Dispose`?**
186
- - [ ] **Timers are stopped and disposed?**
187
-
188
- ### RenderMode
189
-
190
- - [ ] **`@rendermode InteractiveServer` is on pages, NOT on layouts?**
191
- - [ ] **Layout components have NO `@rendermode` directive?**
192
-
193
- ---
194
-
195
- *MORPH-SPEC by Polymorphism Tech*