opencode-agent-kit 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (361) hide show
  1. package/README.md +796 -0
  2. package/bin/commands/init.mjs +221 -0
  3. package/bin/init.mjs +21 -0
  4. package/package.json +22 -0
  5. package/template/.opencode/agent-docs/backend/README.md +0 -0
  6. package/template/.opencode/agent-docs/backend/node/BACKEND_PATTERNS.md +82 -0
  7. package/template/.opencode/agent-docs/backend/node/BACKEND_QUICK_START.md +49 -0
  8. package/template/.opencode/agent-docs/frontend/next/README.md +0 -0
  9. package/template/.opencode/agent-docs/frontend/nuxt/API_PATTERNS.md +807 -0
  10. package/template/.opencode/agent-docs/frontend/nuxt/CHEATSHEET.md +676 -0
  11. package/template/.opencode/agent-docs/frontend/nuxt/COMPLETION_REPORT.md +613 -0
  12. package/template/.opencode/agent-docs/frontend/nuxt/EXAMPLES.md +956 -0
  13. package/template/.opencode/agent-docs/frontend/nuxt/INDEX.md +596 -0
  14. package/template/.opencode/agent-docs/frontend/nuxt/MCP_GUIDE.md +881 -0
  15. package/template/.opencode/agent-docs/frontend/nuxt/MENTOR_CURRICULUM_30_DAYS.md +256 -0
  16. package/template/.opencode/agent-docs/frontend/nuxt/MENTOR_CURRICULUM_CHECKLIST.md +156 -0
  17. package/template/.opencode/agent-docs/frontend/nuxt/MENTOR_WEEKLY_ASSIGNMENTS.md +191 -0
  18. package/template/.opencode/agent-docs/frontend/nuxt/QUICK_START.md +509 -0
  19. package/template/.opencode/agent-docs/frontend/nuxt/README.md +506 -0
  20. package/template/.opencode/agent-docs/frontend/nuxt/README_AGENTS.md +140 -0
  21. package/template/.opencode/agent-docs/frontend/nuxt/README_DOCS.md +65 -0
  22. package/template/.opencode/agent-docs/frontend/nuxt/SUMMARY.md +474 -0
  23. package/template/.opencode/agent-docs/frontend/nuxt/TEAM_OPERATING_GUIDE.md +54 -0
  24. package/template/.opencode/agent-docs/frontend/nuxt/TESTING_GUIDE.md +904 -0
  25. package/template/.opencode/agent-docs/frontend/nuxt/WORKFLOWS.md +758 -0
  26. package/template/.opencode/agent-docs/frontend/react/API_PATTERNS.md +187 -0
  27. package/template/.opencode/agent-docs/frontend/react/CHEATSHEET.md +87 -0
  28. package/template/.opencode/agent-docs/frontend/react/INDEX.md +45 -0
  29. package/template/.opencode/agent-docs/frontend/react/QUICK_START.md +43 -0
  30. package/template/.opencode/agent-docs/frontend/react/README.md +159 -0
  31. package/template/.opencode/agent-docs/frontend/vue/README.md +0 -0
  32. package/template/.opencode/agent-docs/mobile/android/README.md +45 -0
  33. package/template/.opencode/agent-docs/mobile/flutter/README.md +44 -0
  34. package/template/.opencode/agents/android-developer.md +418 -0
  35. package/template/.opencode/agents/code-igniter-3-fullstack.md +345 -0
  36. package/template/.opencode/agents/code-reviewer.md +517 -0
  37. package/template/.opencode/agents/database-specialist.md +455 -0
  38. package/template/.opencode/agents/devops-specialist.md +562 -0
  39. package/template/.opencode/agents/flutter-developer.md +556 -0
  40. package/template/.opencode/agents/it-leader.md +911 -0
  41. package/template/.opencode/agents/laravel-advanced.md +691 -0
  42. package/template/.opencode/agents/node-backend-developer.md +343 -0
  43. package/template/.opencode/agents/nuxt-frontend-developer-mentor.md +402 -0
  44. package/template/.opencode/agents/nuxt-frontend-developer.md +1573 -0
  45. package/template/.opencode/agents/react-frontend-developer.md +1017 -0
  46. package/template/.opencode/agents/seo-specialist.md +681 -0
  47. package/template/.opencode/agents/ui-ux-designer.md +783 -0
  48. package/template/.opencode/commands/android-build/command.md +25 -0
  49. package/template/.opencode/commands/android-test/command.md +23 -0
  50. package/template/.opencode/commands/build-fix.md +29 -0
  51. package/template/.opencode/commands/checkpoint.md +74 -0
  52. package/template/.opencode/commands/code-review.md +40 -0
  53. package/template/.opencode/commands/e2e.md +363 -0
  54. package/template/.opencode/commands/eval.md +120 -0
  55. package/template/.opencode/commands/evolve.md +193 -0
  56. package/template/.opencode/commands/flutter-build/command.md +25 -0
  57. package/template/.opencode/commands/flutter-test/command.md +24 -0
  58. package/template/.opencode/commands/go-build.md +183 -0
  59. package/template/.opencode/commands/go-review.md +148 -0
  60. package/template/.opencode/commands/go-test.md +268 -0
  61. package/template/.opencode/commands/gpc-release/command.md +30 -0
  62. package/template/.opencode/commands/instinct-export.md +91 -0
  63. package/template/.opencode/commands/instinct-import.md +142 -0
  64. package/template/.opencode/commands/instinct-status.md +86 -0
  65. package/template/.opencode/commands/learn.md +70 -0
  66. package/template/.opencode/commands/multi-backend.md +158 -0
  67. package/template/.opencode/commands/multi-execute.md +310 -0
  68. package/template/.opencode/commands/multi-frontend.md +158 -0
  69. package/template/.opencode/commands/multi-plan.md +261 -0
  70. package/template/.opencode/commands/multi-workflow.md +183 -0
  71. package/template/.opencode/commands/orchestrate.md +172 -0
  72. package/template/.opencode/commands/plan.md +113 -0
  73. package/template/.opencode/commands/pm2.md +271 -0
  74. package/template/.opencode/commands/python-review.md +297 -0
  75. package/template/.opencode/commands/refactor-clean.md +28 -0
  76. package/template/.opencode/commands/sessions.md +305 -0
  77. package/template/.opencode/commands/setup-pm.md +80 -0
  78. package/template/.opencode/commands/skill-create.md +174 -0
  79. package/template/.opencode/commands/tdd.md +326 -0
  80. package/template/.opencode/commands/test-coverage.md +27 -0
  81. package/template/.opencode/commands/update-codemaps.md +17 -0
  82. package/template/.opencode/commands/update-docs.md +31 -0
  83. package/template/.opencode/commands/verify.md +59 -0
  84. package/template/.opencode/config.example.json +309 -0
  85. package/template/.opencode/config.json +341 -0
  86. package/template/.opencode/contexts/dev.md +20 -0
  87. package/template/.opencode/contexts/research.md +26 -0
  88. package/template/.opencode/contexts/review.md +22 -0
  89. package/template/.opencode/hooks/hooks.json +169 -0
  90. package/template/.opencode/instructions/INSTRUCTIONS.md +388 -0
  91. package/template/.opencode/package.json +5 -0
  92. package/template/.opencode/rules/README.md +82 -0
  93. package/template/.opencode/rules/android/gradle.md +62 -0
  94. package/template/.opencode/rules/android/testing.md +27 -0
  95. package/template/.opencode/rules/common/agents.md +49 -0
  96. package/template/.opencode/rules/common/coding-style.md +48 -0
  97. package/template/.opencode/rules/common/git-workflow.md +45 -0
  98. package/template/.opencode/rules/common/hooks.md +30 -0
  99. package/template/.opencode/rules/common/patterns.md +31 -0
  100. package/template/.opencode/rules/common/performance.md +55 -0
  101. package/template/.opencode/rules/common/security.md +29 -0
  102. package/template/.opencode/rules/common/testing.md +29 -0
  103. package/template/.opencode/rules/flutter/state-management.md +57 -0
  104. package/template/.opencode/rules/flutter/testing.md +42 -0
  105. package/template/.opencode/rules/golang/coding-style.md +26 -0
  106. package/template/.opencode/rules/golang/hooks.md +11 -0
  107. package/template/.opencode/rules/golang/patterns.md +39 -0
  108. package/template/.opencode/rules/golang/security.md +28 -0
  109. package/template/.opencode/rules/golang/testing.md +25 -0
  110. package/template/.opencode/rules/mobile/performance.md +36 -0
  111. package/template/.opencode/rules/python/coding-style.md +37 -0
  112. package/template/.opencode/rules/python/hooks.md +14 -0
  113. package/template/.opencode/rules/python/patterns.md +34 -0
  114. package/template/.opencode/rules/python/security.md +25 -0
  115. package/template/.opencode/rules/python/testing.md +33 -0
  116. package/template/.opencode/rules/typescript/coding-style.md +58 -0
  117. package/template/.opencode/rules/typescript/hooks.md +15 -0
  118. package/template/.opencode/rules/typescript/patterns.md +45 -0
  119. package/template/.opencode/rules/typescript/security.md +21 -0
  120. package/template/.opencode/rules/typescript/testing.md +11 -0
  121. package/template/.opencode/skills/api-documentation/SKILL.md +188 -0
  122. package/template/.opencode/skills/backend-patterns/SKILL.md +587 -0
  123. package/template/.opencode/skills/building-components/SKILL.md +37 -0
  124. package/template/.opencode/skills/building-components/references/accessibility.mdx +819 -0
  125. package/template/.opencode/skills/building-components/references/as-child.mdx +324 -0
  126. package/template/.opencode/skills/building-components/references/composition.mdx +239 -0
  127. package/template/.opencode/skills/building-components/references/data-attributes.mdx +413 -0
  128. package/template/.opencode/skills/building-components/references/definitions.mdx +258 -0
  129. package/template/.opencode/skills/building-components/references/design-tokens.mdx +57 -0
  130. package/template/.opencode/skills/building-components/references/docs.mdx +155 -0
  131. package/template/.opencode/skills/building-components/references/marketplaces.mdx +144 -0
  132. package/template/.opencode/skills/building-components/references/npm.mdx +166 -0
  133. package/template/.opencode/skills/building-components/references/polymorphism.mdx +583 -0
  134. package/template/.opencode/skills/building-components/references/principles.mdx +61 -0
  135. package/template/.opencode/skills/building-components/references/registry.mdx +169 -0
  136. package/template/.opencode/skills/building-components/references/state.mdx +99 -0
  137. package/template/.opencode/skills/building-components/references/styling.mdx +286 -0
  138. package/template/.opencode/skills/building-components/references/types.mdx +191 -0
  139. package/template/.opencode/skills/clickhouse-io/SKILL.md +429 -0
  140. package/template/.opencode/skills/coding-standards/SKILL.md +520 -0
  141. package/template/.opencode/skills/configure-ecc/SKILL.md +298 -0
  142. package/template/.opencode/skills/continuous-learning/SKILL.md +110 -0
  143. package/template/.opencode/skills/continuous-learning/config.json +18 -0
  144. package/template/.opencode/skills/continuous-learning/evaluate-session.sh +60 -0
  145. package/template/.opencode/skills/continuous-learning-v2/SKILL.md +284 -0
  146. package/template/.opencode/skills/continuous-learning-v2/agents/observer.md +137 -0
  147. package/template/.opencode/skills/continuous-learning-v2/agents/start-observer.sh +134 -0
  148. package/template/.opencode/skills/continuous-learning-v2/config.json +41 -0
  149. package/template/.opencode/skills/continuous-learning-v2/hooks/observe.sh +153 -0
  150. package/template/.opencode/skills/continuous-learning-v2/scripts/instinct-cli.py +489 -0
  151. package/template/.opencode/skills/continuous-learning-v2/scripts/test_parse_instinct.py +82 -0
  152. package/template/.opencode/skills/dart-add-unit-test/SKILL.md +122 -0
  153. package/template/.opencode/skills/dart-build-cli-app/SKILL.md +185 -0
  154. package/template/.opencode/skills/dart-collect-coverage/SKILL.md +141 -0
  155. package/template/.opencode/skills/dart-fix-runtime-errors/SKILL.md +166 -0
  156. package/template/.opencode/skills/dart-generate-test-mocks/SKILL.md +155 -0
  157. package/template/.opencode/skills/dart-migrate-to-checks-package/SKILL.md +126 -0
  158. package/template/.opencode/skills/dart-resolve-package-conflicts/SKILL.md +116 -0
  159. package/template/.opencode/skills/dart-run-static-analysis/SKILL.md +104 -0
  160. package/template/.opencode/skills/dart-use-pattern-matching/SKILL.md +146 -0
  161. package/template/.opencode/skills/django-patterns/SKILL.md +733 -0
  162. package/template/.opencode/skills/django-security/SKILL.md +592 -0
  163. package/template/.opencode/skills/django-tdd/SKILL.md +728 -0
  164. package/template/.opencode/skills/django-verification/SKILL.md +460 -0
  165. package/template/.opencode/skills/eval-harness/SKILL.md +227 -0
  166. package/template/.opencode/skills/firebase-basics/SKILL.md +103 -0
  167. package/template/.opencode/skills/firebase-basics/references/additional-skills.md +113 -0
  168. package/template/.opencode/skills/firebase-basics/references/cli-usage.md +31 -0
  169. package/template/.opencode/skills/firebase-basics/references/client-library-usage.md +45 -0
  170. package/template/.opencode/skills/firebase-basics/references/core-concepts.md +61 -0
  171. package/template/.opencode/skills/firebase-basics/references/iac-usage.md +40 -0
  172. package/template/.opencode/skills/firebase-basics/references/iam-security.md +74 -0
  173. package/template/.opencode/skills/firebase-basics/references/mcp-usage.md +63 -0
  174. package/template/.opencode/skills/flutter/SKILL.md +292 -0
  175. package/template/.opencode/skills/flutter-add-integration-test/SKILL.md +163 -0
  176. package/template/.opencode/skills/flutter-add-widget-preview/SKILL.md +145 -0
  177. package/template/.opencode/skills/flutter-add-widget-test/SKILL.md +154 -0
  178. package/template/.opencode/skills/flutter-apply-architecture-best-practices/SKILL.md +162 -0
  179. package/template/.opencode/skills/flutter-build-responsive-layout/SKILL.md +139 -0
  180. package/template/.opencode/skills/flutter-fix-layout-issues/SKILL.md +130 -0
  181. package/template/.opencode/skills/flutter-implement-json-serialization/SKILL.md +153 -0
  182. package/template/.opencode/skills/flutter-setup-declarative-routing/SKILL.md +255 -0
  183. package/template/.opencode/skills/flutter-setup-localization/SKILL.md +210 -0
  184. package/template/.opencode/skills/flutter-use-http-package/SKILL.md +174 -0
  185. package/template/.opencode/skills/frontend-design/SKILL.md +89 -0
  186. package/template/.opencode/skills/frontend-patterns/SKILL.md +631 -0
  187. package/template/.opencode/skills/golang-patterns/SKILL.md +673 -0
  188. package/template/.opencode/skills/golang-testing/SKILL.md +719 -0
  189. package/template/.opencode/skills/impeccable/SKILL.md +165 -0
  190. package/template/.opencode/skills/impeccable/agents/impeccable-asset-producer.md +101 -0
  191. package/template/.opencode/skills/impeccable/reference/adapt.md +190 -0
  192. package/template/.opencode/skills/impeccable/reference/animate.md +175 -0
  193. package/template/.opencode/skills/impeccable/reference/audit.md +133 -0
  194. package/template/.opencode/skills/impeccable/reference/bolder.md +113 -0
  195. package/template/.opencode/skills/impeccable/reference/brand.md +118 -0
  196. package/template/.opencode/skills/impeccable/reference/clarify.md +174 -0
  197. package/template/.opencode/skills/impeccable/reference/codex.md +105 -0
  198. package/template/.opencode/skills/impeccable/reference/cognitive-load.md +106 -0
  199. package/template/.opencode/skills/impeccable/reference/color-and-contrast.md +105 -0
  200. package/template/.opencode/skills/impeccable/reference/colorize.md +154 -0
  201. package/template/.opencode/skills/impeccable/reference/craft.md +123 -0
  202. package/template/.opencode/skills/impeccable/reference/critique.md +273 -0
  203. package/template/.opencode/skills/impeccable/reference/delight.md +302 -0
  204. package/template/.opencode/skills/impeccable/reference/distill.md +111 -0
  205. package/template/.opencode/skills/impeccable/reference/document.md +427 -0
  206. package/template/.opencode/skills/impeccable/reference/extract.md +69 -0
  207. package/template/.opencode/skills/impeccable/reference/harden.md +347 -0
  208. package/template/.opencode/skills/impeccable/reference/heuristics-scoring.md +234 -0
  209. package/template/.opencode/skills/impeccable/reference/interaction-design.md +195 -0
  210. package/template/.opencode/skills/impeccable/reference/layout.md +141 -0
  211. package/template/.opencode/skills/impeccable/reference/live.md +622 -0
  212. package/template/.opencode/skills/impeccable/reference/motion-design.md +109 -0
  213. package/template/.opencode/skills/impeccable/reference/onboard.md +234 -0
  214. package/template/.opencode/skills/impeccable/reference/optimize.md +258 -0
  215. package/template/.opencode/skills/impeccable/reference/overdrive.md +130 -0
  216. package/template/.opencode/skills/impeccable/reference/personas.md +179 -0
  217. package/template/.opencode/skills/impeccable/reference/polish.md +242 -0
  218. package/template/.opencode/skills/impeccable/reference/product.md +62 -0
  219. package/template/.opencode/skills/impeccable/reference/quieter.md +99 -0
  220. package/template/.opencode/skills/impeccable/reference/responsive-design.md +114 -0
  221. package/template/.opencode/skills/impeccable/reference/shape.md +165 -0
  222. package/template/.opencode/skills/impeccable/reference/spatial-design.md +100 -0
  223. package/template/.opencode/skills/impeccable/reference/teach.md +156 -0
  224. package/template/.opencode/skills/impeccable/reference/typeset.md +124 -0
  225. package/template/.opencode/skills/impeccable/reference/typography.md +159 -0
  226. package/template/.opencode/skills/impeccable/reference/ux-writing.md +107 -0
  227. package/template/.opencode/skills/impeccable/scripts/cleanup-deprecated.mjs +284 -0
  228. package/template/.opencode/skills/impeccable/scripts/command-metadata.json +94 -0
  229. package/template/.opencode/skills/impeccable/scripts/critique-storage.mjs +242 -0
  230. package/template/.opencode/skills/impeccable/scripts/design-parser.mjs +820 -0
  231. package/template/.opencode/skills/impeccable/scripts/detect-csp.mjs +198 -0
  232. package/template/.opencode/skills/impeccable/scripts/detect.mjs +21 -0
  233. package/template/.opencode/skills/impeccable/scripts/impeccable-paths.mjs +110 -0
  234. package/template/.opencode/skills/impeccable/scripts/is-generated.mjs +69 -0
  235. package/template/.opencode/skills/impeccable/scripts/live-accept.mjs +595 -0
  236. package/template/.opencode/skills/impeccable/scripts/live-browser-session.js +123 -0
  237. package/template/.opencode/skills/impeccable/scripts/live-browser.js +4860 -0
  238. package/template/.opencode/skills/impeccable/scripts/live-complete.mjs +75 -0
  239. package/template/.opencode/skills/impeccable/scripts/live-completion.mjs +18 -0
  240. package/template/.opencode/skills/impeccable/scripts/live-inject.mjs +446 -0
  241. package/template/.opencode/skills/impeccable/scripts/live-poll.mjs +200 -0
  242. package/template/.opencode/skills/impeccable/scripts/live-resume.mjs +48 -0
  243. package/template/.opencode/skills/impeccable/scripts/live-server.mjs +838 -0
  244. package/template/.opencode/skills/impeccable/scripts/live-session-store.mjs +254 -0
  245. package/template/.opencode/skills/impeccable/scripts/live-status.mjs +47 -0
  246. package/template/.opencode/skills/impeccable/scripts/live-wrap.mjs +632 -0
  247. package/template/.opencode/skills/impeccable/scripts/live.mjs +247 -0
  248. package/template/.opencode/skills/impeccable/scripts/load-context.mjs +141 -0
  249. package/template/.opencode/skills/impeccable/scripts/modern-screenshot.umd.js +14 -0
  250. package/template/.opencode/skills/impeccable/scripts/pin.mjs +214 -0
  251. package/template/.opencode/skills/iterative-retrieval/SKILL.md +202 -0
  252. package/template/.opencode/skills/java-coding-standards/SKILL.md +138 -0
  253. package/template/.opencode/skills/jetpack-compose/.skillfish.json +10 -0
  254. package/template/.opencode/skills/jetpack-compose/SKILL.md +420 -0
  255. package/template/.opencode/skills/jpa-patterns/SKILL.md +141 -0
  256. package/template/.opencode/skills/nutrient-document-processing/SKILL.md +165 -0
  257. package/template/.opencode/skills/nuxt-ui/SKILL.md +334 -0
  258. package/template/.opencode/skills/nuxt-ui/references/components.md +377 -0
  259. package/template/.opencode/skills/nuxt-ui/references/composables.md +127 -0
  260. package/template/.opencode/skills/nuxt-ui/references/layouts/chat.md +266 -0
  261. package/template/.opencode/skills/nuxt-ui/references/layouts/dashboard.md +220 -0
  262. package/template/.opencode/skills/nuxt-ui/references/layouts/docs.md +141 -0
  263. package/template/.opencode/skills/nuxt-ui/references/layouts/editor.md +168 -0
  264. package/template/.opencode/skills/nuxt-ui/references/layouts/page.md +260 -0
  265. package/template/.opencode/skills/nuxt-ui/references/theming.md +427 -0
  266. package/template/.opencode/skills/postgres-patterns/SKILL.md +146 -0
  267. package/template/.opencode/skills/project-guidelines-example/SKILL.md +345 -0
  268. package/template/.opencode/skills/python-patterns/SKILL.md +749 -0
  269. package/template/.opencode/skills/python-testing/SKILL.md +815 -0
  270. package/template/.opencode/skills/security-review/SKILL.md +494 -0
  271. package/template/.opencode/skills/security-review/cloud-infrastructure-security.md +361 -0
  272. package/template/.opencode/skills/shadcn-ui/README.md +248 -0
  273. package/template/.opencode/skills/shadcn-ui/SKILL.md +326 -0
  274. package/template/.opencode/skills/shadcn-ui/examples/auth-layout.tsx +177 -0
  275. package/template/.opencode/skills/shadcn-ui/examples/data-table.tsx +313 -0
  276. package/template/.opencode/skills/shadcn-ui/examples/form-pattern.tsx +177 -0
  277. package/template/.opencode/skills/shadcn-ui/resources/component-catalog.md +481 -0
  278. package/template/.opencode/skills/shadcn-ui/resources/customization-guide.md +516 -0
  279. package/template/.opencode/skills/shadcn-ui/resources/migration-guide.md +463 -0
  280. package/template/.opencode/skills/shadcn-ui/resources/setup-guide.md +412 -0
  281. package/template/.opencode/skills/shadcn-ui/scripts/verify-setup.sh +134 -0
  282. package/template/.opencode/skills/springboot-patterns/SKILL.md +304 -0
  283. package/template/.opencode/skills/springboot-security/SKILL.md +119 -0
  284. package/template/.opencode/skills/springboot-tdd/SKILL.md +157 -0
  285. package/template/.opencode/skills/springboot-verification/SKILL.md +100 -0
  286. package/template/.opencode/skills/strategic-compact/SKILL.md +63 -0
  287. package/template/.opencode/skills/strategic-compact/suggest-compact.sh +52 -0
  288. package/template/.opencode/skills/tdd-workflow/SKILL.md +409 -0
  289. package/template/.opencode/skills/vercel-composition-patterns/AGENTS.md +946 -0
  290. package/template/.opencode/skills/vercel-composition-patterns/SKILL.md +89 -0
  291. package/template/.opencode/skills/vercel-composition-patterns/rules/architecture-avoid-boolean-props.md +100 -0
  292. package/template/.opencode/skills/vercel-composition-patterns/rules/architecture-compound-components.md +112 -0
  293. package/template/.opencode/skills/vercel-composition-patterns/rules/patterns-children-over-render-props.md +87 -0
  294. package/template/.opencode/skills/vercel-composition-patterns/rules/patterns-explicit-variants.md +100 -0
  295. package/template/.opencode/skills/vercel-composition-patterns/rules/react19-no-forwardref.md +42 -0
  296. package/template/.opencode/skills/vercel-composition-patterns/rules/state-context-interface.md +191 -0
  297. package/template/.opencode/skills/vercel-composition-patterns/rules/state-decouple-implementation.md +113 -0
  298. package/template/.opencode/skills/vercel-composition-patterns/rules/state-lift-state.md +125 -0
  299. package/template/.opencode/skills/vercel-react-best-practices/AGENTS.md +2934 -0
  300. package/template/.opencode/skills/vercel-react-best-practices/SKILL.md +136 -0
  301. package/template/.opencode/skills/vercel-react-best-practices/rules/advanced-event-handler-refs.md +55 -0
  302. package/template/.opencode/skills/vercel-react-best-practices/rules/advanced-init-once.md +42 -0
  303. package/template/.opencode/skills/vercel-react-best-practices/rules/advanced-use-latest.md +39 -0
  304. package/template/.opencode/skills/vercel-react-best-practices/rules/async-api-routes.md +38 -0
  305. package/template/.opencode/skills/vercel-react-best-practices/rules/async-defer-await.md +80 -0
  306. package/template/.opencode/skills/vercel-react-best-practices/rules/async-dependencies.md +51 -0
  307. package/template/.opencode/skills/vercel-react-best-practices/rules/async-parallel.md +28 -0
  308. package/template/.opencode/skills/vercel-react-best-practices/rules/async-suspense-boundaries.md +99 -0
  309. package/template/.opencode/skills/vercel-react-best-practices/rules/bundle-barrel-imports.md +59 -0
  310. package/template/.opencode/skills/vercel-react-best-practices/rules/bundle-conditional.md +31 -0
  311. package/template/.opencode/skills/vercel-react-best-practices/rules/bundle-defer-third-party.md +49 -0
  312. package/template/.opencode/skills/vercel-react-best-practices/rules/bundle-dynamic-imports.md +35 -0
  313. package/template/.opencode/skills/vercel-react-best-practices/rules/bundle-preload.md +50 -0
  314. package/template/.opencode/skills/vercel-react-best-practices/rules/client-event-listeners.md +74 -0
  315. package/template/.opencode/skills/vercel-react-best-practices/rules/client-localstorage-schema.md +71 -0
  316. package/template/.opencode/skills/vercel-react-best-practices/rules/client-passive-event-listeners.md +48 -0
  317. package/template/.opencode/skills/vercel-react-best-practices/rules/client-swr-dedup.md +56 -0
  318. package/template/.opencode/skills/vercel-react-best-practices/rules/js-batch-dom-css.md +107 -0
  319. package/template/.opencode/skills/vercel-react-best-practices/rules/js-cache-function-results.md +80 -0
  320. package/template/.opencode/skills/vercel-react-best-practices/rules/js-cache-property-access.md +28 -0
  321. package/template/.opencode/skills/vercel-react-best-practices/rules/js-cache-storage.md +70 -0
  322. package/template/.opencode/skills/vercel-react-best-practices/rules/js-combine-iterations.md +32 -0
  323. package/template/.opencode/skills/vercel-react-best-practices/rules/js-early-exit.md +50 -0
  324. package/template/.opencode/skills/vercel-react-best-practices/rules/js-hoist-regexp.md +45 -0
  325. package/template/.opencode/skills/vercel-react-best-practices/rules/js-index-maps.md +37 -0
  326. package/template/.opencode/skills/vercel-react-best-practices/rules/js-length-check-first.md +49 -0
  327. package/template/.opencode/skills/vercel-react-best-practices/rules/js-min-max-loop.md +82 -0
  328. package/template/.opencode/skills/vercel-react-best-practices/rules/js-set-map-lookups.md +24 -0
  329. package/template/.opencode/skills/vercel-react-best-practices/rules/js-tosorted-immutable.md +57 -0
  330. package/template/.opencode/skills/vercel-react-best-practices/rules/rendering-activity.md +26 -0
  331. package/template/.opencode/skills/vercel-react-best-practices/rules/rendering-animate-svg-wrapper.md +47 -0
  332. package/template/.opencode/skills/vercel-react-best-practices/rules/rendering-conditional-render.md +40 -0
  333. package/template/.opencode/skills/vercel-react-best-practices/rules/rendering-content-visibility.md +38 -0
  334. package/template/.opencode/skills/vercel-react-best-practices/rules/rendering-hoist-jsx.md +46 -0
  335. package/template/.opencode/skills/vercel-react-best-practices/rules/rendering-hydration-no-flicker.md +82 -0
  336. package/template/.opencode/skills/vercel-react-best-practices/rules/rendering-hydration-suppress-warning.md +30 -0
  337. package/template/.opencode/skills/vercel-react-best-practices/rules/rendering-svg-precision.md +28 -0
  338. package/template/.opencode/skills/vercel-react-best-practices/rules/rendering-usetransition-loading.md +75 -0
  339. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-defer-reads.md +39 -0
  340. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-dependencies.md +45 -0
  341. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-derived-state-no-effect.md +40 -0
  342. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-derived-state.md +29 -0
  343. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-functional-setstate.md +74 -0
  344. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-lazy-state-init.md +58 -0
  345. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-memo-with-default-value.md +38 -0
  346. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-memo.md +44 -0
  347. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-move-effect-to-event.md +45 -0
  348. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-simple-expression-in-memo.md +35 -0
  349. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-transitions.md +40 -0
  350. package/template/.opencode/skills/vercel-react-best-practices/rules/rerender-use-ref-transient-values.md +73 -0
  351. package/template/.opencode/skills/vercel-react-best-practices/rules/server-after-nonblocking.md +73 -0
  352. package/template/.opencode/skills/vercel-react-best-practices/rules/server-auth-actions.md +96 -0
  353. package/template/.opencode/skills/vercel-react-best-practices/rules/server-cache-lru.md +41 -0
  354. package/template/.opencode/skills/vercel-react-best-practices/rules/server-cache-react.md +76 -0
  355. package/template/.opencode/skills/vercel-react-best-practices/rules/server-dedup-props.md +65 -0
  356. package/template/.opencode/skills/vercel-react-best-practices/rules/server-parallel-fetching.md +83 -0
  357. package/template/.opencode/skills/vercel-react-best-practices/rules/server-serialization.md +38 -0
  358. package/template/.opencode/skills/verification-loop/SKILL.md +120 -0
  359. package/template/.opencode/skills/web-design-guidelines/SKILL.md +39 -0
  360. package/template/AGENTS.md +32 -0
  361. package/template/opencode.json +354 -0
@@ -0,0 +1,631 @@
1
+ ---
2
+ name: frontend-patterns
3
+ description: Frontend development patterns for React, Next.js, state management, performance optimization, and UI best practices.
4
+ ---
5
+
6
+ # Frontend Development Patterns
7
+
8
+ Modern frontend patterns for React, Next.js, and performant user interfaces.
9
+
10
+ ## Component Patterns
11
+
12
+ ### Composition Over Inheritance
13
+
14
+ ```typescript
15
+ // ✅ GOOD: Component composition
16
+ interface CardProps {
17
+ children: React.ReactNode
18
+ variant?: 'default' | 'outlined'
19
+ }
20
+
21
+ export function Card({ children, variant = 'default' }: CardProps) {
22
+ return <div className={`card card-${variant}`}>{children}</div>
23
+ }
24
+
25
+ export function CardHeader({ children }: { children: React.ReactNode }) {
26
+ return <div className="card-header">{children}</div>
27
+ }
28
+
29
+ export function CardBody({ children }: { children: React.ReactNode }) {
30
+ return <div className="card-body">{children}</div>
31
+ }
32
+
33
+ // Usage
34
+ <Card>
35
+ <CardHeader>Title</CardHeader>
36
+ <CardBody>Content</CardBody>
37
+ </Card>
38
+ ```
39
+
40
+ ### Compound Components
41
+
42
+ ```typescript
43
+ interface TabsContextValue {
44
+ activeTab: string
45
+ setActiveTab: (tab: string) => void
46
+ }
47
+
48
+ const TabsContext = createContext<TabsContextValue | undefined>(undefined)
49
+
50
+ export function Tabs({ children, defaultTab }: {
51
+ children: React.ReactNode
52
+ defaultTab: string
53
+ }) {
54
+ const [activeTab, setActiveTab] = useState(defaultTab)
55
+
56
+ return (
57
+ <TabsContext.Provider value={{ activeTab, setActiveTab }}>
58
+ {children}
59
+ </TabsContext.Provider>
60
+ )
61
+ }
62
+
63
+ export function TabList({ children }: { children: React.ReactNode }) {
64
+ return <div className="tab-list">{children}</div>
65
+ }
66
+
67
+ export function Tab({ id, children }: { id: string, children: React.ReactNode }) {
68
+ const context = useContext(TabsContext)
69
+ if (!context) throw new Error('Tab must be used within Tabs')
70
+
71
+ return (
72
+ <button
73
+ className={context.activeTab === id ? 'active' : ''}
74
+ onClick={() => context.setActiveTab(id)}
75
+ >
76
+ {children}
77
+ </button>
78
+ )
79
+ }
80
+
81
+ // Usage
82
+ <Tabs defaultTab="overview">
83
+ <TabList>
84
+ <Tab id="overview">Overview</Tab>
85
+ <Tab id="details">Details</Tab>
86
+ </TabList>
87
+ </Tabs>
88
+ ```
89
+
90
+ ### Render Props Pattern
91
+
92
+ ```typescript
93
+ interface DataLoaderProps<T> {
94
+ url: string
95
+ children: (data: T | null, loading: boolean, error: Error | null) => React.ReactNode
96
+ }
97
+
98
+ export function DataLoader<T>({ url, children }: DataLoaderProps<T>) {
99
+ const [data, setData] = useState<T | null>(null)
100
+ const [loading, setLoading] = useState(true)
101
+ const [error, setError] = useState<Error | null>(null)
102
+
103
+ useEffect(() => {
104
+ fetch(url)
105
+ .then(res => res.json())
106
+ .then(setData)
107
+ .catch(setError)
108
+ .finally(() => setLoading(false))
109
+ }, [url])
110
+
111
+ return <>{children(data, loading, error)}</>
112
+ }
113
+
114
+ // Usage
115
+ <DataLoader<Market[]> url="/api/markets">
116
+ {(markets, loading, error) => {
117
+ if (loading) return <Spinner />
118
+ if (error) return <Error error={error} />
119
+ return <MarketList markets={markets!} />
120
+ }}
121
+ </DataLoader>
122
+ ```
123
+
124
+ ## Custom Hooks Patterns
125
+
126
+ ### State Management Hook
127
+
128
+ ```typescript
129
+ export function useToggle(initialValue = false): [boolean, () => void] {
130
+ const [value, setValue] = useState(initialValue)
131
+
132
+ const toggle = useCallback(() => {
133
+ setValue(v => !v)
134
+ }, [])
135
+
136
+ return [value, toggle]
137
+ }
138
+
139
+ // Usage
140
+ const [isOpen, toggleOpen] = useToggle()
141
+ ```
142
+
143
+ ### Async Data Fetching Hook
144
+
145
+ ```typescript
146
+ interface UseQueryOptions<T> {
147
+ onSuccess?: (data: T) => void
148
+ onError?: (error: Error) => void
149
+ enabled?: boolean
150
+ }
151
+
152
+ export function useQuery<T>(
153
+ key: string,
154
+ fetcher: () => Promise<T>,
155
+ options?: UseQueryOptions<T>
156
+ ) {
157
+ const [data, setData] = useState<T | null>(null)
158
+ const [error, setError] = useState<Error | null>(null)
159
+ const [loading, setLoading] = useState(false)
160
+
161
+ const refetch = useCallback(async () => {
162
+ setLoading(true)
163
+ setError(null)
164
+
165
+ try {
166
+ const result = await fetcher()
167
+ setData(result)
168
+ options?.onSuccess?.(result)
169
+ } catch (err) {
170
+ const error = err as Error
171
+ setError(error)
172
+ options?.onError?.(error)
173
+ } finally {
174
+ setLoading(false)
175
+ }
176
+ }, [fetcher, options])
177
+
178
+ useEffect(() => {
179
+ if (options?.enabled !== false) {
180
+ refetch()
181
+ }
182
+ }, [key, refetch, options?.enabled])
183
+
184
+ return { data, error, loading, refetch }
185
+ }
186
+
187
+ // Usage
188
+ const { data: markets, loading, error, refetch } = useQuery(
189
+ 'markets',
190
+ () => fetch('/api/markets').then(r => r.json()),
191
+ {
192
+ onSuccess: data => console.log('Fetched', data.length, 'markets'),
193
+ onError: err => console.error('Failed:', err)
194
+ }
195
+ )
196
+ ```
197
+
198
+ ### Debounce Hook
199
+
200
+ ```typescript
201
+ export function useDebounce<T>(value: T, delay: number): T {
202
+ const [debouncedValue, setDebouncedValue] = useState<T>(value)
203
+
204
+ useEffect(() => {
205
+ const handler = setTimeout(() => {
206
+ setDebouncedValue(value)
207
+ }, delay)
208
+
209
+ return () => clearTimeout(handler)
210
+ }, [value, delay])
211
+
212
+ return debouncedValue
213
+ }
214
+
215
+ // Usage
216
+ const [searchQuery, setSearchQuery] = useState('')
217
+ const debouncedQuery = useDebounce(searchQuery, 500)
218
+
219
+ useEffect(() => {
220
+ if (debouncedQuery) {
221
+ performSearch(debouncedQuery)
222
+ }
223
+ }, [debouncedQuery])
224
+ ```
225
+
226
+ ## State Management Patterns
227
+
228
+ ### Context + Reducer Pattern
229
+
230
+ ```typescript
231
+ interface State {
232
+ markets: Market[]
233
+ selectedMarket: Market | null
234
+ loading: boolean
235
+ }
236
+
237
+ type Action =
238
+ | { type: 'SET_MARKETS'; payload: Market[] }
239
+ | { type: 'SELECT_MARKET'; payload: Market }
240
+ | { type: 'SET_LOADING'; payload: boolean }
241
+
242
+ function reducer(state: State, action: Action): State {
243
+ switch (action.type) {
244
+ case 'SET_MARKETS':
245
+ return { ...state, markets: action.payload }
246
+ case 'SELECT_MARKET':
247
+ return { ...state, selectedMarket: action.payload }
248
+ case 'SET_LOADING':
249
+ return { ...state, loading: action.payload }
250
+ default:
251
+ return state
252
+ }
253
+ }
254
+
255
+ const MarketContext = createContext<{
256
+ state: State
257
+ dispatch: Dispatch<Action>
258
+ } | undefined>(undefined)
259
+
260
+ export function MarketProvider({ children }: { children: React.ReactNode }) {
261
+ const [state, dispatch] = useReducer(reducer, {
262
+ markets: [],
263
+ selectedMarket: null,
264
+ loading: false
265
+ })
266
+
267
+ return (
268
+ <MarketContext.Provider value={{ state, dispatch }}>
269
+ {children}
270
+ </MarketContext.Provider>
271
+ )
272
+ }
273
+
274
+ export function useMarkets() {
275
+ const context = useContext(MarketContext)
276
+ if (!context) throw new Error('useMarkets must be used within MarketProvider')
277
+ return context
278
+ }
279
+ ```
280
+
281
+ ## Performance Optimization
282
+
283
+ ### Memoization
284
+
285
+ ```typescript
286
+ // ✅ useMemo for expensive computations
287
+ const sortedMarkets = useMemo(() => {
288
+ return markets.sort((a, b) => b.volume - a.volume)
289
+ }, [markets])
290
+
291
+ // ✅ useCallback for functions passed to children
292
+ const handleSearch = useCallback((query: string) => {
293
+ setSearchQuery(query)
294
+ }, [])
295
+
296
+ // ✅ React.memo for pure components
297
+ export const MarketCard = React.memo<MarketCardProps>(({ market }) => {
298
+ return (
299
+ <div className="market-card">
300
+ <h3>{market.name}</h3>
301
+ <p>{market.description}</p>
302
+ </div>
303
+ )
304
+ })
305
+ ```
306
+
307
+ ### Code Splitting & Lazy Loading
308
+
309
+ ```typescript
310
+ import { lazy, Suspense } from 'react'
311
+
312
+ // ✅ Lazy load heavy components
313
+ const HeavyChart = lazy(() => import('./HeavyChart'))
314
+ const ThreeJsBackground = lazy(() => import('./ThreeJsBackground'))
315
+
316
+ export function Dashboard() {
317
+ return (
318
+ <div>
319
+ <Suspense fallback={<ChartSkeleton />}>
320
+ <HeavyChart data={data} />
321
+ </Suspense>
322
+
323
+ <Suspense fallback={null}>
324
+ <ThreeJsBackground />
325
+ </Suspense>
326
+ </div>
327
+ )
328
+ }
329
+ ```
330
+
331
+ ### Virtualization for Long Lists
332
+
333
+ ```typescript
334
+ import { useVirtualizer } from '@tanstack/react-virtual'
335
+
336
+ export function VirtualMarketList({ markets }: { markets: Market[] }) {
337
+ const parentRef = useRef<HTMLDivElement>(null)
338
+
339
+ const virtualizer = useVirtualizer({
340
+ count: markets.length,
341
+ getScrollElement: () => parentRef.current,
342
+ estimateSize: () => 100, // Estimated row height
343
+ overscan: 5 // Extra items to render
344
+ })
345
+
346
+ return (
347
+ <div ref={parentRef} style={{ height: '600px', overflow: 'auto' }}>
348
+ <div
349
+ style={{
350
+ height: `${virtualizer.getTotalSize()}px`,
351
+ position: 'relative'
352
+ }}
353
+ >
354
+ {virtualizer.getVirtualItems().map(virtualRow => (
355
+ <div
356
+ key={virtualRow.index}
357
+ style={{
358
+ position: 'absolute',
359
+ top: 0,
360
+ left: 0,
361
+ width: '100%',
362
+ height: `${virtualRow.size}px`,
363
+ transform: `translateY(${virtualRow.start}px)`
364
+ }}
365
+ >
366
+ <MarketCard market={markets[virtualRow.index]} />
367
+ </div>
368
+ ))}
369
+ </div>
370
+ </div>
371
+ )
372
+ }
373
+ ```
374
+
375
+ ## Form Handling Patterns
376
+
377
+ ### Controlled Form with Validation
378
+
379
+ ```typescript
380
+ interface FormData {
381
+ name: string
382
+ description: string
383
+ endDate: string
384
+ }
385
+
386
+ interface FormErrors {
387
+ name?: string
388
+ description?: string
389
+ endDate?: string
390
+ }
391
+
392
+ export function CreateMarketForm() {
393
+ const [formData, setFormData] = useState<FormData>({
394
+ name: '',
395
+ description: '',
396
+ endDate: ''
397
+ })
398
+
399
+ const [errors, setErrors] = useState<FormErrors>({})
400
+
401
+ const validate = (): boolean => {
402
+ const newErrors: FormErrors = {}
403
+
404
+ if (!formData.name.trim()) {
405
+ newErrors.name = 'Name is required'
406
+ } else if (formData.name.length > 200) {
407
+ newErrors.name = 'Name must be under 200 characters'
408
+ }
409
+
410
+ if (!formData.description.trim()) {
411
+ newErrors.description = 'Description is required'
412
+ }
413
+
414
+ if (!formData.endDate) {
415
+ newErrors.endDate = 'End date is required'
416
+ }
417
+
418
+ setErrors(newErrors)
419
+ return Object.keys(newErrors).length === 0
420
+ }
421
+
422
+ const handleSubmit = async (e: React.FormEvent) => {
423
+ e.preventDefault()
424
+
425
+ if (!validate()) return
426
+
427
+ try {
428
+ await createMarket(formData)
429
+ // Success handling
430
+ } catch (error) {
431
+ // Error handling
432
+ }
433
+ }
434
+
435
+ return (
436
+ <form onSubmit={handleSubmit}>
437
+ <input
438
+ value={formData.name}
439
+ onChange={e => setFormData(prev => ({ ...prev, name: e.target.value }))}
440
+ placeholder="Market name"
441
+ />
442
+ {errors.name && <span className="error">{errors.name}</span>}
443
+
444
+ {/* Other fields */}
445
+
446
+ <button type="submit">Create Market</button>
447
+ </form>
448
+ )
449
+ }
450
+ ```
451
+
452
+ ## Error Boundary Pattern
453
+
454
+ ```typescript
455
+ interface ErrorBoundaryState {
456
+ hasError: boolean
457
+ error: Error | null
458
+ }
459
+
460
+ export class ErrorBoundary extends React.Component<
461
+ { children: React.ReactNode },
462
+ ErrorBoundaryState
463
+ > {
464
+ state: ErrorBoundaryState = {
465
+ hasError: false,
466
+ error: null
467
+ }
468
+
469
+ static getDerivedStateFromError(error: Error): ErrorBoundaryState {
470
+ return { hasError: true, error }
471
+ }
472
+
473
+ componentDidCatch(error: Error, errorInfo: React.ErrorInfo) {
474
+ console.error('Error boundary caught:', error, errorInfo)
475
+ }
476
+
477
+ render() {
478
+ if (this.state.hasError) {
479
+ return (
480
+ <div className="error-fallback">
481
+ <h2>Something went wrong</h2>
482
+ <p>{this.state.error?.message}</p>
483
+ <button onClick={() => this.setState({ hasError: false })}>
484
+ Try again
485
+ </button>
486
+ </div>
487
+ )
488
+ }
489
+
490
+ return this.props.children
491
+ }
492
+ }
493
+
494
+ // Usage
495
+ <ErrorBoundary>
496
+ <App />
497
+ </ErrorBoundary>
498
+ ```
499
+
500
+ ## Animation Patterns
501
+
502
+ ### Framer Motion Animations
503
+
504
+ ```typescript
505
+ import { motion, AnimatePresence } from 'framer-motion'
506
+
507
+ // ✅ List animations
508
+ export function AnimatedMarketList({ markets }: { markets: Market[] }) {
509
+ return (
510
+ <AnimatePresence>
511
+ {markets.map(market => (
512
+ <motion.div
513
+ key={market.id}
514
+ initial={{ opacity: 0, y: 20 }}
515
+ animate={{ opacity: 1, y: 0 }}
516
+ exit={{ opacity: 0, y: -20 }}
517
+ transition={{ duration: 0.3 }}
518
+ >
519
+ <MarketCard market={market} />
520
+ </motion.div>
521
+ ))}
522
+ </AnimatePresence>
523
+ )
524
+ }
525
+
526
+ // ✅ Modal animations
527
+ export function Modal({ isOpen, onClose, children }: ModalProps) {
528
+ return (
529
+ <AnimatePresence>
530
+ {isOpen && (
531
+ <>
532
+ <motion.div
533
+ className="modal-overlay"
534
+ initial={{ opacity: 0 }}
535
+ animate={{ opacity: 1 }}
536
+ exit={{ opacity: 0 }}
537
+ onClick={onClose}
538
+ />
539
+ <motion.div
540
+ className="modal-content"
541
+ initial={{ opacity: 0, scale: 0.9, y: 20 }}
542
+ animate={{ opacity: 1, scale: 1, y: 0 }}
543
+ exit={{ opacity: 0, scale: 0.9, y: 20 }}
544
+ >
545
+ {children}
546
+ </motion.div>
547
+ </>
548
+ )}
549
+ </AnimatePresence>
550
+ )
551
+ }
552
+ ```
553
+
554
+ ## Accessibility Patterns
555
+
556
+ ### Keyboard Navigation
557
+
558
+ ```typescript
559
+ export function Dropdown({ options, onSelect }: DropdownProps) {
560
+ const [isOpen, setIsOpen] = useState(false)
561
+ const [activeIndex, setActiveIndex] = useState(0)
562
+
563
+ const handleKeyDown = (e: React.KeyboardEvent) => {
564
+ switch (e.key) {
565
+ case 'ArrowDown':
566
+ e.preventDefault()
567
+ setActiveIndex(i => Math.min(i + 1, options.length - 1))
568
+ break
569
+ case 'ArrowUp':
570
+ e.preventDefault()
571
+ setActiveIndex(i => Math.max(i - 1, 0))
572
+ break
573
+ case 'Enter':
574
+ e.preventDefault()
575
+ onSelect(options[activeIndex])
576
+ setIsOpen(false)
577
+ break
578
+ case 'Escape':
579
+ setIsOpen(false)
580
+ break
581
+ }
582
+ }
583
+
584
+ return (
585
+ <div
586
+ role="combobox"
587
+ aria-expanded={isOpen}
588
+ aria-haspopup="listbox"
589
+ onKeyDown={handleKeyDown}
590
+ >
591
+ {/* Dropdown implementation */}
592
+ </div>
593
+ )
594
+ }
595
+ ```
596
+
597
+ ### Focus Management
598
+
599
+ ```typescript
600
+ export function Modal({ isOpen, onClose, children }: ModalProps) {
601
+ const modalRef = useRef<HTMLDivElement>(null)
602
+ const previousFocusRef = useRef<HTMLElement | null>(null)
603
+
604
+ useEffect(() => {
605
+ if (isOpen) {
606
+ // Save currently focused element
607
+ previousFocusRef.current = document.activeElement as HTMLElement
608
+
609
+ // Focus modal
610
+ modalRef.current?.focus()
611
+ } else {
612
+ // Restore focus when closing
613
+ previousFocusRef.current?.focus()
614
+ }
615
+ }, [isOpen])
616
+
617
+ return isOpen ? (
618
+ <div
619
+ ref={modalRef}
620
+ role="dialog"
621
+ aria-modal="true"
622
+ tabIndex={-1}
623
+ onKeyDown={e => e.key === 'Escape' && onClose()}
624
+ >
625
+ {children}
626
+ </div>
627
+ ) : null
628
+ }
629
+ ```
630
+
631
+ **Remember**: Modern frontend patterns enable maintainable, performant user interfaces. Choose patterns that fit your project complexity.