@polymorphism-tech/morph-spec 4.6.0 → 4.7.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (239) hide show
  1. package/README.md +394 -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
@@ -0,0 +1,126 @@
1
+ # Next.js Data Fetching Standard
2
+
3
+ > **Scope:** frontend/nextjs/data-fetching
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** data fetching, tanstack query, react query, server component, fetch, api, useQuery, useMutation
6
+ > **Load When:** fetching data from .NET API
7
+
8
+ Server Components for initial page load. TanStack Query for client mutations and interactive data. Never `useEffect` for fetching.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS use Server Components for initial data that doesn't require interactivity
13
+ - ALWAYS use TanStack Query (`useQuery`, `useMutation`) for client-side data needs
14
+ - NEVER use `useEffect(() => { fetch(...) }, [])` — this is the old pattern
15
+ - ALWAYS validate API responses with Zod before using them
16
+ - ALWAYS use query key factories for consistent cache invalidation
17
+
18
+ ## TanStack Query Setup
19
+
20
+ ```tsx
21
+ // lib/query-client.tsx
22
+ 'use client';
23
+
24
+ import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
25
+ import { useState } from 'react';
26
+
27
+ export function QueryProvider({ children }: { children: React.ReactNode }) {
28
+ const [queryClient] = useState(() => new QueryClient({
29
+ defaultOptions: { queries: { staleTime: 60 * 1000, retry: 1 } },
30
+ }));
31
+ return <QueryClientProvider client={queryClient}>{children}</QueryClientProvider>;
32
+ }
33
+ ```
34
+
35
+ ## Query Key Factory Pattern
36
+
37
+ ```ts
38
+ // features/users/hooks/query-keys.ts
39
+ export const userKeys = {
40
+ all: ['users'] as const,
41
+ lists: () => [...userKeys.all, 'list'] as const,
42
+ list: (filters: Record<string, unknown>) => [...userKeys.lists(), filters] as const,
43
+ details: () => [...userKeys.all, 'detail'] as const,
44
+ detail: (id: string) => [...userKeys.details(), id] as const,
45
+ };
46
+ ```
47
+
48
+ ## useQuery Hook Pattern
49
+
50
+ ```ts
51
+ // features/users/hooks/use-users.ts
52
+ import { useQuery } from '@tanstack/react-query';
53
+ import { userKeys } from './query-keys';
54
+ import { z } from 'zod';
55
+ import { userSchema } from '@/features/users/types/user.schemas';
56
+
57
+ const usersResponseSchema = z.array(userSchema);
58
+
59
+ export function useUsers() {
60
+ return useQuery({
61
+ queryKey: userKeys.lists(),
62
+ queryFn: async () => {
63
+ const res = await fetch('/api/proxy/users');
64
+ if (!res.ok) throw new Error('Failed to fetch users');
65
+ return usersResponseSchema.parse(await res.json());
66
+ },
67
+ });
68
+ }
69
+ ```
70
+
71
+ ## useMutation Hook Pattern
72
+
73
+ ```ts
74
+ // features/users/hooks/use-create-user.ts
75
+ import { useMutation, useQueryClient } from '@tanstack/react-query';
76
+ import { userKeys } from './query-keys';
77
+ import type { CreateUserInput } from '@/features/users/types/user.types';
78
+
79
+ export function useCreateUser() {
80
+ const queryClient = useQueryClient();
81
+ return useMutation({
82
+ mutationFn: async (input: CreateUserInput) => {
83
+ const res = await fetch('/api/proxy/users', {
84
+ method: 'POST',
85
+ headers: { 'Content-Type': 'application/json' },
86
+ body: JSON.stringify(input),
87
+ });
88
+ if (!res.ok) throw new Error('Failed to create user');
89
+ return res.json();
90
+ },
91
+ onSuccess: () => {
92
+ queryClient.invalidateQueries({ queryKey: userKeys.lists() });
93
+ },
94
+ });
95
+ }
96
+ ```
97
+
98
+ ## Server Component + TanStack Query Handoff
99
+
100
+ ```tsx
101
+ // app/(dashboard)/users/page.tsx — Server Component: initial fetch
102
+ export default async function UsersPage() {
103
+ const initialUsers = await fetch(`${process.env.API_URL}/api/users`).then(r => r.json());
104
+ return <UserListClient initialData={initialUsers} />;
105
+ }
106
+
107
+ // features/users/components/user-list-client.tsx — Client Component
108
+ 'use client';
109
+ export function UserListClient({ initialData }: { initialData: User[] }) {
110
+ const { data: users = initialData } = useUsers();
111
+ // mutations, filtering, etc.
112
+ }
113
+ ```
114
+
115
+ ## Common Mistakes
116
+
117
+ | Wrong | Right | Why |
118
+ |-------|-------|-----|
119
+ | `useEffect(() => { fetch(...) }, [])` | `useQuery(...)` | Two renders, no cache, no SSR |
120
+ | No Zod on API response | `schema.parse(await res.json())` | Type safety at runtime |
121
+ | Hardcoded query keys: `['users']` | Key factory: `userKeys.lists()` | Consistent invalidation |
122
+ | `queryClient.invalidateQueries(['users'])` | `{ queryKey: userKeys.lists() }` | Object syntax required in v5 |
123
+
124
+ ---
125
+
126
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,128 @@
1
+ # Next.js Forms Standard
2
+
3
+ > **Scope:** frontend/nextjs/forms
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** form, react-hook-form, zod, validation, input, submit, zodResolver
6
+ > **Load When:** implementing any form in Next.js
7
+
8
+ react-hook-form + Zod + shadcn Form components. Schema defines both validation rules and TypeScript types.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS define the Zod schema first — derive the TypeScript type from it with `z.infer<>`
13
+ - ALWAYS use `zodResolver` to connect schema to react-hook-form
14
+ - ALWAYS use shadcn `<Form>`, `<FormField>`, `<FormItem>`, `<FormMessage>` for consistent UI
15
+ - NEVER use `useState` for form field values — react-hook-form handles this
16
+ - ALWAYS use `useMutation` from TanStack Query for form submission
17
+
18
+ ## Complete Form Pattern
19
+
20
+ ```tsx
21
+ // features/users/types/user.schemas.ts
22
+ import { z } from 'zod';
23
+
24
+ export const createUserSchema = z.object({
25
+ name: z.string().min(2, 'Name must be at least 2 characters'),
26
+ email: z.string().email('Invalid email address'),
27
+ role: z.enum(['admin', 'user'], { required_error: 'Role is required' }),
28
+ });
29
+
30
+ export type CreateUserInput = z.infer<typeof createUserSchema>;
31
+ ```
32
+
33
+ ```tsx
34
+ // features/users/components/create-user-form.tsx
35
+ 'use client';
36
+
37
+ import { useForm } from 'react-hook-form';
38
+ import { zodResolver } from '@hookform/resolvers/zod';
39
+ import {
40
+ Form, FormControl, FormField, FormItem, FormLabel, FormMessage
41
+ } from '@/components/ui/form';
42
+ import { Input } from '@/components/ui/input';
43
+ import { Button } from '@/components/ui/button';
44
+ import { useCreateUser } from '@/features/users/hooks/use-create-user';
45
+ import { createUserSchema, type CreateUserInput } from '@/features/users/types/user.schemas';
46
+
47
+ export function CreateUserForm({ onSuccess }: { onSuccess?: () => void }) {
48
+ const form = useForm<CreateUserInput>({
49
+ resolver: zodResolver(createUserSchema),
50
+ defaultValues: { name: '', email: '', role: 'user' },
51
+ });
52
+
53
+ const { mutate: createUser, isPending } = useCreateUser();
54
+
55
+ function onSubmit(values: CreateUserInput) {
56
+ createUser(values, {
57
+ onSuccess: () => { form.reset(); onSuccess?.(); },
58
+ onError: (error) => form.setError('root', { message: error.message }),
59
+ });
60
+ }
61
+
62
+ return (
63
+ <Form {...form}>
64
+ <form onSubmit={form.handleSubmit(onSubmit)} className="space-y-4">
65
+ <FormField
66
+ control={form.control}
67
+ name="name"
68
+ render={({ field }) => (
69
+ <FormItem>
70
+ <FormLabel>Name</FormLabel>
71
+ <FormControl><Input placeholder="John Doe" {...field} /></FormControl>
72
+ <FormMessage />
73
+ </FormItem>
74
+ )}
75
+ />
76
+ <FormField
77
+ control={form.control}
78
+ name="email"
79
+ render={({ field }) => (
80
+ <FormItem>
81
+ <FormLabel>Email</FormLabel>
82
+ <FormControl><Input type="email" {...field} /></FormControl>
83
+ <FormMessage />
84
+ </FormItem>
85
+ )}
86
+ />
87
+ {form.formState.errors.root && (
88
+ <p className="text-sm text-destructive">{form.formState.errors.root.message}</p>
89
+ )}
90
+ <Button type="submit" disabled={isPending}>
91
+ {isPending ? 'Creating...' : 'Create User'}
92
+ </Button>
93
+ </form>
94
+ </Form>
95
+ );
96
+ }
97
+ ```
98
+
99
+ ## Edit Form Pattern
100
+
101
+ ```tsx
102
+ // Edit forms reuse the same schema with partial()
103
+ const form = useForm<UpdateUserInput>({
104
+ resolver: zodResolver(updateUserSchema),
105
+ defaultValues: { name: user.name, email: user.email },
106
+ });
107
+ ```
108
+
109
+ ## Schema Composition
110
+
111
+ ```ts
112
+ export const userSchema = z.object({ id: z.string(), name: z.string(), email: z.string() });
113
+ export const createUserSchema = userSchema.omit({ id: true });
114
+ export const updateUserSchema = createUserSchema.partial();
115
+ ```
116
+
117
+ ## Common Mistakes
118
+
119
+ | Wrong | Right | Why |
120
+ |-------|-------|-----|
121
+ | `useState` for each field | `useForm` | Unnecessary re-renders on every keystroke |
122
+ | Manual error display | `<FormMessage />` | Consistent UI, auto-connects to field errors |
123
+ | `fetch` in submit handler | `useMutation` | Loading state, error handling, cache invalidation |
124
+ | Type written manually | `z.infer<typeof schema>` | Single source of truth |
125
+
126
+ ---
127
+
128
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,67 @@
1
+ # Next.js Naming Conventions Standard
2
+
3
+ > **Scope:** frontend/nextjs/naming-conventions
4
+ > **Layer:** 1 (always)
5
+ > **Keywords:** naming, conventions, file names, component names, hooks, typescript
6
+ > **Load When:** creating any new file in a Next.js project
7
+
8
+ Consistent naming prevents filesystem bugs (Linux case-sensitivity) and keeps the codebase predictable.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS use kebab-case for file names — never PascalCase or camelCase
13
+ - ALWAYS use PascalCase for React component exports
14
+ - ALWAYS use camelCase starting with `use` for hook exports
15
+ - ALWAYS suffix schema files with `.schemas.ts` and type files with `.types.ts`
16
+ - NEVER mix cases in the same category — no `UserCard.tsx` alongside `user-profile.tsx`
17
+
18
+ ## Complete Reference Table
19
+
20
+ | Artifact | File Name | Export Name | Example |
21
+ |----------|-----------|-------------|---------|
22
+ | React Component | `user-card.tsx` | `UserCard` | `export function UserCard()` |
23
+ | Client Component | `user-form.tsx` | `UserForm` | `export function UserForm()` + `'use client'` |
24
+ | TanStack Query hook | `use-users.ts` | `useUsers` | `export function useUsers()` |
25
+ | Mutation hook | `use-create-user.ts` | `useCreateUser` | `export function useCreateUser()` |
26
+ | Utility hook | `use-debounce.ts` | `useDebounce` | `export function useDebounce()` |
27
+ | Zod schema file | `user.schemas.ts` | `createUserSchema` | `export const createUserSchema = z.object(...)` |
28
+ | TypeScript types | `user.types.ts` | `User`, `CreateUserInput` | `export type User = z.infer<typeof userSchema>` |
29
+ | Feature index | `index.ts` | (re-exports) | `export { UserCard } from './components/user-card'` |
30
+ | Page file | `page.tsx` | `default` | `export default function UsersPage()` |
31
+
32
+ ## Hook Naming Rules
33
+
34
+ ```
35
+ GET list: use-users.ts → useUsers()
36
+ GET single: use-user.ts → useUser(id: string)
37
+ POST: use-create-user.ts → useCreateUser()
38
+ PUT/PATCH: use-update-user.ts → useUpdateUser()
39
+ DELETE: use-delete-user.ts → useDeleteUser()
40
+ ```
41
+
42
+ ## Schema and Type Naming Rules
43
+
44
+ ```typescript
45
+ // user.schemas.ts
46
+ export const userSchema = z.object({ id: z.string(), name: z.string() });
47
+ export const createUserSchema = z.object({ name: z.string().min(2), email: z.string().email() });
48
+ export const updateUserSchema = createUserSchema.partial();
49
+
50
+ // user.types.ts — derive from schemas, don't duplicate
51
+ export type User = z.infer<typeof userSchema>;
52
+ export type CreateUserInput = z.infer<typeof createUserSchema>;
53
+ export type UpdateUserInput = z.infer<typeof updateUserSchema>;
54
+ ```
55
+
56
+ ## Common Mistakes
57
+
58
+ | Wrong | Right | Why |
59
+ |-------|-------|-----|
60
+ | `UserCard.tsx` | `user-card.tsx` | Linux servers are case-sensitive |
61
+ | `useUsers.ts` | `use-users.ts` | Inconsistent with Next.js file conventions |
62
+ | `export default function UserCard` | `export function UserCard` | Named exports are more refactor-safe |
63
+ | `type User = { id: string }` | `type User = z.infer<typeof userSchema>` | Single source of truth |
64
+
65
+ ---
66
+
67
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,102 @@
1
+ # Next.js Project Structure Standard
2
+
3
+ > **Scope:** frontend/nextjs/project-structure
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** project structure, folder structure, feature folder, src layout
6
+ > **Load When:** creating new files or features
7
+
8
+ Feature-based architecture with a shared core. Features are self-contained; shared code is explicit.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS use feature-based folders: `features/{feature-name}/`
13
+ - ALWAYS keep `app/` for routing only — import from `features/`, not the other way
14
+ - NEVER import from one feature into another — extract to `components/` or `lib/` if shared
15
+ - ALWAYS use `src/` directory (configured in `tsconfig.json` with `@/` alias)
16
+ - NEVER put business logic in `components/` — it is presentation only
17
+
18
+ ## Canonical Folder Tree
19
+
20
+ ```
21
+ src/
22
+ ├── app/ # Next.js App Router — routes only
23
+ │ ├── layout.tsx # Root layout + providers
24
+ │ ├── page.tsx # Home page
25
+ │ └── (dashboard)/
26
+ │ ├── layout.tsx
27
+ │ ├── users/
28
+ │ │ ├── page.tsx
29
+ │ │ ├── loading.tsx
30
+ │ │ └── error.tsx
31
+ │ └── billing/
32
+ │ └── page.tsx
33
+
34
+ ├── features/ # Domain features
35
+ │ └── {feature-name}/
36
+ │ ├── components/ # Feature-specific UI components
37
+ │ │ ├── user-list.tsx
38
+ │ │ └── user-card.tsx
39
+ │ ├── hooks/ # TanStack Query hooks for this feature
40
+ │ │ ├── use-users.ts
41
+ │ │ └── use-create-user.ts
42
+ │ ├── types/
43
+ │ │ ├── user.types.ts # TypeScript types
44
+ │ │ └── user.schemas.ts # Zod schemas + z.infer<> types
45
+ │ └── index.ts # Public API — only export what's needed
46
+
47
+ ├── components/ # Shared UI — no business logic
48
+ │ ├── ui/ # shadcn/ui — never edit directly
49
+ │ │ ├── button.tsx
50
+ │ │ └── card.tsx
51
+ │ ├── data-table.tsx
52
+ │ ├── page-header.tsx
53
+ │ └── empty-state.tsx
54
+
55
+ ├── hooks/ # Shared utility hooks — no API calls
56
+ │ ├── use-debounce.ts
57
+ │ └── use-media-query.ts
58
+
59
+ ├── lib/
60
+ │ ├── api-client.ts # Typed fetch wrapper for .NET API
61
+ │ └── query-client.tsx # TanStack Query client + provider
62
+
63
+ ├── types/
64
+ │ ├── api.ts # Shared API shapes: PaginatedResponse<T>, ApiError
65
+ │ └── env.d.ts
66
+
67
+ └── env.mjs # Zod-validated env vars
68
+ ```
69
+
70
+ ## Feature Index Pattern
71
+
72
+ ```ts
73
+ // features/users/index.ts — explicit public API
74
+ export { UserList } from './components/user-list';
75
+ export { UserCard } from './components/user-card';
76
+ export { useUsers } from './hooks/use-users';
77
+ export { useCreateUser } from './hooks/use-create-user';
78
+ export type { User, CreateUserInput } from './types/user.types';
79
+ ```
80
+
81
+ Import from the index, not deep paths:
82
+ ```ts
83
+ // GOOD
84
+ import { UserList, useUsers } from '@/features/users';
85
+
86
+ // BAD
87
+ import { UserList } from '@/features/users/components/user-list';
88
+ ```
89
+
90
+ ## Feature Boundary Rule
91
+
92
+ ```
93
+ app/ → imports from features/ ✓
94
+ features/users/ → imports from components/ and lib/ ✓
95
+ features/users/ → imports from features/billing/ ✗ (extract shared instead)
96
+ components/ → imports from components/ui/ ✓
97
+ components/ → imports from features/ ✗
98
+ ```
99
+
100
+ ---
101
+
102
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,72 @@
1
+ # Next.js State Management Standard
2
+
3
+ > **Scope:** frontend/nextjs/state-management
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** state, zustand, context, redux, client state, server state, useState
6
+ > **Load When:** deciding how to manage state in Next.js
7
+
8
+ Server state (API data) lives in TanStack Query. UI state lives in React. No global state library unless genuinely needed.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS use TanStack Query for anything that comes from the .NET API
13
+ - ALWAYS use `useState` for local UI state (modal open, selected tab, form step)
14
+ - NEVER install Zustand or Redux without first exhausting Server Components + TanStack Query
15
+ - NEVER use React Context for server state — Context does not cache or deduplicate
16
+ - ALWAYS derive state from server data rather than syncing it to local state
17
+
18
+ ## State Decision Tree
19
+
20
+ ```
21
+ Is this data from the .NET API?
22
+ YES → TanStack Query (useQuery / useMutation)
23
+
24
+ Is this local UI state (open/closed, selected, current step)?
25
+ YES → useState in the component that needs it
26
+
27
+ Does multiple components need this UI state?
28
+ YES — is it parent-child? → prop drilling (2-3 levels is fine)
29
+ YES — is it truly global? → React Context (theme, auth user, locale)
30
+ YES — complex with many actions? → Consider Zustand (last resort)
31
+ ```
32
+
33
+ ## Server State (TanStack Query)
34
+
35
+ ```ts
36
+ // GOOD — server state in TanStack Query
37
+ const { data: users } = useUsers(); // cached, deduplicated, background-refetched
38
+
39
+ // BAD — copying server state to useState
40
+ const [users, setUsers] = useState([]);
41
+ useEffect(() => { fetchUsers().then(setUsers); }, []);
42
+ ```
43
+
44
+ ## Local UI State (useState)
45
+
46
+ ```tsx
47
+ function UserList() {
48
+ const [isCreateDialogOpen, setIsCreateDialogOpen] = useState(false);
49
+ const [selectedUserId, setSelectedUserId] = useState<string | null>(null);
50
+ }
51
+ ```
52
+
53
+ ## When Zustand Is Justified
54
+
55
+ Only add Zustand if ALL of these are true:
56
+ 1. State is needed in 5+ unrelated components
57
+ 2. State has complex update logic (not just toggle/set)
58
+ 3. Context causes measurable re-render performance issues
59
+ 4. State is not from the server
60
+
61
+ ## Common Mistakes
62
+
63
+ | Wrong | Right | Why |
64
+ |-------|-------|-----|
65
+ | `useEffect` + `useState` for API data | TanStack Query | Stale data, no cache, double-render |
66
+ | Global store for user list | `useUsers()` | TanStack Query handles sync/cache |
67
+ | Context for everything | useState + prop drilling first | Context causes all consumers to re-render |
68
+ | Zustand by default | Start with useState | YAGNI — most state is local |
69
+
70
+ ---
71
+
72
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,111 @@
1
+ # Next.js Testing Standard
2
+
3
+ > **Scope:** frontend/nextjs/testing
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** testing, jest, vitest, testing-library, msw, component test, unit test
6
+ > **Load When:** writing tests for Next.js components or hooks
7
+
8
+ Jest + React Testing Library for component tests. MSW for API mocking. Co-locate tests with the code they test.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS co-locate test files: `user-card.test.tsx` next to `user-card.tsx`
13
+ - ALWAYS use `@testing-library/user-event` for interactions — not `fireEvent`
14
+ - ALWAYS mock the API layer with MSW — never mock `fetch` directly
15
+ - NEVER test implementation details — test what the user sees
16
+ - ALWAYS test the happy path + one error path per component
17
+
18
+ ## Test File Co-location
19
+
20
+ ```
21
+ features/users/
22
+ ├── components/
23
+ │ ├── user-card.tsx
24
+ │ ├── user-card.test.tsx ← co-located
25
+ │ ├── user-list.tsx
26
+ │ └── user-list.test.tsx
27
+ ├── hooks/
28
+ │ ├── use-users.ts
29
+ │ └── use-users.test.ts ← co-located
30
+ ```
31
+
32
+ ## Component Test Pattern
33
+
34
+ ```tsx
35
+ // features/users/components/user-card.test.tsx
36
+ import { render, screen } from '@testing-library/react';
37
+ import userEvent from '@testing-library/user-event';
38
+ import { UserCard } from './user-card';
39
+
40
+ const mockUser = { id: '1', name: 'João Silva', email: 'joao@example.com', role: 'user' as const };
41
+
42
+ describe('UserCard', () => {
43
+ it('renders user name and email', () => {
44
+ render(<UserCard user={mockUser} />);
45
+ expect(screen.getByText('João Silva')).toBeInTheDocument();
46
+ expect(screen.getByText('joao@example.com')).toBeInTheDocument();
47
+ });
48
+
49
+ it('calls onEdit with user id when edit button clicked', async () => {
50
+ const onEdit = vi.fn();
51
+ render(<UserCard user={mockUser} onEdit={onEdit} />);
52
+ await userEvent.click(screen.getByRole('button', { name: /edit/i }));
53
+ expect(onEdit).toHaveBeenCalledWith('1');
54
+ });
55
+ });
56
+ ```
57
+
58
+ ## Hook Test with MSW
59
+
60
+ ```ts
61
+ // features/users/hooks/use-users.test.ts
62
+ import { renderHook, waitFor } from '@testing-library/react';
63
+ import { http, HttpResponse } from 'msw';
64
+ import { server } from '@/test/msw-server';
65
+ import { QueryClientWrapper } from '@/test/helpers';
66
+ import { useUsers } from './use-users';
67
+
68
+ describe('useUsers', () => {
69
+ it('returns users from API', async () => {
70
+ server.use(
71
+ http.get('/api/proxy/users', () =>
72
+ HttpResponse.json([{ id: '1', name: 'João', email: 'j@ex.com' }])
73
+ )
74
+ );
75
+ const { result } = renderHook(() => useUsers(), { wrapper: QueryClientWrapper });
76
+ await waitFor(() => expect(result.current.isSuccess).toBe(true));
77
+ expect(result.current.data).toHaveLength(1);
78
+ });
79
+
80
+ it('returns error state when API fails', async () => {
81
+ server.use(http.get('/api/proxy/users', () => new HttpResponse(null, { status: 500 })));
82
+ const { result } = renderHook(() => useUsers(), { wrapper: QueryClientWrapper });
83
+ await waitFor(() => expect(result.current.isError).toBe(true));
84
+ });
85
+ });
86
+ ```
87
+
88
+ ## Test Helper Setup
89
+
90
+ ```tsx
91
+ // test/helpers.tsx
92
+ import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
93
+
94
+ export function QueryClientWrapper({ children }: { children: React.ReactNode }) {
95
+ const queryClient = new QueryClient({ defaultOptions: { queries: { retry: false } } });
96
+ return <QueryClientProvider client={queryClient}>{children}</QueryClientProvider>;
97
+ }
98
+ ```
99
+
100
+ ## Common Mistakes
101
+
102
+ | Wrong | Right | Why |
103
+ |-------|-------|-----|
104
+ | `fireEvent.click(button)` | `await userEvent.click(button)` | userEvent simulates real browser events |
105
+ | Mock `global.fetch` | Mock with MSW | MSW intercepts at network level |
106
+ | `expect(component).toMatchSnapshot()` | `expect(screen.getByText(...))` | Snapshots break on any change |
107
+ | Test file in `__tests__/` folder | Co-locate with source | Easier to find, deleted with the component |
108
+
109
+ ---
110
+
111
+ *MORPH-SPEC by Polymorphism Tech*