create-inox-app 0.1.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 (510) hide show
  1. package/README.md +110 -0
  2. package/dist/cli.js +22196 -0
  3. package/dist/prompt-Bp8saGVV.js +852 -0
  4. package/dist/templates/base/.gitignore.hbs +9 -0
  5. package/dist/templates/base/package.json.hbs +29 -0
  6. package/dist/templates/base/packages/typescript-config/base.json +17 -0
  7. package/dist/templates/base/packages/typescript-config/nextjs.json +8 -0
  8. package/dist/templates/base/packages/typescript-config/package.json +10 -0
  9. package/dist/templates/base/packages/typescript-config/react-library.json +7 -0
  10. package/dist/templates/base/turbo.json.hbs +32 -0
  11. package/dist/templates/ci/.gitlab-ci.yml.hbs +70 -0
  12. package/dist/templates/ci/scripts/build.sh.hbs +94 -0
  13. package/dist/templates/ci/scripts/prebuild.sh.hbs +11 -0
  14. package/dist/templates/claude-config/app/.agents/skills/agent-browser/SKILL.md +325 -0
  15. package/dist/templates/claude-config/app/.agents/skills/agent-browser/references/authentication.md +202 -0
  16. package/dist/templates/claude-config/app/.agents/skills/agent-browser/references/commands.md +259 -0
  17. package/dist/templates/claude-config/app/.agents/skills/agent-browser/references/proxy-support.md +188 -0
  18. package/dist/templates/claude-config/app/.agents/skills/agent-browser/references/session-management.md +193 -0
  19. package/dist/templates/claude-config/app/.agents/skills/agent-browser/references/snapshot-refs.md +194 -0
  20. package/dist/templates/claude-config/app/.agents/skills/agent-browser/references/video-recording.md +173 -0
  21. package/dist/templates/claude-config/app/.agents/skills/agent-browser/templates/authenticated-session.sh +100 -0
  22. package/dist/templates/claude-config/app/.agents/skills/agent-browser/templates/capture-workflow.sh +69 -0
  23. package/dist/templates/claude-config/app/.agents/skills/agent-browser/templates/form-automation.sh +62 -0
  24. package/dist/templates/claude-config/app/.agents/skills/ai-elements/SKILL.md +150 -0
  25. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/agent.md +131 -0
  26. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/artifact.md +84 -0
  27. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/attachments.md +190 -0
  28. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/audio-player.md +134 -0
  29. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/canvas.md +32 -0
  30. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/chain-of-thought.md +81 -0
  31. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/checkpoint.md +163 -0
  32. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/code-block.md +170 -0
  33. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/commit.md +177 -0
  34. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/confirmation.md +252 -0
  35. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/connection.md +32 -0
  36. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/context.md +126 -0
  37. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/controls.md +30 -0
  38. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/conversation.md +210 -0
  39. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/edge.md +50 -0
  40. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/environment-variables.md +102 -0
  41. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/file-tree.md +72 -0
  42. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/image.md +143 -0
  43. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/inline-citation.md +293 -0
  44. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/jsx-preview.md +101 -0
  45. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/message.md +256 -0
  46. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/mic-selector.md +186 -0
  47. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/model-selector.md +112 -0
  48. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/node.md +71 -0
  49. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/open-in-chat.md +67 -0
  50. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/package-info.md +95 -0
  51. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/panel.md +31 -0
  52. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/persona.md +158 -0
  53. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/plan.md +79 -0
  54. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/prompt-input.md +555 -0
  55. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/queue.md +172 -0
  56. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/reasoning.md +219 -0
  57. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/sandbox.md +126 -0
  58. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/schema-display.md +102 -0
  59. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/shimmer.md +48 -0
  60. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/snippet.md +64 -0
  61. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/sources.md +193 -0
  62. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/speech-input.md +160 -0
  63. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/stack-trace.md +218 -0
  64. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/suggestion.md +121 -0
  65. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/task.md +215 -0
  66. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/terminal.md +103 -0
  67. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/test-results.md +157 -0
  68. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/tool.md +275 -0
  69. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/toolbar.md +30 -0
  70. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/transcription.md +120 -0
  71. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/voice-selector.md +241 -0
  72. package/dist/templates/claude-config/app/.agents/skills/ai-elements/references/web-preview.md +197 -0
  73. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/agent.tsx +61 -0
  74. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/artifact.tsx +111 -0
  75. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/attachments-inline.tsx +117 -0
  76. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/attachments-list.tsx +88 -0
  77. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/attachments.tsx +78 -0
  78. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/audio-player-remote.tsx +35 -0
  79. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/audio-player.tsx +69 -0
  80. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/chain-of-thought.tsx +74 -0
  81. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/checkpoint.tsx +90 -0
  82. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/code-block-dark.tsx +46 -0
  83. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/code-block.tsx +115 -0
  84. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/commit.tsx +94 -0
  85. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/confirmation-accepted.tsx +35 -0
  86. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/confirmation-rejected.tsx +35 -0
  87. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/confirmation-request.tsx +54 -0
  88. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/confirmation.tsx +53 -0
  89. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/context.tsx +45 -0
  90. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/conversation.tsx +176 -0
  91. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/environment-variables.tsx +55 -0
  92. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/file-tree-basic.tsx +14 -0
  93. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/file-tree-expanded.tsx +17 -0
  94. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/file-tree-selection.tsx +20 -0
  95. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/file-tree.tsx +38 -0
  96. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/image.tsx +20 -0
  97. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/inline-citation.tsx +95 -0
  98. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/jsx-preview.tsx +99 -0
  99. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/message.tsx +324 -0
  100. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/mic-selector.tsx +46 -0
  101. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/model-selector.tsx +362 -0
  102. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/open-in-chat.tsx +33 -0
  103. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/package-info.tsx +46 -0
  104. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/persona-command.tsx +96 -0
  105. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/persona-glint.tsx +96 -0
  106. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/persona-halo.tsx +96 -0
  107. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/persona-mana.tsx +96 -0
  108. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/persona-obsidian.tsx +96 -0
  109. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/persona-opal.tsx +96 -0
  110. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/plan.tsx +63 -0
  111. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/prompt-input-cursor.tsx +459 -0
  112. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/prompt-input-tooltip.tsx +40 -0
  113. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/prompt-input.tsx +247 -0
  114. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/queue-prompt-input.tsx +367 -0
  115. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/queue.tsx +276 -0
  116. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/reasoning.tsx +67 -0
  117. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/sandbox.tsx +166 -0
  118. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/schema-display-basic.tsx +7 -0
  119. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/schema-display-body.tsx +20 -0
  120. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/schema-display-nested.tsx +23 -0
  121. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/schema-display-params.tsx +16 -0
  122. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/schema-display.tsx +110 -0
  123. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/shimmer-duration.tsx +29 -0
  124. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/shimmer-elements.tsx +39 -0
  125. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/shimmer.tsx +17 -0
  126. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/snippet-plain.tsx +19 -0
  127. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/snippet.tsx +25 -0
  128. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/sources-custom.tsx +34 -0
  129. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/sources.tsx +27 -0
  130. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/speech-input.tsx +79 -0
  131. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/stack-trace-collapsed.tsx +39 -0
  132. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/stack-trace-no-internal.tsx +41 -0
  133. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/stack-trace.tsx +54 -0
  134. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/suggestion-input.tsx +137 -0
  135. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/suggestion.tsx +28 -0
  136. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/task.tsx +60 -0
  137. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/terminal-basic.tsx +7 -0
  138. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/terminal-clear.tsx +19 -0
  139. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/terminal-streaming.tsx +38 -0
  140. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/terminal.tsx +77 -0
  141. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/test-results-basic.tsx +27 -0
  142. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/test-results-errors.tsx +51 -0
  143. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/test-results-suites.tsx +40 -0
  144. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/test-results.tsx +78 -0
  145. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/tool-input-available.tsx +31 -0
  146. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/tool-input-streaming.tsx +30 -0
  147. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/tool-output-available.tsx +76 -0
  148. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/tool-output-error.tsx +45 -0
  149. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/tool.tsx +205 -0
  150. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/transcription.tsx +284 -0
  151. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/voice-selector.tsx +228 -0
  152. package/dist/templates/claude-config/app/.agents/skills/ai-elements/scripts/web-preview.tsx +99 -0
  153. package/dist/templates/claude-config/app/.agents/skills/ai-sdk/SKILL.md +78 -0
  154. package/dist/templates/claude-config/app/.agents/skills/ai-sdk/references/ai-gateway.md +66 -0
  155. package/dist/templates/claude-config/app/.agents/skills/ai-sdk/references/common-errors.md +439 -0
  156. package/dist/templates/claude-config/app/.agents/skills/ai-sdk/references/devtools.md +52 -0
  157. package/dist/templates/claude-config/app/.agents/skills/ai-sdk/references/type-safe-agents.md +200 -0
  158. package/dist/templates/claude-config/app/.agents/skills/better-auth-best-practices/SKILL.md +166 -0
  159. package/dist/templates/claude-config/app/.agents/skills/building-components/SKILL.md +37 -0
  160. package/dist/templates/claude-config/app/.agents/skills/building-components/references/accessibility.mdx +819 -0
  161. package/dist/templates/claude-config/app/.agents/skills/building-components/references/as-child.mdx +324 -0
  162. package/dist/templates/claude-config/app/.agents/skills/building-components/references/composition.mdx +239 -0
  163. package/dist/templates/claude-config/app/.agents/skills/building-components/references/data-attributes.mdx +413 -0
  164. package/dist/templates/claude-config/app/.agents/skills/building-components/references/definitions.mdx +258 -0
  165. package/dist/templates/claude-config/app/.agents/skills/building-components/references/design-tokens.mdx +57 -0
  166. package/dist/templates/claude-config/app/.agents/skills/building-components/references/docs.mdx +155 -0
  167. package/dist/templates/claude-config/app/.agents/skills/building-components/references/marketplaces.mdx +144 -0
  168. package/dist/templates/claude-config/app/.agents/skills/building-components/references/npm.mdx +166 -0
  169. package/dist/templates/claude-config/app/.agents/skills/building-components/references/polymorphism.mdx +583 -0
  170. package/dist/templates/claude-config/app/.agents/skills/building-components/references/principles.mdx +61 -0
  171. package/dist/templates/claude-config/app/.agents/skills/building-components/references/registry.mdx +169 -0
  172. package/dist/templates/claude-config/app/.agents/skills/building-components/references/state.mdx +99 -0
  173. package/dist/templates/claude-config/app/.agents/skills/building-components/references/styling.mdx +286 -0
  174. package/dist/templates/claude-config/app/.agents/skills/building-components/references/types.mdx +191 -0
  175. package/dist/templates/claude-config/app/.agents/skills/docker-expert/SKILL.md +409 -0
  176. package/dist/templates/claude-config/app/.agents/skills/email-and-password-best-practices/SKILL.md +224 -0
  177. package/dist/templates/claude-config/app/.agents/skills/find-skills/SKILL.md +133 -0
  178. package/dist/templates/claude-config/app/.agents/skills/frontend-design/LICENSE.txt +177 -0
  179. package/dist/templates/claude-config/app/.agents/skills/frontend-design/SKILL.md +42 -0
  180. package/dist/templates/claude-config/app/.agents/skills/git-commit/SKILL.md +124 -0
  181. package/dist/templates/claude-config/app/.agents/skills/gitlab-ci-patterns/SKILL.md +271 -0
  182. package/dist/templates/claude-config/app/.agents/skills/hono/SKILL.md +90 -0
  183. package/dist/templates/claude-config/app/.agents/skills/langfuse-observability/SKILL.md +139 -0
  184. package/dist/templates/claude-config/app/.agents/skills/logging-best-practices/SKILL.md +127 -0
  185. package/dist/templates/claude-config/app/.agents/skills/logging-best-practices/rules/context.md +157 -0
  186. package/dist/templates/claude-config/app/.agents/skills/logging-best-practices/rules/pitfalls.md +118 -0
  187. package/dist/templates/claude-config/app/.agents/skills/logging-best-practices/rules/structure.md +193 -0
  188. package/dist/templates/claude-config/app/.agents/skills/logging-best-practices/rules/wide-events.md +113 -0
  189. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/SKILL.md +153 -0
  190. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/async-patterns.md +87 -0
  191. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/bundling.md +180 -0
  192. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/data-patterns.md +297 -0
  193. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/debug-tricks.md +105 -0
  194. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/directives.md +73 -0
  195. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/error-handling.md +227 -0
  196. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/file-conventions.md +140 -0
  197. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/font.md +245 -0
  198. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/functions.md +108 -0
  199. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/hydration-error.md +91 -0
  200. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/image.md +173 -0
  201. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/metadata.md +301 -0
  202. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/parallel-routes.md +287 -0
  203. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/route-handlers.md +146 -0
  204. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/rsc-boundaries.md +159 -0
  205. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/runtime-selection.md +39 -0
  206. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/scripts.md +141 -0
  207. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/self-hosting.md +371 -0
  208. package/dist/templates/claude-config/app/.agents/skills/next-best-practices/suspense-boundaries.md +67 -0
  209. package/dist/templates/claude-config/app/.agents/skills/next-cache-components/SKILL.md +411 -0
  210. package/dist/templates/claude-config/app/.agents/skills/next-upgrade/SKILL.md +50 -0
  211. package/dist/templates/claude-config/app/.agents/skills/postgres-drizzle/SKILL.md +179 -0
  212. package/dist/templates/claude-config/app/.agents/skills/postgres-drizzle/references/CHEATSHEET.md +418 -0
  213. package/dist/templates/claude-config/app/.agents/skills/postgres-drizzle/references/MIGRATIONS.md +536 -0
  214. package/dist/templates/claude-config/app/.agents/skills/postgres-drizzle/references/PERFORMANCE.md +559 -0
  215. package/dist/templates/claude-config/app/.agents/skills/postgres-drizzle/references/POSTGRES.md +588 -0
  216. package/dist/templates/claude-config/app/.agents/skills/postgres-drizzle/references/QUERIES.md +764 -0
  217. package/dist/templates/claude-config/app/.agents/skills/postgres-drizzle/references/RELATIONS.md +624 -0
  218. package/dist/templates/claude-config/app/.agents/skills/postgres-drizzle/references/SCHEMA.md +554 -0
  219. package/dist/templates/claude-config/app/.agents/skills/skill-creator/LICENSE.txt +202 -0
  220. package/dist/templates/claude-config/app/.agents/skills/skill-creator/SKILL.md +356 -0
  221. package/dist/templates/claude-config/app/.agents/skills/skill-creator/references/output-patterns.md +82 -0
  222. package/dist/templates/claude-config/app/.agents/skills/skill-creator/references/workflows.md +28 -0
  223. package/dist/templates/claude-config/app/.agents/skills/skill-creator/scripts/init_skill.py +303 -0
  224. package/dist/templates/claude-config/app/.agents/skills/skill-creator/scripts/package_skill.py +113 -0
  225. package/dist/templates/claude-config/app/.agents/skills/skill-creator/scripts/quick_validate.py +95 -0
  226. package/dist/templates/claude-config/app/.agents/skills/streamdown/SKILL.md +164 -0
  227. package/dist/templates/claude-config/app/.agents/skills/streamdown/assets/examples/basic-streaming.tsx +34 -0
  228. package/dist/templates/claude-config/app/.agents/skills/streamdown/assets/examples/custom-security.tsx +38 -0
  229. package/dist/templates/claude-config/app/.agents/skills/streamdown/assets/examples/full-featured.tsx +60 -0
  230. package/dist/templates/claude-config/app/.agents/skills/streamdown/assets/examples/static-mode.tsx +17 -0
  231. package/dist/templates/claude-config/app/.agents/skills/streamdown/assets/examples/with-caret.tsx +39 -0
  232. package/dist/templates/claude-config/app/.agents/skills/streamdown/references/api.md +278 -0
  233. package/dist/templates/claude-config/app/.agents/skills/streamdown/references/features.md +201 -0
  234. package/dist/templates/claude-config/app/.agents/skills/streamdown/references/plugins.md +239 -0
  235. package/dist/templates/claude-config/app/.agents/skills/streamdown/references/security.md +192 -0
  236. package/dist/templates/claude-config/app/.agents/skills/streamdown/references/styling.md +166 -0
  237. package/dist/templates/claude-config/app/.agents/skills/tdd/SKILL.md +107 -0
  238. package/dist/templates/claude-config/app/.agents/skills/tdd/deep-modules.md +33 -0
  239. package/dist/templates/claude-config/app/.agents/skills/tdd/interface-design.md +31 -0
  240. package/dist/templates/claude-config/app/.agents/skills/tdd/mocking.md +60 -0
  241. package/dist/templates/claude-config/app/.agents/skills/tdd/refactoring.md +10 -0
  242. package/dist/templates/claude-config/app/.agents/skills/tdd/tests.md +61 -0
  243. package/dist/templates/claude-config/app/.agents/skills/turborepo/SKILL.md +914 -0
  244. package/dist/templates/claude-config/app/.agents/skills/turborepo/command/turborepo.md +70 -0
  245. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/best-practices/RULE.md +241 -0
  246. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/best-practices/dependencies.md +246 -0
  247. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/best-practices/packages.md +335 -0
  248. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/best-practices/structure.md +270 -0
  249. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/boundaries/RULE.md +126 -0
  250. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/caching/RULE.md +107 -0
  251. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/caching/gotchas.md +169 -0
  252. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/caching/remote-cache.md +127 -0
  253. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/ci/RULE.md +79 -0
  254. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/ci/github-actions.md +162 -0
  255. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/ci/patterns.md +145 -0
  256. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/ci/vercel.md +103 -0
  257. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/cli/RULE.md +100 -0
  258. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/cli/commands.md +297 -0
  259. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/configuration/RULE.md +211 -0
  260. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/configuration/global-options.md +191 -0
  261. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/configuration/gotchas.md +348 -0
  262. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/configuration/tasks.md +281 -0
  263. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/environment/RULE.md +96 -0
  264. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/environment/gotchas.md +141 -0
  265. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/environment/modes.md +101 -0
  266. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/filtering/RULE.md +148 -0
  267. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/filtering/patterns.md +152 -0
  268. package/dist/templates/claude-config/app/.agents/skills/turborepo/references/watch/RULE.md +99 -0
  269. package/dist/templates/claude-config/app/.agents/skills/vercel-composition-patterns/AGENTS.md +917 -0
  270. package/dist/templates/claude-config/app/.agents/skills/vercel-composition-patterns/SKILL.md +88 -0
  271. package/dist/templates/claude-config/app/.agents/skills/vercel-composition-patterns/rules/architecture-avoid-boolean-props.md +94 -0
  272. package/dist/templates/claude-config/app/.agents/skills/vercel-composition-patterns/rules/architecture-compound-components.md +108 -0
  273. package/dist/templates/claude-config/app/.agents/skills/vercel-composition-patterns/rules/patterns-children-over-render-props.md +84 -0
  274. package/dist/templates/claude-config/app/.agents/skills/vercel-composition-patterns/rules/patterns-explicit-variants.md +94 -0
  275. package/dist/templates/claude-config/app/.agents/skills/vercel-composition-patterns/rules/react19-no-forwardref.md +42 -0
  276. package/dist/templates/claude-config/app/.agents/skills/vercel-composition-patterns/rules/state-context-interface.md +191 -0
  277. package/dist/templates/claude-config/app/.agents/skills/vercel-composition-patterns/rules/state-decouple-implementation.md +103 -0
  278. package/dist/templates/claude-config/app/.agents/skills/vercel-composition-patterns/rules/state-lift-state.md +125 -0
  279. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/AGENTS.md +2883 -0
  280. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/SKILL.md +138 -0
  281. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/advanced-event-handler-refs.md +55 -0
  282. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/advanced-init-once.md +42 -0
  283. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/advanced-use-latest.md +39 -0
  284. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/async-api-routes.md +35 -0
  285. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/async-defer-await.md +80 -0
  286. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/async-dependencies.md +48 -0
  287. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/async-parallel.md +24 -0
  288. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/async-suspense-boundaries.md +99 -0
  289. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/bundle-barrel-imports.md +59 -0
  290. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/bundle-conditional.md +37 -0
  291. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/bundle-defer-third-party.md +48 -0
  292. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/bundle-dynamic-imports.md +34 -0
  293. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/bundle-preload.md +44 -0
  294. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/client-event-listeners.md +78 -0
  295. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/client-localstorage-schema.md +74 -0
  296. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/client-passive-event-listeners.md +48 -0
  297. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/client-swr-dedup.md +56 -0
  298. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-batch-dom-css.md +110 -0
  299. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-cache-function-results.md +80 -0
  300. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-cache-property-access.md +28 -0
  301. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-cache-storage.md +68 -0
  302. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-combine-iterations.md +32 -0
  303. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-early-exit.md +50 -0
  304. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-hoist-regexp.md +45 -0
  305. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-index-maps.md +37 -0
  306. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-length-check-first.md +50 -0
  307. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-min-max-loop.md +82 -0
  308. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-set-map-lookups.md +24 -0
  309. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/js-tosorted-immutable.md +57 -0
  310. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rendering-activity.md +26 -0
  311. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rendering-animate-svg-wrapper.md +38 -0
  312. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rendering-conditional-render.md +32 -0
  313. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rendering-content-visibility.md +38 -0
  314. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rendering-hoist-jsx.md +36 -0
  315. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-no-flicker.md +72 -0
  316. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-suppress-warning.md +26 -0
  317. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rendering-svg-precision.md +28 -0
  318. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rendering-usetransition-loading.md +75 -0
  319. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-defer-reads.md +39 -0
  320. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-dependencies.md +45 -0
  321. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state-no-effect.md +40 -0
  322. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state.md +29 -0
  323. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-functional-setstate.md +77 -0
  324. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-lazy-state-init.md +56 -0
  325. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-memo-with-default-value.md +36 -0
  326. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-memo.md +44 -0
  327. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-move-effect-to-event.md +45 -0
  328. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-simple-expression-in-memo.md +35 -0
  329. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-transitions.md +40 -0
  330. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/rerender-use-ref-transient-values.md +73 -0
  331. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/server-after-nonblocking.md +73 -0
  332. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/server-auth-actions.md +96 -0
  333. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/server-cache-lru.md +41 -0
  334. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/server-cache-react.md +76 -0
  335. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/server-dedup-props.md +65 -0
  336. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/server-parallel-fetching.md +83 -0
  337. package/dist/templates/claude-config/app/.agents/skills/vercel-react-best-practices/rules/server-serialization.md +38 -0
  338. package/dist/templates/claude-config/app/.agents/skills/web-design-guidelines/SKILL.md +40 -0
  339. package/dist/templates/claude-config/app/.agents/skills/workflow/SKILL.md +181 -0
  340. package/dist/templates/claude-config/app/.claude/commands/council.md +7 -0
  341. package/dist/templates/claude-config/app/.claude/commands/deslop.md +25 -0
  342. package/dist/templates/claude-config/app/.claude/commands/diagram.md +101 -0
  343. package/dist/templates/claude-config/app/.claude/commands/fix-merge.md +47 -0
  344. package/dist/templates/claude-config/app/.claude/commands/you-might-not-need-an-effect.md +10 -0
  345. package/dist/templates/claude-config/app/.claude/hooks/post-tool-use-tracker.sh +187 -0
  346. package/dist/templates/claude-config/app/.claude/hooks/skill-activation-prompt.sh +5 -0
  347. package/dist/templates/claude-config/app/.claude/hooks/skill-activation-prompt.ts +134 -0
  348. package/dist/templates/claude-config/app/.claude/rules/comments.md +12 -0
  349. package/dist/templates/claude-config/app/.claude/rules/typescript.md +31 -0
  350. package/dist/templates/claude-config/app/.claude/settings.json.hbs +74 -0
  351. package/dist/templates/claude-config/app/.claude/skills/skill-rules.json.hbs +309 -0
  352. package/dist/templates/claude-config/app/AGENTS.md.hbs +185 -0
  353. package/dist/templates/claude-config/app/CLAUDE.md.hbs +5 -0
  354. package/dist/templates/claude-config/kustomize/.agents/skills/git-commit/SKILL.md +124 -0
  355. package/dist/templates/claude-config/kustomize/.agents/skills/k8s-manifest-generator/SKILL.md +534 -0
  356. package/dist/templates/claude-config/kustomize/.agents/skills/k8s-manifest-generator/assets/configmap-template.yaml +296 -0
  357. package/dist/templates/claude-config/kustomize/.agents/skills/k8s-manifest-generator/assets/deployment-template.yaml +203 -0
  358. package/dist/templates/claude-config/kustomize/.agents/skills/k8s-manifest-generator/assets/service-template.yaml +171 -0
  359. package/dist/templates/claude-config/kustomize/.agents/skills/k8s-manifest-generator/references/deployment-spec.md +780 -0
  360. package/dist/templates/claude-config/kustomize/.agents/skills/k8s-manifest-generator/references/service-spec.md +748 -0
  361. package/dist/templates/claude-config/kustomize/.claude/settings.json +42 -0
  362. package/dist/templates/claude-config/kustomize/.claude/skills/git-commit/SKILL.md +124 -0
  363. package/dist/templates/claude-config/kustomize/.claude/skills/k8s-manifest-generator/SKILL.md +534 -0
  364. package/dist/templates/claude-config/kustomize/.claude/skills/k8s-manifest-generator/assets/configmap-template.yaml +296 -0
  365. package/dist/templates/claude-config/kustomize/.claude/skills/k8s-manifest-generator/assets/deployment-template.yaml +203 -0
  366. package/dist/templates/claude-config/kustomize/.claude/skills/k8s-manifest-generator/assets/service-template.yaml +171 -0
  367. package/dist/templates/claude-config/kustomize/.claude/skills/k8s-manifest-generator/references/deployment-spec.md +780 -0
  368. package/dist/templates/claude-config/kustomize/.claude/skills/k8s-manifest-generator/references/service-spec.md +748 -0
  369. package/dist/templates/docker/.dockerignore.hbs +8 -0
  370. package/dist/templates/docker/Dockerfile.hbs +81 -0
  371. package/dist/templates/docker/docker-compose.yml.hbs +55 -0
  372. package/dist/templates/kustomize/base/deployments/app.yaml.hbs +44 -0
  373. package/dist/templates/kustomize/base/ingress/app.yaml.hbs +20 -0
  374. package/dist/templates/kustomize/base/jobs/app-jobs.yaml.hbs +43 -0
  375. package/dist/templates/kustomize/base/kustomization.yaml.hbs +13 -0
  376. package/dist/templates/kustomize/base/secrets/regcred.yaml.hbs +7 -0
  377. package/dist/templates/kustomize/base/services/app.yaml.hbs +12 -0
  378. package/dist/templates/kustomize/overlays/{{env}}/configs/app.config.env.hbs +21 -0
  379. package/dist/templates/kustomize/overlays/{{env}}/configs/paradedb.config.env.hbs +3 -0
  380. package/dist/templates/kustomize/overlays/{{env}}/deployments/paradedb.yaml.hbs +28 -0
  381. package/dist/templates/kustomize/overlays/{{env}}/deployments/redis.yaml.hbs +18 -0
  382. package/dist/templates/kustomize/overlays/{{env}}/kustomization.yaml.hbs +36 -0
  383. package/dist/templates/kustomize/overlays/{{env}}/patch-operation/ingress-host.yaml.hbs +15 -0
  384. package/dist/templates/kustomize/overlays/{{env}}/services/paradedb.yaml.hbs +11 -0
  385. package/dist/templates/kustomize/overlays/{{env}}/services/redis.yaml.hbs +11 -0
  386. package/dist/templates/kustomize/overlays/{{env}}/storage/paradedb-pvc.yaml.hbs +10 -0
  387. package/dist/templates/presets/ai-chat-app/apps/web/.env.example +11 -0
  388. package/dist/templates/presets/ai-chat-app/apps/web/app/api/auth/[...all]/route.ts +5 -0
  389. package/dist/templates/presets/ai-chat-app/apps/web/app/api/chat/input-guardrail.ts +47 -0
  390. package/dist/templates/presets/ai-chat-app/apps/web/app/api/chat/route.ts +266 -0
  391. package/dist/templates/presets/ai-chat-app/apps/web/app/api/chat/sessions/[id]/messages/route.ts +38 -0
  392. package/dist/templates/presets/ai-chat-app/apps/web/app/api/chat/sessions/[id]/route.ts +59 -0
  393. package/dist/templates/presets/ai-chat-app/apps/web/app/api/chat/sessions/route.ts +24 -0
  394. package/dist/templates/presets/ai-chat-app/apps/web/app/api/health/route.ts +21 -0
  395. package/dist/templates/presets/ai-chat-app/apps/web/app/chat/[id]/page.tsx +54 -0
  396. package/dist/templates/presets/ai-chat-app/apps/web/app/chat/layout.tsx +23 -0
  397. package/dist/templates/presets/ai-chat-app/apps/web/app/chat/page.tsx +9 -0
  398. package/dist/templates/presets/ai-chat-app/apps/web/app/error.tsx +20 -0
  399. package/dist/templates/presets/ai-chat-app/apps/web/app/global-error.tsx +42 -0
  400. package/dist/templates/presets/ai-chat-app/apps/web/app/globals.css +187 -0
  401. package/dist/templates/presets/ai-chat-app/apps/web/app/layout.tsx.hbs +30 -0
  402. package/dist/templates/presets/ai-chat-app/apps/web/app/login/login-form.tsx +96 -0
  403. package/dist/templates/presets/ai-chat-app/apps/web/app/login/page.tsx +14 -0
  404. package/dist/templates/presets/ai-chat-app/apps/web/app/not-found.tsx +15 -0
  405. package/dist/templates/presets/ai-chat-app/apps/web/components/ai-elements/code-block.tsx +518 -0
  406. package/dist/templates/presets/ai-chat-app/apps/web/components/ai-elements/context.tsx +356 -0
  407. package/dist/templates/presets/ai-chat-app/apps/web/components/ai-elements/conversation.tsx +150 -0
  408. package/dist/templates/presets/ai-chat-app/apps/web/components/ai-elements/environment-variables.tsx +303 -0
  409. package/dist/templates/presets/ai-chat-app/apps/web/components/ai-elements/message.tsx +295 -0
  410. package/dist/templates/presets/ai-chat-app/apps/web/components/ai-elements/prompt-input.tsx +1179 -0
  411. package/dist/templates/presets/ai-chat-app/apps/web/components/ai-elements/sources.tsx +54 -0
  412. package/dist/templates/presets/ai-chat-app/apps/web/components/ai-elements/tool.tsx +157 -0
  413. package/dist/templates/presets/ai-chat-app/apps/web/components/app-sidebar.tsx +110 -0
  414. package/dist/templates/presets/ai-chat-app/apps/web/components/chat-sessions-provider.tsx +89 -0
  415. package/dist/templates/presets/ai-chat-app/apps/web/components/chat-sidebar-list.tsx +174 -0
  416. package/dist/templates/presets/ai-chat-app/apps/web/components/chat-view.tsx +156 -0
  417. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/accordion.tsx +72 -0
  418. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/alert-dialog.tsx +162 -0
  419. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/alert.tsx +73 -0
  420. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/aspect-ratio.tsx +22 -0
  421. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/avatar.tsx +93 -0
  422. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/badge.tsx +49 -0
  423. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/breadcrumb.tsx +103 -0
  424. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/button-group.tsx +78 -0
  425. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/button.tsx +60 -0
  426. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/calendar.tsx +185 -0
  427. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/card.tsx +92 -0
  428. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/carousel.tsx +231 -0
  429. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/chart.tsx +325 -0
  430. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/checkbox.tsx +28 -0
  431. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/collapsible.tsx +17 -0
  432. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/combobox.tsx +273 -0
  433. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/command.tsx +181 -0
  434. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/context-menu.tsx +245 -0
  435. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/dialog.tsx +135 -0
  436. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/direction.tsx +3 -0
  437. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/drawer.tsx +120 -0
  438. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/dropdown-menu.tsx +258 -0
  439. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/empty.tsx +94 -0
  440. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/field.tsx +224 -0
  441. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/hover-card.tsx +46 -0
  442. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/input-group.tsx +146 -0
  443. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/input-otp.tsx +86 -0
  444. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/input.tsx +20 -0
  445. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/item.tsx +188 -0
  446. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/kbd.tsx +26 -0
  447. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/label.tsx +20 -0
  448. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/menubar.tsx +270 -0
  449. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/native-select.tsx +43 -0
  450. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/navigation-menu.tsx +161 -0
  451. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/pagination.tsx +118 -0
  452. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/popover.tsx +77 -0
  453. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/progress.tsx +66 -0
  454. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/radio-group.tsx +39 -0
  455. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/resizable.tsx +42 -0
  456. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/scroll-area.tsx +51 -0
  457. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/select.tsx +190 -0
  458. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/separator.tsx +21 -0
  459. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/sheet.tsx +125 -0
  460. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/sidebar.tsx +689 -0
  461. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/skeleton.tsx +13 -0
  462. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/slider.tsx +54 -0
  463. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/sonner.tsx +45 -0
  464. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/spinner.tsx +15 -0
  465. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/switch.tsx +32 -0
  466. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/table.tsx +89 -0
  467. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/tabs.tsx +75 -0
  468. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/textarea.tsx +18 -0
  469. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/toggle-group.tsx +87 -0
  470. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/toggle.tsx +44 -0
  471. package/dist/templates/presets/ai-chat-app/apps/web/components/ui/tooltip.tsx +54 -0
  472. package/dist/templates/presets/ai-chat-app/apps/web/components.json +20 -0
  473. package/dist/templates/presets/ai-chat-app/apps/web/drizzle.config.ts +10 -0
  474. package/dist/templates/presets/ai-chat-app/apps/web/env.ts +30 -0
  475. package/dist/templates/presets/ai-chat-app/apps/web/hooks/use-mobile.ts +19 -0
  476. package/dist/templates/presets/ai-chat-app/apps/web/instrumentation.ts +21 -0
  477. package/dist/templates/presets/ai-chat-app/apps/web/lib/auth-client.ts +3 -0
  478. package/dist/templates/presets/ai-chat-app/apps/web/lib/auth-session.ts +9 -0
  479. package/dist/templates/presets/ai-chat-app/apps/web/lib/auth.ts +16 -0
  480. package/dist/templates/presets/ai-chat-app/apps/web/lib/db/index.ts +11 -0
  481. package/dist/templates/presets/ai-chat-app/apps/web/lib/db/migrate.ts +53 -0
  482. package/dist/templates/presets/ai-chat-app/apps/web/lib/db/schema.ts +94 -0
  483. package/dist/templates/presets/ai-chat-app/apps/web/lib/db/seed.ts +123 -0
  484. package/dist/templates/presets/ai-chat-app/apps/web/lib/group-by-date.ts +34 -0
  485. package/dist/templates/presets/ai-chat-app/apps/web/lib/logger.ts +21 -0
  486. package/dist/templates/presets/ai-chat-app/apps/web/lib/openrouter.ts +7 -0
  487. package/dist/templates/presets/ai-chat-app/apps/web/lib/schemas/auth.ts +8 -0
  488. package/dist/templates/presets/ai-chat-app/apps/web/lib/utils.ts +4 -0
  489. package/dist/templates/presets/ai-chat-app/apps/web/next-env.d.ts +6 -0
  490. package/dist/templates/presets/ai-chat-app/apps/web/next.config.ts +14 -0
  491. package/dist/templates/presets/ai-chat-app/apps/web/package.json +70 -0
  492. package/dist/templates/presets/ai-chat-app/apps/web/postcss.config.mjs +7 -0
  493. package/dist/templates/presets/ai-chat-app/apps/web/proxy.ts +13 -0
  494. package/dist/templates/presets/ai-chat-app/apps/web/tsconfig.json +14 -0
  495. package/dist/templates/presets/blank/apps/web/.env.example +1 -0
  496. package/dist/templates/presets/blank/apps/web/app/api/health/route.ts +11 -0
  497. package/dist/templates/presets/blank/apps/web/app/globals.css +49 -0
  498. package/dist/templates/presets/blank/apps/web/app/layout.tsx.hbs +30 -0
  499. package/dist/templates/presets/blank/apps/web/app/page.tsx.hbs +10 -0
  500. package/dist/templates/presets/blank/apps/web/env.ts +10 -0
  501. package/dist/templates/presets/blank/apps/web/next-env.d.ts +2 -0
  502. package/dist/templates/presets/blank/apps/web/next.config.ts +9 -0
  503. package/dist/templates/presets/blank/apps/web/package.json.hbs +27 -0
  504. package/dist/templates/presets/blank/apps/web/postcss.config.mjs +7 -0
  505. package/dist/templates/presets/blank/apps/web/tsconfig.json +14 -0
  506. package/dist/templates/tooling/.oxfmtrc.json +6 -0
  507. package/dist/templates/tooling/.oxlintrc.json +20 -0
  508. package/dist/templates/tooling/commitlint.config.ts.hbs +3 -0
  509. package/dist/templates/tooling/lefthook.yml.hbs +25 -0
  510. package/package.json +47 -0
@@ -0,0 +1,917 @@
1
+ # React Composition Patterns
2
+
3
+ **Version 1.0.0**
4
+ Engineering
5
+ January 2026
6
+
7
+ > **Note:**
8
+ > This document is mainly for agents and LLMs to follow when maintaining,
9
+ > generating, or refactoring React codebases using composition. Humans
10
+ > may also find it useful, but guidance here is optimized for automation
11
+ > and consistency by AI-assisted workflows.
12
+
13
+ ---
14
+
15
+ ## Abstract
16
+
17
+ Composition patterns for building flexible, maintainable React components. Avoid boolean prop proliferation by using compound components, lifting state, and composing internals. These patterns make codebases easier for both humans and AI agents to work with as they scale.
18
+
19
+ ---
20
+
21
+ ## Table of Contents
22
+
23
+ 1. [Component Architecture](#1-component-architecture) — **HIGH**
24
+ - 1.1 [Avoid Boolean Prop Proliferation](#11-avoid-boolean-prop-proliferation)
25
+ - 1.2 [Use Compound Components](#12-use-compound-components)
26
+ 2. [State Management](#2-state-management) — **MEDIUM**
27
+ - 2.1 [Decouple State Management from UI](#21-decouple-state-management-from-ui)
28
+ - 2.2 [Define Generic Context Interfaces for Dependency Injection](#22-define-generic-context-interfaces-for-dependency-injection)
29
+ - 2.3 [Lift State into Provider Components](#23-lift-state-into-provider-components)
30
+ 3. [Implementation Patterns](#3-implementation-patterns) — **MEDIUM**
31
+ - 3.1 [Create Explicit Component Variants](#31-create-explicit-component-variants)
32
+ - 3.2 [Prefer Composing Children Over Render Props](#32-prefer-composing-children-over-render-props)
33
+ 4. [React 19 APIs](#4-react-19-apis) — **MEDIUM**
34
+ - 4.1 [React 19 API Changes](#41-react-19-api-changes)
35
+
36
+ ---
37
+
38
+ ## 1. Component Architecture
39
+
40
+ **Impact: HIGH**
41
+
42
+ Fundamental patterns for structuring components to avoid prop
43
+ proliferation and enable flexible composition.
44
+
45
+ ### 1.1 Avoid Boolean Prop Proliferation
46
+
47
+ **Impact: CRITICAL (prevents unmaintainable component variants)**
48
+
49
+ Don't add boolean props like `isThread`, `isEditing`, `isDMThread` to customize
50
+
51
+ component behavior. Each boolean doubles possible states and creates
52
+
53
+ unmaintainable conditional logic. Use composition instead.
54
+
55
+ **Incorrect: boolean props create exponential complexity**
56
+
57
+ ```tsx
58
+ function Composer({
59
+ onSubmit,
60
+ isThread,
61
+ channelId,
62
+ isDMThread,
63
+ dmId,
64
+ isEditing,
65
+ isForwarding,
66
+ }: Props) {
67
+ return (
68
+ <form>
69
+ <Header />
70
+ <Input />
71
+ {isDMThread ? (
72
+ <AlsoSendToDMField id={dmId} />
73
+ ) : isThread ? (
74
+ <AlsoSendToChannelField id={channelId} />
75
+ ) : null}
76
+ {isEditing ? <EditActions /> : isForwarding ? <ForwardActions /> : <DefaultActions />}
77
+ <Footer onSubmit={onSubmit} />
78
+ </form>
79
+ );
80
+ }
81
+ ```
82
+
83
+ **Correct: composition eliminates conditionals**
84
+
85
+ ```tsx
86
+ // Channel composer
87
+ function ChannelComposer() {
88
+ return (
89
+ <Composer.Frame>
90
+ <Composer.Header />
91
+ <Composer.Input />
92
+ <Composer.Footer>
93
+ <Composer.Attachments />
94
+ <Composer.Formatting />
95
+ <Composer.Emojis />
96
+ <Composer.Submit />
97
+ </Composer.Footer>
98
+ </Composer.Frame>
99
+ );
100
+ }
101
+
102
+ // Thread composer - adds "also send to channel" field
103
+ function ThreadComposer({ channelId }: { channelId: string }) {
104
+ return (
105
+ <Composer.Frame>
106
+ <Composer.Header />
107
+ <Composer.Input />
108
+ <AlsoSendToChannelField id={channelId} />
109
+ <Composer.Footer>
110
+ <Composer.Formatting />
111
+ <Composer.Emojis />
112
+ <Composer.Submit />
113
+ </Composer.Footer>
114
+ </Composer.Frame>
115
+ );
116
+ }
117
+
118
+ // Edit composer - different footer actions
119
+ function EditComposer() {
120
+ return (
121
+ <Composer.Frame>
122
+ <Composer.Input />
123
+ <Composer.Footer>
124
+ <Composer.Formatting />
125
+ <Composer.Emojis />
126
+ <Composer.CancelEdit />
127
+ <Composer.SaveEdit />
128
+ </Composer.Footer>
129
+ </Composer.Frame>
130
+ );
131
+ }
132
+ ```
133
+
134
+ Each variant is explicit about what it renders. We can share internals without
135
+
136
+ sharing a single monolithic parent.
137
+
138
+ ### 1.2 Use Compound Components
139
+
140
+ **Impact: HIGH (enables flexible composition without prop drilling)**
141
+
142
+ Structure complex components as compound components with a shared context. Each
143
+
144
+ subcomponent accesses shared state via context, not props. Consumers compose the
145
+
146
+ pieces they need.
147
+
148
+ **Incorrect: monolithic component with render props**
149
+
150
+ ```tsx
151
+ function Composer({
152
+ renderHeader,
153
+ renderFooter,
154
+ renderActions,
155
+ showAttachments,
156
+ showFormatting,
157
+ showEmojis,
158
+ }: Props) {
159
+ return (
160
+ <form>
161
+ {renderHeader?.()}
162
+ <Input />
163
+ {showAttachments && <Attachments />}
164
+ {renderFooter ? (
165
+ renderFooter()
166
+ ) : (
167
+ <Footer>
168
+ {showFormatting && <Formatting />}
169
+ {showEmojis && <Emojis />}
170
+ {renderActions?.()}
171
+ </Footer>
172
+ )}
173
+ </form>
174
+ );
175
+ }
176
+ ```
177
+
178
+ **Correct: compound components with shared context**
179
+
180
+ ```tsx
181
+ const ComposerContext = createContext<ComposerContextValue | null>(null);
182
+
183
+ function ComposerProvider({ children, state, actions, meta }: ProviderProps) {
184
+ return <ComposerContext value={{ state, actions, meta }}>{children}</ComposerContext>;
185
+ }
186
+
187
+ function ComposerFrame({ children }: { children: React.ReactNode }) {
188
+ return <form>{children}</form>;
189
+ }
190
+
191
+ function ComposerInput() {
192
+ const {
193
+ state,
194
+ actions: { update },
195
+ meta: { inputRef },
196
+ } = use(ComposerContext);
197
+ return (
198
+ <TextInput
199
+ ref={inputRef}
200
+ value={state.input}
201
+ onChangeText={(text) => update((s) => ({ ...s, input: text }))}
202
+ />
203
+ );
204
+ }
205
+
206
+ function ComposerSubmit() {
207
+ const {
208
+ actions: { submit },
209
+ } = use(ComposerContext);
210
+ return <Button onPress={submit}>Send</Button>;
211
+ }
212
+
213
+ // Export as compound component
214
+ const Composer = {
215
+ Provider: ComposerProvider,
216
+ Frame: ComposerFrame,
217
+ Input: ComposerInput,
218
+ Submit: ComposerSubmit,
219
+ Header: ComposerHeader,
220
+ Footer: ComposerFooter,
221
+ Attachments: ComposerAttachments,
222
+ Formatting: ComposerFormatting,
223
+ Emojis: ComposerEmojis,
224
+ };
225
+ ```
226
+
227
+ **Usage:**
228
+
229
+ ```tsx
230
+ <Composer.Provider state={state} actions={actions} meta={meta}>
231
+ <Composer.Frame>
232
+ <Composer.Header />
233
+ <Composer.Input />
234
+ <Composer.Footer>
235
+ <Composer.Formatting />
236
+ <Composer.Submit />
237
+ </Composer.Footer>
238
+ </Composer.Frame>
239
+ </Composer.Provider>
240
+ ```
241
+
242
+ Consumers explicitly compose exactly what they need. No hidden conditionals. And the state, actions and meta are dependency-injected by a parent provider, allowing multiple usages of the same component structure.
243
+
244
+ ---
245
+
246
+ ## 2. State Management
247
+
248
+ **Impact: MEDIUM**
249
+
250
+ Patterns for lifting state and managing shared context across
251
+ composed components.
252
+
253
+ ### 2.1 Decouple State Management from UI
254
+
255
+ **Impact: MEDIUM (enables swapping state implementations without changing UI)**
256
+
257
+ The provider component should be the only place that knows how state is managed.
258
+
259
+ UI components consume the context interface—they don't know if state comes from
260
+
261
+ useState, Zustand, or a server sync.
262
+
263
+ **Incorrect: UI coupled to state implementation**
264
+
265
+ ```tsx
266
+ function ChannelComposer({ channelId }: { channelId: string }) {
267
+ // UI component knows about global state implementation
268
+ const state = useGlobalChannelState(channelId);
269
+ const { submit, updateInput } = useChannelSync(channelId);
270
+
271
+ return (
272
+ <Composer.Frame>
273
+ <Composer.Input value={state.input} onChange={(text) => sync.updateInput(text)} />
274
+ <Composer.Submit onPress={() => sync.submit()} />
275
+ </Composer.Frame>
276
+ );
277
+ }
278
+ ```
279
+
280
+ **Correct: state management isolated in provider**
281
+
282
+ ```tsx
283
+ // Provider handles all state management details
284
+ function ChannelProvider({
285
+ channelId,
286
+ children,
287
+ }: {
288
+ channelId: string;
289
+ children: React.ReactNode;
290
+ }) {
291
+ const { state, update, submit } = useGlobalChannel(channelId);
292
+ const inputRef = useRef(null);
293
+
294
+ return (
295
+ <Composer.Provider state={state} actions={{ update, submit }} meta={{ inputRef }}>
296
+ {children}
297
+ </Composer.Provider>
298
+ );
299
+ }
300
+
301
+ // UI component only knows about the context interface
302
+ function ChannelComposer() {
303
+ return (
304
+ <Composer.Frame>
305
+ <Composer.Header />
306
+ <Composer.Input />
307
+ <Composer.Footer>
308
+ <Composer.Submit />
309
+ </Composer.Footer>
310
+ </Composer.Frame>
311
+ );
312
+ }
313
+
314
+ // Usage
315
+ function Channel({ channelId }: { channelId: string }) {
316
+ return (
317
+ <ChannelProvider channelId={channelId}>
318
+ <ChannelComposer />
319
+ </ChannelProvider>
320
+ );
321
+ }
322
+ ```
323
+
324
+ **Different providers, same UI:**
325
+
326
+ ```tsx
327
+ // Local state for ephemeral forms
328
+ function ForwardMessageProvider({ children }) {
329
+ const [state, setState] = useState(initialState);
330
+ const forwardMessage = useForwardMessage();
331
+
332
+ return (
333
+ <Composer.Provider state={state} actions={{ update: setState, submit: forwardMessage }}>
334
+ {children}
335
+ </Composer.Provider>
336
+ );
337
+ }
338
+
339
+ // Global synced state for channels
340
+ function ChannelProvider({ channelId, children }) {
341
+ const { state, update, submit } = useGlobalChannel(channelId);
342
+
343
+ return (
344
+ <Composer.Provider state={state} actions={{ update, submit }}>
345
+ {children}
346
+ </Composer.Provider>
347
+ );
348
+ }
349
+ ```
350
+
351
+ The same `Composer.Input` component works with both providers because it only
352
+
353
+ depends on the context interface, not the implementation.
354
+
355
+ ### 2.2 Define Generic Context Interfaces for Dependency Injection
356
+
357
+ **Impact: HIGH (enables dependency-injectable state across use-cases)**
358
+
359
+ Define a **generic interface** for your component context with three parts:
360
+
361
+ `state`, `actions`, and `meta`. This interface is a contract that any provider
362
+
363
+ can implement—enabling the same UI components to work with completely different
364
+
365
+ state implementations.
366
+
367
+ **Core principle:** Lift state, compose internals, make state
368
+
369
+ dependency-injectable.
370
+
371
+ **Incorrect: UI coupled to specific state implementation**
372
+
373
+ ```tsx
374
+ function ComposerInput() {
375
+ // Tightly coupled to a specific hook
376
+ const { input, setInput } = useChannelComposerState();
377
+ return <TextInput value={input} onChangeText={setInput} />;
378
+ }
379
+ ```
380
+
381
+ **Correct: generic interface enables dependency injection**
382
+
383
+ ```tsx
384
+ // Define a GENERIC interface that any provider can implement
385
+ interface ComposerState {
386
+ input: string;
387
+ attachments: Attachment[];
388
+ isSubmitting: boolean;
389
+ }
390
+
391
+ interface ComposerActions {
392
+ update: (updater: (state: ComposerState) => ComposerState) => void;
393
+ submit: () => void;
394
+ }
395
+
396
+ interface ComposerMeta {
397
+ inputRef: React.RefObject<TextInput>;
398
+ }
399
+
400
+ interface ComposerContextValue {
401
+ state: ComposerState;
402
+ actions: ComposerActions;
403
+ meta: ComposerMeta;
404
+ }
405
+
406
+ const ComposerContext = createContext<ComposerContextValue | null>(null);
407
+ ```
408
+
409
+ **UI components consume the interface, not the implementation:**
410
+
411
+ ```tsx
412
+ function ComposerInput() {
413
+ const {
414
+ state,
415
+ actions: { update },
416
+ meta,
417
+ } = use(ComposerContext);
418
+
419
+ // This component works with ANY provider that implements the interface
420
+ return (
421
+ <TextInput
422
+ ref={meta.inputRef}
423
+ value={state.input}
424
+ onChangeText={(text) => update((s) => ({ ...s, input: text }))}
425
+ />
426
+ );
427
+ }
428
+ ```
429
+
430
+ **Different providers implement the same interface:**
431
+
432
+ ```tsx
433
+ // Provider A: Local state for ephemeral forms
434
+ function ForwardMessageProvider({ children }: { children: React.ReactNode }) {
435
+ const [state, setState] = useState(initialState);
436
+ const inputRef = useRef(null);
437
+ const submit = useForwardMessage();
438
+
439
+ return (
440
+ <ComposerContext
441
+ value={{
442
+ state,
443
+ actions: { update: setState, submit },
444
+ meta: { inputRef },
445
+ }}
446
+ >
447
+ {children}
448
+ </ComposerContext>
449
+ );
450
+ }
451
+
452
+ // Provider B: Global synced state for channels
453
+ function ChannelProvider({ channelId, children }: Props) {
454
+ const { state, update, submit } = useGlobalChannel(channelId);
455
+ const inputRef = useRef(null);
456
+
457
+ return (
458
+ <ComposerContext
459
+ value={{
460
+ state,
461
+ actions: { update, submit },
462
+ meta: { inputRef },
463
+ }}
464
+ >
465
+ {children}
466
+ </ComposerContext>
467
+ );
468
+ }
469
+ ```
470
+
471
+ **The same composed UI works with both:**
472
+
473
+ ```tsx
474
+ // Works with ForwardMessageProvider (local state)
475
+ <ForwardMessageProvider>
476
+ <Composer.Frame>
477
+ <Composer.Input />
478
+ <Composer.Submit />
479
+ </Composer.Frame>
480
+ </ForwardMessageProvider>
481
+
482
+ // Works with ChannelProvider (global synced state)
483
+ <ChannelProvider channelId="abc">
484
+ <Composer.Frame>
485
+ <Composer.Input />
486
+ <Composer.Submit />
487
+ </Composer.Frame>
488
+ </ChannelProvider>
489
+ ```
490
+
491
+ **Custom UI outside the component can access state and actions:**
492
+
493
+ ```tsx
494
+ function ForwardMessageDialog() {
495
+ return (
496
+ <ForwardMessageProvider>
497
+ <Dialog>
498
+ {/* The composer UI */}
499
+ <Composer.Frame>
500
+ <Composer.Input placeholder="Add a message, if you'd like." />
501
+ <Composer.Footer>
502
+ <Composer.Formatting />
503
+ <Composer.Emojis />
504
+ </Composer.Footer>
505
+ </Composer.Frame>
506
+
507
+ {/* Custom UI OUTSIDE the composer, but INSIDE the provider */}
508
+ <MessagePreview />
509
+
510
+ {/* Actions at the bottom of the dialog */}
511
+ <DialogActions>
512
+ <CancelButton />
513
+ <ForwardButton />
514
+ </DialogActions>
515
+ </Dialog>
516
+ </ForwardMessageProvider>
517
+ );
518
+ }
519
+
520
+ // This button lives OUTSIDE Composer.Frame but can still submit based on its context!
521
+ function ForwardButton() {
522
+ const {
523
+ actions: { submit },
524
+ } = use(ComposerContext);
525
+ return <Button onPress={submit}>Forward</Button>;
526
+ }
527
+
528
+ // This preview lives OUTSIDE Composer.Frame but can read composer's state!
529
+ function MessagePreview() {
530
+ const { state } = use(ComposerContext);
531
+ return <Preview message={state.input} attachments={state.attachments} />;
532
+ }
533
+ ```
534
+
535
+ The provider boundary is what matters—not the visual nesting. Components that
536
+
537
+ need shared state don't have to be inside the `Composer.Frame`. They just need
538
+
539
+ to be within the provider.
540
+
541
+ The `ForwardButton` and `MessagePreview` are not visually inside the composer
542
+
543
+ box, but they can still access its state and actions. This is the power of
544
+
545
+ lifting state into providers.
546
+
547
+ The UI is reusable bits you compose together. The state is dependency-injected
548
+
549
+ by the provider. Swap the provider, keep the UI.
550
+
551
+ ### 2.3 Lift State into Provider Components
552
+
553
+ **Impact: HIGH (enables state sharing outside component boundaries)**
554
+
555
+ Move state management into dedicated provider components. This allows sibling
556
+
557
+ components outside the main UI to access and modify state without prop drilling
558
+
559
+ or awkward refs.
560
+
561
+ **Incorrect: state trapped inside component**
562
+
563
+ ```tsx
564
+ function ForwardMessageComposer() {
565
+ const [state, setState] = useState(initialState);
566
+ const forwardMessage = useForwardMessage();
567
+
568
+ return (
569
+ <Composer.Frame>
570
+ <Composer.Input />
571
+ <Composer.Footer />
572
+ </Composer.Frame>
573
+ );
574
+ }
575
+
576
+ // Problem: How does this button access composer state?
577
+ function ForwardMessageDialog() {
578
+ return (
579
+ <Dialog>
580
+ <ForwardMessageComposer />
581
+ <MessagePreview /> {/* Needs composer state */}
582
+ <DialogActions>
583
+ <CancelButton />
584
+ <ForwardButton /> {/* Needs to call submit */}
585
+ </DialogActions>
586
+ </Dialog>
587
+ );
588
+ }
589
+ ```
590
+
591
+ **Incorrect: useEffect to sync state up**
592
+
593
+ ```tsx
594
+ function ForwardMessageDialog() {
595
+ const [input, setInput] = useState("");
596
+ return (
597
+ <Dialog>
598
+ <ForwardMessageComposer onInputChange={setInput} />
599
+ <MessagePreview input={input} />
600
+ </Dialog>
601
+ );
602
+ }
603
+
604
+ function ForwardMessageComposer({ onInputChange }) {
605
+ const [state, setState] = useState(initialState);
606
+ useEffect(() => {
607
+ onInputChange(state.input); // Sync on every change 😬
608
+ }, [state.input]);
609
+ }
610
+ ```
611
+
612
+ **Incorrect: reading state from ref on submit**
613
+
614
+ ```tsx
615
+ function ForwardMessageDialog() {
616
+ const stateRef = useRef(null);
617
+ return (
618
+ <Dialog>
619
+ <ForwardMessageComposer stateRef={stateRef} />
620
+ <ForwardButton onPress={() => submit(stateRef.current)} />
621
+ </Dialog>
622
+ );
623
+ }
624
+ ```
625
+
626
+ **Correct: state lifted to provider**
627
+
628
+ ```tsx
629
+ function ForwardMessageProvider({ children }: { children: React.ReactNode }) {
630
+ const [state, setState] = useState(initialState);
631
+ const forwardMessage = useForwardMessage();
632
+ const inputRef = useRef(null);
633
+
634
+ return (
635
+ <Composer.Provider
636
+ state={state}
637
+ actions={{ update: setState, submit: forwardMessage }}
638
+ meta={{ inputRef }}
639
+ >
640
+ {children}
641
+ </Composer.Provider>
642
+ );
643
+ }
644
+
645
+ function ForwardMessageDialog() {
646
+ return (
647
+ <ForwardMessageProvider>
648
+ <Dialog>
649
+ <ForwardMessageComposer />
650
+ <MessagePreview /> {/* Custom components can access state and actions */}
651
+ <DialogActions>
652
+ <CancelButton />
653
+ <ForwardButton /> {/* Custom components can access state and actions */}
654
+ </DialogActions>
655
+ </Dialog>
656
+ </ForwardMessageProvider>
657
+ );
658
+ }
659
+
660
+ function ForwardButton() {
661
+ const { actions } = use(Composer.Context);
662
+ return <Button onPress={actions.submit}>Forward</Button>;
663
+ }
664
+ ```
665
+
666
+ The ForwardButton lives outside the Composer.Frame but still has access to the
667
+
668
+ submit action because it's within the provider. Even though it's a one-off
669
+
670
+ component, it can still access the composer's state and actions from outside the
671
+
672
+ UI itself.
673
+
674
+ **Key insight:** Components that need shared state don't have to be visually
675
+
676
+ nested inside each other—they just need to be within the same provider.
677
+
678
+ ---
679
+
680
+ ## 3. Implementation Patterns
681
+
682
+ **Impact: MEDIUM**
683
+
684
+ Specific techniques for implementing compound components and
685
+ context providers.
686
+
687
+ ### 3.1 Create Explicit Component Variants
688
+
689
+ **Impact: MEDIUM (self-documenting code, no hidden conditionals)**
690
+
691
+ Instead of one component with many boolean props, create explicit variant
692
+
693
+ components. Each variant composes the pieces it needs. The code documents
694
+
695
+ itself.
696
+
697
+ **Incorrect: one component, many modes**
698
+
699
+ ```tsx
700
+ // What does this component actually render?
701
+ <Composer isThread isEditing={false} channelId="abc" showAttachments showFormatting={false} />
702
+ ```
703
+
704
+ **Correct: explicit variants**
705
+
706
+ ```tsx
707
+ // Immediately clear what this renders
708
+ <ThreadComposer channelId="abc" />
709
+
710
+ // Or
711
+ <EditMessageComposer messageId="xyz" />
712
+
713
+ // Or
714
+ <ForwardMessageComposer messageId="123" />
715
+ ```
716
+
717
+ Each implementation is unique, explicit and self-contained. Yet they can each
718
+
719
+ use shared parts.
720
+
721
+ **Implementation:**
722
+
723
+ ```tsx
724
+ function ThreadComposer({ channelId }: { channelId: string }) {
725
+ return (
726
+ <ThreadProvider channelId={channelId}>
727
+ <Composer.Frame>
728
+ <Composer.Input />
729
+ <AlsoSendToChannelField channelId={channelId} />
730
+ <Composer.Footer>
731
+ <Composer.Formatting />
732
+ <Composer.Emojis />
733
+ <Composer.Submit />
734
+ </Composer.Footer>
735
+ </Composer.Frame>
736
+ </ThreadProvider>
737
+ );
738
+ }
739
+
740
+ function EditMessageComposer({ messageId }: { messageId: string }) {
741
+ return (
742
+ <EditMessageProvider messageId={messageId}>
743
+ <Composer.Frame>
744
+ <Composer.Input />
745
+ <Composer.Footer>
746
+ <Composer.Formatting />
747
+ <Composer.Emojis />
748
+ <Composer.CancelEdit />
749
+ <Composer.SaveEdit />
750
+ </Composer.Footer>
751
+ </Composer.Frame>
752
+ </EditMessageProvider>
753
+ );
754
+ }
755
+
756
+ function ForwardMessageComposer({ messageId }: { messageId: string }) {
757
+ return (
758
+ <ForwardMessageProvider messageId={messageId}>
759
+ <Composer.Frame>
760
+ <Composer.Input placeholder="Add a message, if you'd like." />
761
+ <Composer.Footer>
762
+ <Composer.Formatting />
763
+ <Composer.Emojis />
764
+ <Composer.Mentions />
765
+ </Composer.Footer>
766
+ </Composer.Frame>
767
+ </ForwardMessageProvider>
768
+ );
769
+ }
770
+ ```
771
+
772
+ Each variant is explicit about:
773
+
774
+ - What provider/state it uses
775
+
776
+ - What UI elements it includes
777
+
778
+ - What actions are available
779
+
780
+ No boolean prop combinations to reason about. No impossible states.
781
+
782
+ ### 3.2 Prefer Composing Children Over Render Props
783
+
784
+ **Impact: MEDIUM (cleaner composition, better readability)**
785
+
786
+ Use `children` for composition instead of `renderX` props. Children are more
787
+
788
+ readable, compose naturally, and don't require understanding callback
789
+
790
+ signatures.
791
+
792
+ **Incorrect: render props**
793
+
794
+ ```tsx
795
+ function Composer({
796
+ renderHeader,
797
+ renderFooter,
798
+ renderActions,
799
+ }: {
800
+ renderHeader?: () => React.ReactNode;
801
+ renderFooter?: () => React.ReactNode;
802
+ renderActions?: () => React.ReactNode;
803
+ }) {
804
+ return (
805
+ <form>
806
+ {renderHeader?.()}
807
+ <Input />
808
+ {renderFooter ? renderFooter() : <DefaultFooter />}
809
+ {renderActions?.()}
810
+ </form>
811
+ );
812
+ }
813
+
814
+ // Usage is awkward and inflexible
815
+ return (
816
+ <Composer
817
+ renderHeader={() => <CustomHeader />}
818
+ renderFooter={() => (
819
+ <>
820
+ <Formatting />
821
+ <Emojis />
822
+ </>
823
+ )}
824
+ renderActions={() => <SubmitButton />}
825
+ />
826
+ );
827
+ ```
828
+
829
+ **Correct: compound components with children**
830
+
831
+ ```tsx
832
+ function ComposerFrame({ children }: { children: React.ReactNode }) {
833
+ return <form>{children}</form>;
834
+ }
835
+
836
+ function ComposerFooter({ children }: { children: React.ReactNode }) {
837
+ return <footer className="flex">{children}</footer>;
838
+ }
839
+
840
+ // Usage is flexible
841
+ return (
842
+ <Composer.Frame>
843
+ <CustomHeader />
844
+ <Composer.Input />
845
+ <Composer.Footer>
846
+ <Composer.Formatting />
847
+ <Composer.Emojis />
848
+ <SubmitButton />
849
+ </Composer.Footer>
850
+ </Composer.Frame>
851
+ );
852
+ ```
853
+
854
+ **When render props are appropriate:**
855
+
856
+ ```tsx
857
+ // Render props work well when you need to pass data back
858
+ <List data={items} renderItem={({ item, index }) => <Item item={item} index={index} />} />
859
+ ```
860
+
861
+ Use render props when the parent needs to provide data or state to the child.
862
+
863
+ Use children when composing static structure.
864
+
865
+ ---
866
+
867
+ ## 4. React 19 APIs
868
+
869
+ **Impact: MEDIUM**
870
+
871
+ React 19+ only. Don't use `forwardRef`; use `use()` instead of `useContext()`.
872
+
873
+ ### 4.1 React 19 API Changes
874
+
875
+ **Impact: MEDIUM (cleaner component definitions and context usage)**
876
+
877
+ > **⚠️ React 19+ only.** Skip this if you're on React 18 or earlier.
878
+
879
+ In React 19, `ref` is now a regular prop (no `forwardRef` wrapper needed), and `use()` replaces `useContext()`.
880
+
881
+ **Incorrect: forwardRef in React 19**
882
+
883
+ ```tsx
884
+ const ComposerInput = forwardRef<TextInput, Props>((props, ref) => {
885
+ return <TextInput ref={ref} {...props} />;
886
+ });
887
+ ```
888
+
889
+ **Correct: ref as a regular prop**
890
+
891
+ ```tsx
892
+ function ComposerInput({ ref, ...props }: Props & { ref?: React.Ref<TextInput> }) {
893
+ return <TextInput ref={ref} {...props} />;
894
+ }
895
+ ```
896
+
897
+ **Incorrect: useContext in React 19**
898
+
899
+ ```tsx
900
+ const value = useContext(MyContext);
901
+ ```
902
+
903
+ **Correct: use instead of useContext**
904
+
905
+ ```tsx
906
+ const value = use(MyContext);
907
+ ```
908
+
909
+ `use()` can also be called conditionally, unlike `useContext()`.
910
+
911
+ ---
912
+
913
+ ## References
914
+
915
+ 1. [https://react.dev](https://react.dev)
916
+ 2. [https://react.dev/learn/passing-data-deeply-with-context](https://react.dev/learn/passing-data-deeply-with-context)
917
+ 3. [https://react.dev/reference/react/use](https://react.dev/reference/react/use)