@better-t-stack/template-generator 3.13.2-dev.6c6cffb

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 (425) hide show
  1. package/dist/fs-writer.d.mts +8 -0
  2. package/dist/fs-writer.d.mts.map +1 -0
  3. package/dist/fs-writer.mjs +50 -0
  4. package/dist/fs-writer.mjs.map +1 -0
  5. package/dist/index.d.mts +182 -0
  6. package/dist/index.d.mts.map +1 -0
  7. package/dist/index.mjs +26727 -0
  8. package/dist/index.mjs.map +1 -0
  9. package/dist/template-reader.d.mts +30 -0
  10. package/dist/template-reader.d.mts.map +1 -0
  11. package/dist/template-reader.mjs +2860 -0
  12. package/dist/template-reader.mjs.map +1 -0
  13. package/dist/types-BubFnV2d.d.mts +36 -0
  14. package/dist/types-BubFnV2d.d.mts.map +1 -0
  15. package/package.json +66 -0
  16. package/scripts/generate-templates.ts +97 -0
  17. package/templates/addons/biome/biome.json.hbs +96 -0
  18. package/templates/addons/husky/.husky/pre-commit +1 -0
  19. package/templates/addons/pwa/apps/web/next/public/favicon/apple-touch-icon.png +0 -0
  20. package/templates/addons/pwa/apps/web/next/public/favicon/favicon-96x96.png +0 -0
  21. package/templates/addons/pwa/apps/web/next/public/favicon/favicon.svg +6 -0
  22. package/templates/addons/pwa/apps/web/next/public/favicon/site.webmanifest.hbs +21 -0
  23. package/templates/addons/pwa/apps/web/next/public/favicon/web-app-manifest-192x192.png +0 -0
  24. package/templates/addons/pwa/apps/web/next/public/favicon/web-app-manifest-512x512.png +0 -0
  25. package/templates/addons/pwa/apps/web/next/src/app/manifest.ts.hbs +26 -0
  26. package/templates/addons/pwa/apps/web/vite/public/logo.png +0 -0
  27. package/templates/addons/pwa/apps/web/vite/pwa-assets.config.ts.hbs +12 -0
  28. package/templates/addons/ruler/.ruler/bts.md.hbs +142 -0
  29. package/templates/addons/ruler/.ruler/ruler.toml.hbs +80 -0
  30. package/templates/addons/turborepo/turbo.json.hbs +74 -0
  31. package/templates/addons/ultracite/biome.json.hbs +26 -0
  32. package/templates/api/orpc/fullstack/next/src/app/api/rpc/[[...rest]]/route.ts.hbs +50 -0
  33. package/templates/api/orpc/fullstack/tanstack-start/src/routes/api/rpc/$.ts.hbs +58 -0
  34. package/templates/api/orpc/native/utils/orpc.ts.hbs +39 -0
  35. package/templates/api/orpc/server/_gitignore +34 -0
  36. package/templates/api/orpc/server/package.json.hbs +15 -0
  37. package/templates/api/orpc/server/src/context.ts.hbs +148 -0
  38. package/templates/api/orpc/server/src/index.ts.hbs +21 -0
  39. package/templates/api/orpc/server/src/routers/index.ts.hbs +55 -0
  40. package/templates/api/orpc/server/tsconfig.json.hbs +10 -0
  41. package/templates/api/orpc/web/nuxt/app/plugins/orpc.ts.hbs +32 -0
  42. package/templates/api/orpc/web/nuxt/app/plugins/vue-query.ts.hbs +44 -0
  43. package/templates/api/orpc/web/react/base/src/utils/orpc.ts.hbs +113 -0
  44. package/templates/api/orpc/web/solid/src/utils/orpc.ts.hbs +30 -0
  45. package/templates/api/orpc/web/svelte/src/lib/orpc.ts.hbs +30 -0
  46. package/templates/api/trpc/fullstack/next/src/app/api/trpc/[trpc]/route.ts.hbs +14 -0
  47. package/templates/api/trpc/fullstack/tanstack-start/src/routes/api/trpc/$.ts.hbs +22 -0
  48. package/templates/api/trpc/native/utils/trpc.ts.hbs +37 -0
  49. package/templates/api/trpc/server/_gitignore +34 -0
  50. package/templates/api/trpc/server/package.json.hbs +14 -0
  51. package/templates/api/trpc/server/src/context.ts.hbs +148 -0
  52. package/templates/api/trpc/server/src/index.ts.hbs +26 -0
  53. package/templates/api/trpc/server/src/routers/index.ts.hbs +55 -0
  54. package/templates/api/trpc/server/tsconfig.json.hbs +10 -0
  55. package/templates/api/trpc/web/react/base/src/utils/trpc.ts.hbs +97 -0
  56. package/templates/auth/better-auth/convex/backend/convex/auth.config.ts.hbs +6 -0
  57. package/templates/auth/better-auth/convex/backend/convex/auth.ts.hbs +68 -0
  58. package/templates/auth/better-auth/convex/backend/convex/http.ts.hbs +12 -0
  59. package/templates/auth/better-auth/convex/backend/convex/privateData.ts.hbs +17 -0
  60. package/templates/auth/better-auth/convex/native/bare/components/sign-in.tsx.hbs +127 -0
  61. package/templates/auth/better-auth/convex/native/bare/components/sign-up.tsx.hbs +138 -0
  62. package/templates/auth/better-auth/convex/native/base/lib/auth-client.ts.hbs +18 -0
  63. package/templates/auth/better-auth/convex/native/unistyles/components/sign-in.tsx.hbs +127 -0
  64. package/templates/auth/better-auth/convex/native/unistyles/components/sign-up.tsx.hbs +145 -0
  65. package/templates/auth/better-auth/convex/native/uniwind/components/sign-in.tsx.hbs +73 -0
  66. package/templates/auth/better-auth/convex/native/uniwind/components/sign-up.tsx.hbs +85 -0
  67. package/templates/auth/better-auth/convex/web/react/next/src/app/api/auth/[...all]/route.ts.hbs +3 -0
  68. package/templates/auth/better-auth/convex/web/react/next/src/app/dashboard/page.tsx.hbs +40 -0
  69. package/templates/auth/better-auth/convex/web/react/next/src/components/sign-in-form.tsx.hbs +129 -0
  70. package/templates/auth/better-auth/convex/web/react/next/src/components/sign-up-form.tsx.hbs +154 -0
  71. package/templates/auth/better-auth/convex/web/react/next/src/components/user-menu.tsx.hbs +48 -0
  72. package/templates/auth/better-auth/convex/web/react/next/src/lib/auth-client.ts.hbs +6 -0
  73. package/templates/auth/better-auth/convex/web/react/next/src/lib/auth-server.ts.hbs +16 -0
  74. package/templates/auth/better-auth/convex/web/react/tanstack-router/src/components/sign-in-form.tsx.hbs +133 -0
  75. package/templates/auth/better-auth/convex/web/react/tanstack-router/src/components/sign-up-form.tsx.hbs +158 -0
  76. package/templates/auth/better-auth/convex/web/react/tanstack-router/src/components/user-menu.tsx.hbs +52 -0
  77. package/templates/auth/better-auth/convex/web/react/tanstack-router/src/lib/auth-client.ts.hbs +11 -0
  78. package/templates/auth/better-auth/convex/web/react/tanstack-router/src/routes/dashboard.tsx.hbs +43 -0
  79. package/templates/auth/better-auth/convex/web/react/tanstack-start/src/components/sign-in-form.tsx.hbs +133 -0
  80. package/templates/auth/better-auth/convex/web/react/tanstack-start/src/components/sign-up-form.tsx.hbs +158 -0
  81. package/templates/auth/better-auth/convex/web/react/tanstack-start/src/components/user-menu.tsx.hbs +47 -0
  82. package/templates/auth/better-auth/convex/web/react/tanstack-start/src/lib/auth-client.ts.hbs +6 -0
  83. package/templates/auth/better-auth/convex/web/react/tanstack-start/src/lib/auth-server.ts.hbs +13 -0
  84. package/templates/auth/better-auth/convex/web/react/tanstack-start/src/routes/api/auth/$.ts.hbs +11 -0
  85. package/templates/auth/better-auth/convex/web/react/tanstack-start/src/routes/dashboard.tsx.hbs +43 -0
  86. package/templates/auth/better-auth/fullstack/next/src/app/api/auth/[...all]/route.ts.hbs +4 -0
  87. package/templates/auth/better-auth/fullstack/tanstack-start/src/routes/api/auth/$.ts.hbs +15 -0
  88. package/templates/auth/better-auth/native/bare/app/(drawer)/index.tsx.hbs +186 -0
  89. package/templates/auth/better-auth/native/bare/components/sign-in.tsx.hbs +131 -0
  90. package/templates/auth/better-auth/native/bare/components/sign-up.tsx.hbs +150 -0
  91. package/templates/auth/better-auth/native/base/lib/auth-client.ts.hbs +16 -0
  92. package/templates/auth/better-auth/native/unistyles/app/(drawer)/index.tsx.hbs +187 -0
  93. package/templates/auth/better-auth/native/unistyles/components/sign-in.tsx.hbs +139 -0
  94. package/templates/auth/better-auth/native/unistyles/components/sign-up.tsx.hbs +157 -0
  95. package/templates/auth/better-auth/native/uniwind/app/(drawer)/index.tsx.hbs +123 -0
  96. package/templates/auth/better-auth/native/uniwind/components/sign-in.tsx.hbs +87 -0
  97. package/templates/auth/better-auth/native/uniwind/components/sign-up.tsx.hbs +128 -0
  98. package/templates/auth/better-auth/server/base/_gitignore +34 -0
  99. package/templates/auth/better-auth/server/base/package.json.hbs +14 -0
  100. package/templates/auth/better-auth/server/base/src/index.ts.hbs +304 -0
  101. package/templates/auth/better-auth/server/base/tsconfig.json.hbs +10 -0
  102. package/templates/auth/better-auth/server/db/drizzle/mysql/src/schema/auth.ts.hbs +100 -0
  103. package/templates/auth/better-auth/server/db/drizzle/postgres/src/schema/auth.ts.hbs +93 -0
  104. package/templates/auth/better-auth/server/db/drizzle/sqlite/src/schema/auth.ts.hbs +107 -0
  105. package/templates/auth/better-auth/server/db/mongoose/mongodb/src/models/auth.model.ts.hbs +68 -0
  106. package/templates/auth/better-auth/server/db/prisma/mongodb/prisma/schema/auth.prisma.hbs +62 -0
  107. package/templates/auth/better-auth/server/db/prisma/mysql/prisma/schema/auth.prisma.hbs +62 -0
  108. package/templates/auth/better-auth/server/db/prisma/postgres/prisma/schema/auth.prisma.hbs +62 -0
  109. package/templates/auth/better-auth/server/db/prisma/sqlite/prisma/schema/auth.prisma.hbs +62 -0
  110. package/templates/auth/better-auth/web/nuxt/app/components/SignInForm.vue.hbs +82 -0
  111. package/templates/auth/better-auth/web/nuxt/app/components/SignUpForm.vue.hbs +91 -0
  112. package/templates/auth/better-auth/web/nuxt/app/components/UserMenu.vue.hbs +42 -0
  113. package/templates/auth/better-auth/web/nuxt/app/middleware/auth.ts.hbs +14 -0
  114. package/templates/auth/better-auth/web/nuxt/app/pages/dashboard.vue.hbs +99 -0
  115. package/templates/auth/better-auth/web/nuxt/app/pages/login.vue.hbs +27 -0
  116. package/templates/auth/better-auth/web/nuxt/app/plugins/auth-client.ts.hbs +21 -0
  117. package/templates/auth/better-auth/web/react/base/src/lib/auth-client.ts.hbs +16 -0
  118. package/templates/auth/better-auth/web/react/next/src/app/dashboard/dashboard.tsx.hbs +60 -0
  119. package/templates/auth/better-auth/web/react/next/src/app/dashboard/page.tsx.hbs +42 -0
  120. package/templates/auth/better-auth/web/react/next/src/app/login/page.tsx.hbs +16 -0
  121. package/templates/auth/better-auth/web/react/next/src/components/sign-in-form.tsx.hbs +135 -0
  122. package/templates/auth/better-auth/web/react/next/src/components/sign-up-form.tsx.hbs +160 -0
  123. package/templates/auth/better-auth/web/react/next/src/components/user-menu.tsx.hbs +62 -0
  124. package/templates/auth/better-auth/web/react/react-router/src/components/sign-in-form.tsx.hbs +135 -0
  125. package/templates/auth/better-auth/web/react/react-router/src/components/sign-up-form.tsx.hbs +160 -0
  126. package/templates/auth/better-auth/web/react/react-router/src/components/user-menu.tsx.hbs +61 -0
  127. package/templates/auth/better-auth/web/react/react-router/src/routes/dashboard.tsx.hbs +80 -0
  128. package/templates/auth/better-auth/web/react/react-router/src/routes/login.tsx.hbs +13 -0
  129. package/templates/auth/better-auth/web/react/tanstack-router/src/components/sign-in-form.tsx.hbs +135 -0
  130. package/templates/auth/better-auth/web/react/tanstack-router/src/components/sign-up-form.tsx.hbs +160 -0
  131. package/templates/auth/better-auth/web/react/tanstack-router/src/components/user-menu.tsx.hbs +63 -0
  132. package/templates/auth/better-auth/web/react/tanstack-router/src/routes/dashboard.tsx.hbs +71 -0
  133. package/templates/auth/better-auth/web/react/tanstack-router/src/routes/login.tsx.hbs +18 -0
  134. package/templates/auth/better-auth/web/react/tanstack-start/src/components/sign-in-form.tsx.hbs +135 -0
  135. package/templates/auth/better-auth/web/react/tanstack-start/src/components/sign-up-form.tsx.hbs +160 -0
  136. package/templates/auth/better-auth/web/react/tanstack-start/src/components/user-menu.tsx.hbs +63 -0
  137. package/templates/auth/better-auth/web/react/tanstack-start/src/functions/get-user.ts.hbs +6 -0
  138. package/templates/auth/better-auth/web/react/tanstack-start/src/middleware/auth.ts.hbs +31 -0
  139. package/templates/auth/better-auth/web/react/tanstack-start/src/routes/dashboard.tsx.hbs +84 -0
  140. package/templates/auth/better-auth/web/react/tanstack-start/src/routes/login.tsx.hbs +18 -0
  141. package/templates/auth/better-auth/web/solid/src/components/sign-in-form.tsx.hbs +124 -0
  142. package/templates/auth/better-auth/web/solid/src/components/sign-up-form.tsx.hbs +148 -0
  143. package/templates/auth/better-auth/web/solid/src/components/user-menu.tsx.hbs +55 -0
  144. package/templates/auth/better-auth/web/solid/src/lib/auth-client.ts.hbs +12 -0
  145. package/templates/auth/better-auth/web/solid/src/routes/dashboard.tsx.hbs +67 -0
  146. package/templates/auth/better-auth/web/solid/src/routes/login.tsx.hbs +23 -0
  147. package/templates/auth/better-auth/web/svelte/src/components/SignInForm.svelte.hbs +109 -0
  148. package/templates/auth/better-auth/web/svelte/src/components/SignUpForm.svelte.hbs +142 -0
  149. package/templates/auth/better-auth/web/svelte/src/components/UserMenu.svelte.hbs +52 -0
  150. package/templates/auth/better-auth/web/svelte/src/lib/auth-client.ts.hbs +12 -0
  151. package/templates/auth/better-auth/web/svelte/src/routes/dashboard/+page.svelte.hbs +59 -0
  152. package/templates/auth/better-auth/web/svelte/src/routes/login/+page.svelte.hbs +12 -0
  153. package/templates/auth/clerk/convex/backend/convex/auth.config.ts.hbs +12 -0
  154. package/templates/auth/clerk/convex/backend/convex/privateData.ts.hbs +16 -0
  155. package/templates/auth/clerk/convex/native/base/app/(auth)/_layout.tsx.hbs +12 -0
  156. package/templates/auth/clerk/convex/native/base/app/(auth)/sign-in.tsx.hbs +67 -0
  157. package/templates/auth/clerk/convex/native/base/app/(auth)/sign-up.tsx.hbs +110 -0
  158. package/templates/auth/clerk/convex/native/base/components/sign-out-button.tsx.hbs +27 -0
  159. package/templates/auth/clerk/convex/web/react/next/src/app/dashboard/page.tsx.hbs +29 -0
  160. package/templates/auth/clerk/convex/web/react/next/src/middleware.ts.hbs +12 -0
  161. package/templates/auth/clerk/convex/web/react/react-router/src/routes/dashboard.tsx.hbs +32 -0
  162. package/templates/auth/clerk/convex/web/react/tanstack-router/src/routes/dashboard.tsx.hbs +37 -0
  163. package/templates/auth/clerk/convex/web/react/tanstack-start/src/routes/dashboard.tsx.hbs +37 -0
  164. package/templates/auth/clerk/convex/web/react/tanstack-start/src/start.ts.hbs +8 -0
  165. package/templates/backend/convex/packages/backend/_gitignore +2 -0
  166. package/templates/backend/convex/packages/backend/convex/README.md +90 -0
  167. package/templates/backend/convex/packages/backend/convex/convex.config.ts.hbs +17 -0
  168. package/templates/backend/convex/packages/backend/convex/healthCheck.ts.hbs +7 -0
  169. package/templates/backend/convex/packages/backend/convex/schema.ts.hbs +11 -0
  170. package/templates/backend/convex/packages/backend/convex/tsconfig.json.hbs +25 -0
  171. package/templates/backend/convex/packages/backend/package.json.hbs +15 -0
  172. package/templates/backend/server/base/_gitignore +55 -0
  173. package/templates/backend/server/base/package.json.hbs +17 -0
  174. package/templates/backend/server/base/tsconfig.json.hbs +13 -0
  175. package/templates/backend/server/base/tsdown.config.ts.hbs +9 -0
  176. package/templates/backend/server/elysia/src/index.ts.hbs +122 -0
  177. package/templates/backend/server/express/src/index.ts.hbs +126 -0
  178. package/templates/backend/server/fastify/src/index.ts.hbs +187 -0
  179. package/templates/backend/server/hono/src/index.ts.hbs +171 -0
  180. package/templates/base/_gitignore +50 -0
  181. package/templates/base/package.json.hbs +10 -0
  182. package/templates/base/tsconfig.json.hbs +3 -0
  183. package/templates/db/base/_gitignore +35 -0
  184. package/templates/db/base/package.json.hbs +14 -0
  185. package/templates/db/base/tsconfig.json.hbs +10 -0
  186. package/templates/db/drizzle/base/src/schema/index.ts.hbs +7 -0
  187. package/templates/db/drizzle/mysql/drizzle.config.ts.hbs +19 -0
  188. package/templates/db/drizzle/mysql/src/index.ts.hbs +54 -0
  189. package/templates/db/drizzle/postgres/drizzle.config.ts.hbs +19 -0
  190. package/templates/db/drizzle/postgres/src/index.ts.hbs +44 -0
  191. package/templates/db/drizzle/sqlite/drizzle.config.ts.hbs +28 -0
  192. package/templates/db/drizzle/sqlite/src/index.ts.hbs +39 -0
  193. package/templates/db/mongoose/mongodb/src/index.ts.hbs +10 -0
  194. package/templates/db/prisma/mongodb/prisma/schema/schema.prisma.hbs +19 -0
  195. package/templates/db/prisma/mongodb/prisma.config.ts.hbs +18 -0
  196. package/templates/db/prisma/mongodb/src/index.ts.hbs +5 -0
  197. package/templates/db/prisma/mysql/prisma/schema/schema.prisma.hbs +21 -0
  198. package/templates/db/prisma/mysql/prisma.config.ts.hbs +21 -0
  199. package/templates/db/prisma/mysql/src/index.ts.hbs +55 -0
  200. package/templates/db/prisma/postgres/prisma/schema/schema.prisma.hbs +21 -0
  201. package/templates/db/prisma/postgres/prisma.config.ts.hbs +21 -0
  202. package/templates/db/prisma/postgres/src/index.ts.hbs +69 -0
  203. package/templates/db/prisma/sqlite/prisma/schema/schema.prisma.hbs +18 -0
  204. package/templates/db/prisma/sqlite/prisma.config.ts.hbs +25 -0
  205. package/templates/db/prisma/sqlite/src/index.ts.hbs +25 -0
  206. package/templates/db-setup/docker-compose/mongodb/docker-compose.yml.hbs +23 -0
  207. package/templates/db-setup/docker-compose/mysql/docker-compose.yml.hbs +24 -0
  208. package/templates/db-setup/docker-compose/postgres/docker-compose.yml.hbs +23 -0
  209. package/templates/examples/ai/convex/packages/backend/convex/agent.ts.hbs +9 -0
  210. package/templates/examples/ai/convex/packages/backend/convex/chat.ts.hbs +67 -0
  211. package/templates/examples/ai/fullstack/next/src/app/api/ai/route.ts.hbs +20 -0
  212. package/templates/examples/ai/fullstack/tanstack-start/src/routes/api/ai/$.ts.hbs +36 -0
  213. package/templates/examples/ai/native/bare/app/(drawer)/ai.tsx.hbs +586 -0
  214. package/templates/examples/ai/native/bare/polyfills.js +22 -0
  215. package/templates/examples/ai/native/unistyles/app/(drawer)/ai.tsx.hbs +588 -0
  216. package/templates/examples/ai/native/unistyles/polyfills.js +22 -0
  217. package/templates/examples/ai/native/uniwind/app/(drawer)/ai.tsx.hbs +331 -0
  218. package/templates/examples/ai/native/uniwind/polyfills.js +22 -0
  219. package/templates/examples/ai/web/nuxt/app/pages/ai.vue.hbs +54 -0
  220. package/templates/examples/ai/web/react/next/src/app/ai/page.tsx.hbs +267 -0
  221. package/templates/examples/ai/web/react/react-router/src/routes/ai.tsx.hbs +235 -0
  222. package/templates/examples/ai/web/react/tanstack-router/src/routes/ai.tsx.hbs +242 -0
  223. package/templates/examples/ai/web/react/tanstack-start/src/routes/ai.tsx.hbs +243 -0
  224. package/templates/examples/ai/web/svelte/src/routes/ai/+page.svelte.hbs +107 -0
  225. package/templates/examples/todo/convex/packages/backend/convex/todos.ts.hbs +42 -0
  226. package/templates/examples/todo/native/bare/app/(drawer)/todos.tsx.hbs +521 -0
  227. package/templates/examples/todo/native/unistyles/app/(drawer)/todos.tsx.hbs +340 -0
  228. package/templates/examples/todo/native/uniwind/app/(drawer)/todos.tsx.hbs +282 -0
  229. package/templates/examples/todo/server/drizzle/base/src/routers/todo.ts.hbs +75 -0
  230. package/templates/examples/todo/server/drizzle/mysql/src/schema/todo.ts +7 -0
  231. package/templates/examples/todo/server/drizzle/postgres/src/schema/todo.ts +7 -0
  232. package/templates/examples/todo/server/drizzle/sqlite/src/schema/todo.ts +7 -0
  233. package/templates/examples/todo/server/mongoose/base/src/routers/todo.ts.hbs +66 -0
  234. package/templates/examples/todo/server/mongoose/mongodb/src/models/todo.model.ts.hbs +24 -0
  235. package/templates/examples/todo/server/prisma/base/src/routers/todo.ts.hbs +116 -0
  236. package/templates/examples/todo/server/prisma/mongodb/prisma/schema/todo.prisma.hbs +7 -0
  237. package/templates/examples/todo/server/prisma/mysql/prisma/schema/todo.prisma.hbs +7 -0
  238. package/templates/examples/todo/server/prisma/postgres/prisma/schema/todo.prisma.hbs +7 -0
  239. package/templates/examples/todo/server/prisma/sqlite/prisma/schema/todo.prisma.hbs +7 -0
  240. package/templates/examples/todo/web/nuxt/app/pages/todos.vue.hbs +220 -0
  241. package/templates/examples/todo/web/react/next/src/app/todos/page.tsx.hbs +245 -0
  242. package/templates/examples/todo/web/react/react-router/src/routes/todos.tsx.hbs +242 -0
  243. package/templates/examples/todo/web/react/tanstack-router/src/routes/todos.tsx.hbs +247 -0
  244. package/templates/examples/todo/web/react/tanstack-start/src/routes/todos.tsx.hbs +272 -0
  245. package/templates/examples/todo/web/solid/src/routes/todos.tsx.hbs +132 -0
  246. package/templates/examples/todo/web/svelte/src/routes/todos/+page.svelte.hbs +317 -0
  247. package/templates/extras/_npmrc.hbs +5 -0
  248. package/templates/extras/bunfig.toml.hbs +6 -0
  249. package/templates/extras/pnpm-workspace.yaml +3 -0
  250. package/templates/frontend/native/bare/_gitignore +18 -0
  251. package/templates/frontend/native/bare/app/(drawer)/(tabs)/_layout.tsx.hbs +41 -0
  252. package/templates/frontend/native/bare/app/(drawer)/(tabs)/index.tsx.hbs +43 -0
  253. package/templates/frontend/native/bare/app/(drawer)/(tabs)/two.tsx.hbs +43 -0
  254. package/templates/frontend/native/bare/app/(drawer)/_layout.tsx.hbs +90 -0
  255. package/templates/frontend/native/bare/app/(drawer)/index.tsx.hbs +234 -0
  256. package/templates/frontend/native/bare/app/+not-found.tsx.hbs +65 -0
  257. package/templates/frontend/native/bare/app/_layout.tsx.hbs +165 -0
  258. package/templates/frontend/native/bare/app/modal.tsx.hbs +34 -0
  259. package/templates/frontend/native/bare/app.json.hbs +50 -0
  260. package/templates/frontend/native/bare/components/container.tsx.hbs +25 -0
  261. package/templates/frontend/native/bare/components/header-button.tsx.hbs +47 -0
  262. package/templates/frontend/native/bare/components/tabbar-icon.tsx.hbs +9 -0
  263. package/templates/frontend/native/bare/lib/android-navigation-bar.tsx.hbs +12 -0
  264. package/templates/frontend/native/bare/lib/constants.ts.hbs +19 -0
  265. package/templates/frontend/native/bare/lib/use-color-scheme.ts.hbs +20 -0
  266. package/templates/frontend/native/bare/metro.config.js.hbs +9 -0
  267. package/templates/frontend/native/bare/package.json.hbs +51 -0
  268. package/templates/frontend/native/bare/tsconfig.json.hbs +11 -0
  269. package/templates/frontend/native/base/assets/images/android-icon-background.png +0 -0
  270. package/templates/frontend/native/base/assets/images/android-icon-foreground.png +0 -0
  271. package/templates/frontend/native/base/assets/images/android-icon-monochrome.png +0 -0
  272. package/templates/frontend/native/base/assets/images/favicon.png +0 -0
  273. package/templates/frontend/native/base/assets/images/icon.png +0 -0
  274. package/templates/frontend/native/base/assets/images/partial-react-logo.png +0 -0
  275. package/templates/frontend/native/base/assets/images/react-logo.png +0 -0
  276. package/templates/frontend/native/base/assets/images/react-logo@2x.png +0 -0
  277. package/templates/frontend/native/base/assets/images/react-logo@3x.png +0 -0
  278. package/templates/frontend/native/base/assets/images/splash-icon.png +0 -0
  279. package/templates/frontend/native/unistyles/_gitignore +24 -0
  280. package/templates/frontend/native/unistyles/app/(drawer)/(tabs)/_layout.tsx.hbs +39 -0
  281. package/templates/frontend/native/unistyles/app/(drawer)/(tabs)/index.tsx.hbs +37 -0
  282. package/templates/frontend/native/unistyles/app/(drawer)/(tabs)/two.tsx.hbs +37 -0
  283. package/templates/frontend/native/unistyles/app/(drawer)/_layout.tsx.hbs +87 -0
  284. package/templates/frontend/native/unistyles/app/(drawer)/index.tsx.hbs +333 -0
  285. package/templates/frontend/native/unistyles/app/+not-found.tsx.hbs +65 -0
  286. package/templates/frontend/native/unistyles/app/_layout.tsx.hbs +169 -0
  287. package/templates/frontend/native/unistyles/app/modal.tsx.hbs +33 -0
  288. package/templates/frontend/native/unistyles/app.json.hbs +49 -0
  289. package/templates/frontend/native/unistyles/babel.config.js.hbs +21 -0
  290. package/templates/frontend/native/unistyles/breakpoints.ts.hbs +9 -0
  291. package/templates/frontend/native/unistyles/components/container.tsx.hbs +15 -0
  292. package/templates/frontend/native/unistyles/components/header-button.tsx.hbs +36 -0
  293. package/templates/frontend/native/unistyles/components/tabbar-icon.tsx.hbs +8 -0
  294. package/templates/frontend/native/unistyles/index.js.hbs +2 -0
  295. package/templates/frontend/native/unistyles/metro.config.js.hbs +5 -0
  296. package/templates/frontend/native/unistyles/package.json.hbs +51 -0
  297. package/templates/frontend/native/unistyles/theme.ts.hbs +98 -0
  298. package/templates/frontend/native/unistyles/tsconfig.json.hbs +12 -0
  299. package/templates/frontend/native/unistyles/unistyles.ts.hbs +27 -0
  300. package/templates/frontend/native/uniwind/_gitignore +21 -0
  301. package/templates/frontend/native/uniwind/app/(drawer)/(tabs)/_layout.tsx.hbs +46 -0
  302. package/templates/frontend/native/uniwind/app/(drawer)/(tabs)/index.tsx.hbs +15 -0
  303. package/templates/frontend/native/uniwind/app/(drawer)/(tabs)/two.tsx.hbs +15 -0
  304. package/templates/frontend/native/uniwind/app/(drawer)/_layout.tsx.hbs +91 -0
  305. package/templates/frontend/native/uniwind/app/(drawer)/index.tsx.hbs +191 -0
  306. package/templates/frontend/native/uniwind/app/+not-found.tsx.hbs +27 -0
  307. package/templates/frontend/native/uniwind/app/_layout.tsx.hbs +132 -0
  308. package/templates/frontend/native/uniwind/app/modal.tsx.hbs +37 -0
  309. package/templates/frontend/native/uniwind/app.json.hbs +19 -0
  310. package/templates/frontend/native/uniwind/components/container.tsx.hbs +33 -0
  311. package/templates/frontend/native/uniwind/components/theme-toggle.tsx.hbs +35 -0
  312. package/templates/frontend/native/uniwind/contexts/app-theme-context.tsx.hbs +62 -0
  313. package/templates/frontend/native/uniwind/global.css +5 -0
  314. package/templates/frontend/native/uniwind/metro.config.js.hbs +13 -0
  315. package/templates/frontend/native/uniwind/package.json.hbs +54 -0
  316. package/templates/frontend/native/uniwind/tsconfig.json.hbs +14 -0
  317. package/templates/frontend/nuxt/_gitignore +27 -0
  318. package/templates/frontend/nuxt/app/app.config.ts.hbs +15 -0
  319. package/templates/frontend/nuxt/app/app.vue.hbs +17 -0
  320. package/templates/frontend/nuxt/app/assets/css/main.css +2 -0
  321. package/templates/frontend/nuxt/app/components/Header.vue.hbs +40 -0
  322. package/templates/frontend/nuxt/app/layouts/default.vue.hbs +10 -0
  323. package/templates/frontend/nuxt/app/pages/index.vue.hbs +97 -0
  324. package/templates/frontend/nuxt/nuxt.config.ts.hbs +29 -0
  325. package/templates/frontend/nuxt/package.json.hbs +24 -0
  326. package/templates/frontend/nuxt/public/favicon.ico +0 -0
  327. package/templates/frontend/nuxt/public/robots.txt +2 -0
  328. package/templates/frontend/nuxt/server/tsconfig.json +3 -0
  329. package/templates/frontend/nuxt/tsconfig.json.hbs +18 -0
  330. package/templates/frontend/react/next/next-env.d.ts.hbs +5 -0
  331. package/templates/frontend/react/next/next.config.ts.hbs +22 -0
  332. package/templates/frontend/react/next/package.json.hbs +34 -0
  333. package/templates/frontend/react/next/postcss.config.mjs.hbs +5 -0
  334. package/templates/frontend/react/next/src/app/favicon.ico +0 -0
  335. package/templates/frontend/react/next/src/app/layout.tsx.hbs +76 -0
  336. package/templates/frontend/react/next/src/app/page.tsx.hbs +79 -0
  337. package/templates/frontend/react/next/src/components/mode-toggle.tsx.hbs +37 -0
  338. package/templates/frontend/react/next/src/components/providers.tsx.hbs +89 -0
  339. package/templates/frontend/react/next/src/components/theme-provider.tsx.hbs +11 -0
  340. package/templates/frontend/react/next/tsconfig.json.hbs +41 -0
  341. package/templates/frontend/react/react-router/package.json.hbs +42 -0
  342. package/templates/frontend/react/react-router/public/favicon.ico +0 -0
  343. package/templates/frontend/react/react-router/react-router.config.ts +6 -0
  344. package/templates/frontend/react/react-router/src/components/mode-toggle.tsx.hbs +29 -0
  345. package/templates/frontend/react/react-router/src/components/theme-provider.tsx.hbs +11 -0
  346. package/templates/frontend/react/react-router/src/root.tsx.hbs +190 -0
  347. package/templates/frontend/react/react-router/src/routes/_index.tsx.hbs +85 -0
  348. package/templates/frontend/react/react-router/src/routes.ts +4 -0
  349. package/templates/frontend/react/react-router/tsconfig.json.hbs +27 -0
  350. package/templates/frontend/react/react-router/vite.config.ts.hbs +12 -0
  351. package/templates/frontend/react/tanstack-router/index.html.hbs +13 -0
  352. package/templates/frontend/react/tanstack-router/package.json.hbs +41 -0
  353. package/templates/frontend/react/tanstack-router/src/components/mode-toggle.tsx.hbs +29 -0
  354. package/templates/frontend/react/tanstack-router/src/components/theme-provider.tsx.hbs +11 -0
  355. package/templates/frontend/react/tanstack-router/src/main.tsx.hbs +90 -0
  356. package/templates/frontend/react/tanstack-router/src/routes/__root.tsx.hbs +103 -0
  357. package/templates/frontend/react/tanstack-router/src/routes/index.tsx.hbs +85 -0
  358. package/templates/frontend/react/tanstack-router/tsconfig.json.hbs +18 -0
  359. package/templates/frontend/react/tanstack-router/vite.config.ts.hbs +21 -0
  360. package/templates/frontend/react/tanstack-start/package.json.hbs +43 -0
  361. package/templates/frontend/react/tanstack-start/public/robots.txt +3 -0
  362. package/templates/frontend/react/tanstack-start/src/router.tsx.hbs +144 -0
  363. package/templates/frontend/react/tanstack-start/src/routes/__root.tsx.hbs +208 -0
  364. package/templates/frontend/react/tanstack-start/src/routes/index.tsx.hbs +85 -0
  365. package/templates/frontend/react/tanstack-start/tsconfig.json.hbs +28 -0
  366. package/templates/frontend/react/tanstack-start/vite.config.ts.hbs +22 -0
  367. package/templates/frontend/react/web-base/_gitignore +60 -0
  368. package/templates/frontend/react/web-base/components.json +24 -0
  369. package/templates/frontend/react/web-base/src/components/header.tsx.hbs +78 -0
  370. package/templates/frontend/react/web-base/src/components/loader.tsx.hbs +9 -0
  371. package/templates/frontend/react/web-base/src/components/ui/button.tsx.hbs +57 -0
  372. package/templates/frontend/react/web-base/src/components/ui/card.tsx.hbs +103 -0
  373. package/templates/frontend/react/web-base/src/components/ui/checkbox.tsx.hbs +26 -0
  374. package/templates/frontend/react/web-base/src/components/ui/dropdown-menu.tsx.hbs +262 -0
  375. package/templates/frontend/react/web-base/src/components/ui/input.tsx.hbs +20 -0
  376. package/templates/frontend/react/web-base/src/components/ui/label.tsx.hbs +20 -0
  377. package/templates/frontend/react/web-base/src/components/ui/skeleton.tsx.hbs +13 -0
  378. package/templates/frontend/react/web-base/src/components/ui/sonner.tsx.hbs +44 -0
  379. package/templates/frontend/react/web-base/src/index.css.hbs +131 -0
  380. package/templates/frontend/react/web-base/src/lib/utils.ts.hbs +6 -0
  381. package/templates/frontend/solid/_gitignore +11 -0
  382. package/templates/frontend/solid/index.html +13 -0
  383. package/templates/frontend/solid/package.json.hbs +24 -0
  384. package/templates/frontend/solid/public/robots.txt +3 -0
  385. package/templates/frontend/solid/src/components/header.tsx.hbs +38 -0
  386. package/templates/frontend/solid/src/components/loader.tsx +9 -0
  387. package/templates/frontend/solid/src/main.tsx.hbs +41 -0
  388. package/templates/frontend/solid/src/routes/__root.tsx.hbs +34 -0
  389. package/templates/frontend/solid/src/routes/index.tsx.hbs +72 -0
  390. package/templates/frontend/solid/src/styles.css +5 -0
  391. package/templates/frontend/solid/tsconfig.json.hbs +29 -0
  392. package/templates/frontend/solid/vite.config.ts.hbs +21 -0
  393. package/templates/frontend/svelte/_gitignore +24 -0
  394. package/templates/frontend/svelte/_npmrc +1 -0
  395. package/templates/frontend/svelte/package.json.hbs +27 -0
  396. package/templates/frontend/svelte/src/app.css +5 -0
  397. package/templates/frontend/svelte/src/app.d.ts +13 -0
  398. package/templates/frontend/svelte/src/app.html +12 -0
  399. package/templates/frontend/svelte/src/components/Header.svelte.hbs +40 -0
  400. package/templates/frontend/svelte/src/lib/index.ts +2 -0
  401. package/templates/frontend/svelte/src/routes/+layout.svelte.hbs +54 -0
  402. package/templates/frontend/svelte/src/routes/+page.svelte.hbs +92 -0
  403. package/templates/frontend/svelte/static/favicon.png +0 -0
  404. package/templates/frontend/svelte/svelte.config.js.hbs +18 -0
  405. package/templates/frontend/svelte/tsconfig.json.hbs +19 -0
  406. package/templates/frontend/svelte/vite.config.ts.hbs +7 -0
  407. package/templates/packages/config/package.json.hbs +5 -0
  408. package/templates/packages/config/tsconfig.base.json.hbs +33 -0
  409. package/templates/packages/env/env.d.ts.hbs +16 -0
  410. package/templates/packages/env/package.json.hbs +7 -0
  411. package/templates/packages/env/src/native.ts.hbs +21 -0
  412. package/templates/packages/env/src/server.ts.hbs +39 -0
  413. package/templates/packages/env/src/web.ts.hbs +98 -0
  414. package/templates/packages/env/tsconfig.json.hbs +3 -0
  415. package/templates/packages/infra/alchemy.run.ts.hbs +271 -0
  416. package/templates/packages/infra/package.json.hbs +10 -0
  417. package/templates/payments/polar/server/base/src/lib/payments.ts.hbs +7 -0
  418. package/templates/payments/polar/web/nuxt/app/pages/success.vue.hbs +11 -0
  419. package/templates/payments/polar/web/react/next/src/app/success/page.tsx.hbs +15 -0
  420. package/templates/payments/polar/web/react/react-router/src/routes/success.tsx.hbs +13 -0
  421. package/templates/payments/polar/web/react/tanstack-router/src/routes/success.tsx.hbs +19 -0
  422. package/templates/payments/polar/web/react/tanstack-start/src/functions/get-payment.ts.hbs +15 -0
  423. package/templates/payments/polar/web/react/tanstack-start/src/routes/success.tsx.hbs +19 -0
  424. package/templates/payments/polar/web/solid/src/routes/success.tsx.hbs +23 -0
  425. package/templates/payments/polar/web/svelte/src/routes/success/+page.svelte.hbs +12 -0
@@ -0,0 +1,186 @@
1
+ import { View, Text, ScrollView, TouchableOpacity, StyleSheet } from "react-native";
2
+ import { Container } from "@/components/container";
3
+ import { useColorScheme } from "@/lib/use-color-scheme";
4
+ import { NAV_THEME } from "@/lib/constants";
5
+ import { authClient } from "@/lib/auth-client";
6
+ import { SignIn } from "@/components/sign-in";
7
+ import { SignUp } from "@/components/sign-up";
8
+ {{#if (eq api "orpc")}}
9
+ import { useQuery } from "@tanstack/react-query";
10
+ import { queryClient, orpc } from "@/utils/orpc";
11
+ {{/if}}
12
+ {{#if (eq api "trpc")}}
13
+ import { useQuery } from "@tanstack/react-query";
14
+ import { queryClient, trpc } from "@/utils/trpc";
15
+ {{/if}}
16
+
17
+ export default function Home() {
18
+ const { colorScheme } = useColorScheme();
19
+ const theme = colorScheme === "dark" ? NAV_THEME.dark : NAV_THEME.light;
20
+ {{#if (eq api "orpc")}}
21
+ const healthCheck = useQuery(orpc.healthCheck.queryOptions());
22
+ const privateData = useQuery(orpc.privateData.queryOptions());
23
+ const isConnected = healthCheck?.data === "OK";
24
+ const isLoading = healthCheck?.isLoading;
25
+ {{/if}}
26
+ {{#if (eq api "trpc")}}
27
+ const healthCheck = useQuery(trpc.healthCheck.queryOptions());
28
+ const privateData = useQuery(trpc.privateData.queryOptions());
29
+ const isConnected = healthCheck?.data === "OK";
30
+ const isLoading = healthCheck?.isLoading;
31
+ {{/if}}
32
+ const { data: session } = authClient.useSession();
33
+
34
+ return (
35
+ <Container>
36
+ <ScrollView style={styles.scrollView}>
37
+ <View style={styles.content}>
38
+ <Text style={[styles.title, { color: theme.text }]}>
39
+ BETTER T STACK
40
+ </Text>
41
+
42
+ {session?.user ? (
43
+ <View style={[styles.userCard, { backgroundColor: theme.card, borderColor: theme.border }]}>
44
+ <View style={styles.userHeader}>
45
+ <Text style={[styles.userText, { color: theme.text }]}>
46
+ Welcome, <Text style={styles.userName}>{session.user.name}</Text>
47
+ </Text>
48
+ </View>
49
+ <Text style={[styles.userEmail, { color: theme.text, opacity: 0.7 }]}>
50
+ {session.user.email}
51
+ </Text>
52
+ <TouchableOpacity style={[styles.signOutButton, { backgroundColor: theme.notification }]} onPress={()=> {
53
+ authClient.signOut();
54
+ {{#if (eq api "orpc")}}
55
+ queryClient.invalidateQueries();
56
+ {{/if}}
57
+ {{#if (eq api "trpc")}}
58
+ queryClient.invalidateQueries();
59
+ {{/if}}
60
+ }}
61
+ >
62
+ <Text style={styles.signOutText}>Sign Out</Text>
63
+ </TouchableOpacity>
64
+ </View>
65
+ ) : null}
66
+
67
+ {{#unless (eq api "none")}}
68
+ <View style={[styles.statusCard, { backgroundColor: theme.card, borderColor: theme.border }]}>
69
+ <Text style={[styles.cardTitle, { color: theme.text }]}>
70
+ System Status
71
+ </Text>
72
+ <View style={styles.statusRow}>
73
+ <View style={[styles.statusIndicator, { backgroundColor: isConnected ? "#10b981" : "#ef4444" }]} />
74
+ <View style={styles.statusContent}>
75
+ <Text style={[styles.statusTitle, { color: theme.text }]}>
76
+ {{#if (eq api "orpc")}}ORPC{{else}}TRPC{{/if}} Backend
77
+ </Text>
78
+ <Text style={[styles.statusText, { color: theme.text, opacity: 0.7 }]}>
79
+ {isLoading
80
+ ? "Checking connection..."
81
+ : isConnected
82
+ ? "Connected to API"
83
+ : "API Disconnected"}
84
+ </Text>
85
+ </View>
86
+ </View>
87
+ </View>
88
+
89
+ <View style={[styles.privateDataCard, { backgroundColor: theme.card, borderColor: theme.border }]}>
90
+ <Text style={[styles.cardTitle, { color: theme.text }]}>
91
+ Private Data
92
+ </Text>
93
+ {privateData && (
94
+ <Text style={[styles.privateDataText, { color: theme.text, opacity: 0.7 }]}>
95
+ {privateData.data?.message}
96
+ </Text>
97
+ )}
98
+ </View>
99
+ {{/unless}}
100
+
101
+ {!session?.user && (
102
+ <>
103
+ <SignIn />
104
+ <SignUp />
105
+ </>
106
+ )}
107
+ </View>
108
+ </ScrollView>
109
+ </Container>
110
+ );
111
+ }
112
+
113
+ const styles = StyleSheet.create({
114
+ scrollView: {
115
+ flex: 1,
116
+ },
117
+ content: {
118
+ padding: 16,
119
+ },
120
+ title: {
121
+ fontSize: 24,
122
+ fontWeight: "bold",
123
+ marginBottom: 16,
124
+ },
125
+ userCard: {
126
+ marginBottom: 16,
127
+ padding: 16,
128
+ borderWidth: 1,
129
+ },
130
+ userHeader: {
131
+ marginBottom: 8,
132
+ },
133
+ userText: {
134
+ fontSize: 16,
135
+ },
136
+ userName: {
137
+ fontWeight: "bold",
138
+ },
139
+ userEmail: {
140
+ fontSize: 14,
141
+ marginBottom: 12,
142
+ },
143
+ signOutButton: {
144
+ padding: 12,
145
+ },
146
+ signOutText: {
147
+ color: "#ffffff",
148
+ },
149
+ statusCard: {
150
+ marginBottom: 16,
151
+ padding: 16,
152
+ borderWidth: 1,
153
+ },
154
+ cardTitle: {
155
+ fontSize: 16,
156
+ fontWeight: "bold",
157
+ marginBottom: 12,
158
+ },
159
+ statusRow: {
160
+ flexDirection: "row",
161
+ alignItems: "center",
162
+ gap: 8,
163
+ },
164
+ statusIndicator: {
165
+ height: 8,
166
+ width: 8,
167
+ },
168
+ statusContent: {
169
+ flex: 1,
170
+ },
171
+ statusTitle: {
172
+ fontSize: 14,
173
+ fontWeight: "bold",
174
+ },
175
+ statusText: {
176
+ fontSize: 12,
177
+ },
178
+ privateDataCard: {
179
+ marginBottom: 16,
180
+ padding: 16,
181
+ borderWidth: 1,
182
+ },
183
+ privateDataText: {
184
+ fontSize: 14,
185
+ },
186
+ });
@@ -0,0 +1,131 @@
1
+ import { authClient } from "@/lib/auth-client";
2
+ {{#if (eq api "trpc")}}
3
+ import { queryClient } from "@/utils/trpc";
4
+ {{/if}}
5
+ {{#if (eq api "orpc")}}
6
+ import { queryClient } from "@/utils/orpc";
7
+ {{/if}}
8
+ import { useState } from "react";
9
+ import {
10
+ ActivityIndicator,
11
+ Text,
12
+ TextInput,
13
+ TouchableOpacity,
14
+ View,
15
+ StyleSheet,
16
+ } from "react-native";
17
+ import { useColorScheme } from "@/lib/use-color-scheme";
18
+ import { NAV_THEME } from "@/lib/constants";
19
+
20
+ function SignIn() {
21
+ const { colorScheme } = useColorScheme();
22
+ const theme = colorScheme === "dark" ? NAV_THEME.dark : NAV_THEME.light;
23
+ const [form, setForm] = useState({ email: "", password: "" });
24
+ const [isLoading, setIsLoading] = useState(false);
25
+ const [error, setError] = useState<string | null>(null);
26
+
27
+ function handleFormChange(field: "email" | "password", value: string) {
28
+ setForm(prev => ({ ...prev, [field]: value }));
29
+ }
30
+
31
+ async function handleLogin() {
32
+ setIsLoading(true);
33
+ setError(null);
34
+
35
+ await authClient.signIn.email(
36
+ {
37
+ email: form.email,
38
+ password: form.password,
39
+ },
40
+ {
41
+ onError(error) {
42
+ setError(error.error?.message || "Failed to sign in");
43
+ setIsLoading(false);
44
+ },
45
+ onSuccess() {
46
+ setForm({ email: "", password: "" });
47
+ {{#if (eq api "orpc")}}
48
+ queryClient.refetchQueries();
49
+ {{/if}}
50
+ {{#if (eq api "trpc")}}
51
+ queryClient.refetchQueries();
52
+ {{/if}}
53
+ },
54
+ onFinished() {
55
+ setIsLoading(false);
56
+ },
57
+ }
58
+ );
59
+ }
60
+
61
+ return (
62
+ <View style={[styles.card, { backgroundColor: theme.card, borderColor: theme.border }]}>
63
+ <Text style={[styles.title, { color: theme.text }]}>Sign In</Text>
64
+
65
+ {error ? (
66
+ <View style={[styles.errorContainer, { backgroundColor: theme.notification + "20" }]}>
67
+ <Text style={[styles.errorText, { color: theme.notification }]}>{error}</Text>
68
+ </View>
69
+ ) : null}
70
+
71
+ <TextInput style={[ styles.input, { color: theme.text, borderColor: theme.border, backgroundColor:
72
+ theme.background }, ]} placeholder="Email" placeholderTextColor={theme.text} value={form.email}
73
+ onChangeText={value=> handleFormChange("email", value)}
74
+ keyboardType="email-address"
75
+ autoCapitalize="none"
76
+ />
77
+
78
+ <TextInput style={[ styles.input, { color: theme.text, borderColor: theme.border, backgroundColor:
79
+ theme.background }, ]} placeholder="Password" placeholderTextColor={theme.text} value={form.password}
80
+ onChangeText={value=> handleFormChange("password", value)}
81
+ secureTextEntry
82
+ />
83
+
84
+ <TouchableOpacity onPress={handleLogin} disabled={isLoading} style={[ styles.button, { backgroundColor:
85
+ theme.primary, opacity: isLoading ? 0.5 : 1 }, ]}>
86
+ {isLoading ? (
87
+ <ActivityIndicator size="small" color="#ffffff" />
88
+ ) : (
89
+ <Text style={styles.buttonText}>Sign In</Text>
90
+ )}
91
+ </TouchableOpacity>
92
+ </View>
93
+ );
94
+ }
95
+
96
+ const styles = StyleSheet.create({
97
+ card: {
98
+ marginTop: 16,
99
+ padding: 16,
100
+ borderWidth: 1,
101
+ },
102
+ title: {
103
+ fontSize: 18,
104
+ fontWeight: "bold",
105
+ marginBottom: 12,
106
+ },
107
+ errorContainer: {
108
+ marginBottom: 12,
109
+ padding: 8,
110
+ },
111
+ errorText: {
112
+ fontSize: 14,
113
+ },
114
+ input: {
115
+ borderWidth: 1,
116
+ padding: 12,
117
+ fontSize: 16,
118
+ marginBottom: 12,
119
+ },
120
+ button: {
121
+ padding: 12,
122
+ alignItems: "center",
123
+ justifyContent: "center",
124
+ },
125
+ buttonText: {
126
+ color: "#ffffff",
127
+ fontSize: 16,
128
+ },
129
+ });
130
+
131
+ export { SignIn };
@@ -0,0 +1,150 @@
1
+ import { authClient } from "@/lib/auth-client";
2
+ {{#if (eq api "trpc")}}
3
+ import { queryClient } from "@/utils/trpc";
4
+ {{/if}}
5
+ {{#if (eq api "orpc")}}
6
+ import { queryClient } from "@/utils/orpc";
7
+ {{/if}}
8
+ import { useState } from "react";
9
+ import {
10
+ ActivityIndicator,
11
+ Text,
12
+ TextInput,
13
+ TouchableOpacity,
14
+ View,
15
+ StyleSheet,
16
+ } from "react-native";
17
+ import { useColorScheme } from "@/lib/use-color-scheme";
18
+ import { NAV_THEME } from "@/lib/constants";
19
+
20
+ function SignUp() {
21
+ const { colorScheme } = useColorScheme();
22
+ const theme = colorScheme === "dark" ? NAV_THEME.dark : NAV_THEME.light;
23
+ const [name, setName] = useState("");
24
+ const [email, setEmail] = useState("");
25
+ const [password, setPassword] = useState("");
26
+ const [isLoading, setIsLoading] = useState(false);
27
+ const [error, setError] = useState<string | null>(null);
28
+
29
+ async function handleSignUp() {
30
+ setIsLoading(true);
31
+ setError(null);
32
+
33
+ await authClient.signUp.email(
34
+ {
35
+ name,
36
+ email,
37
+ password,
38
+ },
39
+ {
40
+ onError(error) {
41
+ setError(error.error?.message || "Failed to sign up");
42
+ setIsLoading(false);
43
+ },
44
+ onSuccess() {
45
+ setName("");
46
+ setEmail("");
47
+ setPassword("");
48
+ {{#if (eq api "orpc")}}
49
+ queryClient.refetchQueries();
50
+ {{/if}}
51
+ {{#if (eq api "trpc")}}
52
+ queryClient.refetchQueries();
53
+ {{/if}}
54
+ },
55
+ onFinished() {
56
+ setIsLoading(false);
57
+ },
58
+ }
59
+ );
60
+ }
61
+
62
+ return (
63
+ <View style={[styles.card, { backgroundColor: theme.card, borderColor: theme.border }]}>
64
+ <Text style={[styles.title, { color: theme.text }]}>Create Account</Text>
65
+
66
+ {error ? (
67
+ <View style={[styles.errorContainer, { backgroundColor: theme.notification + "20" }]}>
68
+ <Text style={[styles.errorText, { color: theme.notification }]}>{error}</Text>
69
+ </View>
70
+ ) : null}
71
+
72
+ <TextInput
73
+ style={[styles.input, { color: theme.text, borderColor: theme.border, backgroundColor: theme.background }]}
74
+ placeholder="Name"
75
+ placeholderTextColor={theme.text}
76
+ value={name}
77
+ onChangeText={setName}
78
+ />
79
+
80
+ <TextInput
81
+ style={[styles.input, { color: theme.text, borderColor: theme.border, backgroundColor: theme.background }]}
82
+ placeholder="Email"
83
+ placeholderTextColor={theme.text}
84
+ value={email}
85
+ onChangeText={setEmail}
86
+ keyboardType="email-address"
87
+ autoCapitalize="none"
88
+ />
89
+
90
+ <TextInput
91
+ style={[styles.input, { color: theme.text, borderColor: theme.border, backgroundColor: theme.background }]}
92
+ placeholder="Password"
93
+ placeholderTextColor={theme.text}
94
+ value={password}
95
+ onChangeText={setPassword}
96
+ secureTextEntry
97
+ />
98
+
99
+ <TouchableOpacity
100
+ onPress={handleSignUp}
101
+ disabled={isLoading}
102
+ style={[styles.button, { backgroundColor: theme.primary, opacity: isLoading ? 0.5 : 1 }]}
103
+ >
104
+ {isLoading ? (
105
+ <ActivityIndicator size="small" color="#ffffff" />
106
+ ) : (
107
+ <Text style={styles.buttonText}>Sign Up</Text>
108
+ )}
109
+ </TouchableOpacity>
110
+ </View>
111
+ );
112
+ }
113
+
114
+ const styles = StyleSheet.create({
115
+ card: {
116
+ marginTop: 16,
117
+ padding: 16,
118
+ borderWidth: 1,
119
+ },
120
+ title: {
121
+ fontSize: 18,
122
+ fontWeight: "bold",
123
+ marginBottom: 12,
124
+ },
125
+ errorContainer: {
126
+ marginBottom: 12,
127
+ padding: 8,
128
+ },
129
+ errorText: {
130
+ fontSize: 14,
131
+ },
132
+ input: {
133
+ borderWidth: 1,
134
+ padding: 12,
135
+ fontSize: 16,
136
+ marginBottom: 12,
137
+ },
138
+ button: {
139
+ padding: 12,
140
+ alignItems: "center",
141
+ justifyContent: "center",
142
+ },
143
+ buttonText: {
144
+ color: "#ffffff",
145
+ fontSize: 16,
146
+ },
147
+ });
148
+
149
+ export { SignUp };
150
+
@@ -0,0 +1,16 @@
1
+ import { expoClient } from "@better-auth/expo/client";
2
+ import { createAuthClient } from "better-auth/react";
3
+ import * as SecureStore from "expo-secure-store";
4
+ import Constants from "expo-constants";
5
+ import { env } from "@{{projectName}}/env/native";
6
+
7
+ export const authClient = createAuthClient({
8
+ baseURL: env.EXPO_PUBLIC_SERVER_URL,
9
+ plugins: [
10
+ expoClient({
11
+ scheme: Constants.expoConfig?.scheme as string,
12
+ storagePrefix: Constants.expoConfig?.scheme as string,
13
+ storage: SecureStore,
14
+ }),
15
+ ],
16
+ });
@@ -0,0 +1,187 @@
1
+ import { authClient } from "@/lib/auth-client";
2
+ import { ScrollView, Text, TouchableOpacity, View } from "react-native";
3
+ import { StyleSheet } from "react-native-unistyles";
4
+
5
+ import { Container } from "@/components/container";
6
+ import { SignIn } from "@/components/sign-in";
7
+ import { SignUp } from "@/components/sign-up";
8
+ {{#if (eq api "orpc")}}
9
+ import { useQuery } from "@tanstack/react-query";
10
+ import { queryClient, orpc } from "@/utils/orpc";
11
+ {{/if}}
12
+ {{#if (eq api "trpc")}}
13
+ import { useQuery } from "@tanstack/react-query";
14
+ import { queryClient, trpc } from "@/utils/trpc";
15
+ {{/if}}
16
+
17
+ export default function Home() {
18
+ {{#if (eq api "orpc")}}
19
+ const healthCheck = useQuery(orpc.healthCheck.queryOptions());
20
+ const privateData = useQuery(orpc.privateData.queryOptions());
21
+ {{/if}}
22
+ {{#if (eq api "trpc")}}
23
+ const healthCheck = useQuery(trpc.healthCheck.queryOptions());
24
+ const privateData = useQuery(trpc.privateData.queryOptions());
25
+ {{/if}}
26
+ const { data: session } = authClient.useSession();
27
+
28
+ return (
29
+ <Container>
30
+ <ScrollView>
31
+ <View style={styles.pageContainer}>
32
+ <Text style={styles.headerTitle}>BETTER T STACK</Text>
33
+ {session?.user ? (
34
+ <View style={styles.sessionInfoCard}>
35
+ <View style={styles.sessionUserRow}>
36
+ <Text style={styles.welcomeText}>
37
+ Welcome,{" "}
38
+ <Text style={styles.userNameText}>{session.user.name}</Text>
39
+ </Text>
40
+ </View>
41
+ <Text style={styles.emailText}>{session.user.email}</Text>
42
+
43
+ <TouchableOpacity
44
+ style={styles.signOutButton}
45
+ onPress={() => {
46
+ authClient.signOut();
47
+ {{#if (eq api "orpc")}}
48
+ queryClient.invalidateQueries();
49
+ {{/if}}
50
+ {{#if (eq api "trpc")}}
51
+ queryClient.invalidateQueries();
52
+ {{/if}}
53
+ }}
54
+ >
55
+ <Text style={styles.signOutButtonText}>Sign Out</Text>
56
+ </TouchableOpacity>
57
+ </View>
58
+ ) : null}
59
+ {{#unless (eq api "none")}}
60
+ <View style={styles.apiStatusCard}>
61
+ <Text style={styles.cardTitle}>API Status</Text>
62
+ <View style={styles.apiStatusRow}>
63
+ <View
64
+ style={[
65
+ styles.statusIndicatorDot,
66
+ healthCheck.data
67
+ ? styles.statusIndicatorGreen
68
+ : styles.statusIndicatorRed,
69
+ ]}
70
+ />
71
+ <Text style={styles.mutedText}>
72
+ {healthCheck.isLoading
73
+ ? "Checking..."
74
+ : healthCheck.data
75
+ ? "Connected to API"
76
+ : "API Disconnected"}
77
+ </Text>
78
+ </View>
79
+ </View>
80
+ <View style={styles.privateDataCard}>
81
+ <Text style={styles.cardTitle}>Private Data</Text>
82
+ {privateData && (
83
+ <View>
84
+ <Text style={styles.mutedText}>
85
+ {privateData.data?.message}
86
+ </Text>
87
+ </View>
88
+ )}
89
+ </View>
90
+ {{/unless}}
91
+ {!session?.user && (
92
+ <>
93
+ <SignIn />
94
+ <SignUp />
95
+ </>
96
+ )}
97
+ </View>
98
+ </ScrollView>
99
+ </Container>
100
+ );
101
+ }
102
+
103
+ const styles = StyleSheet.create((theme) => ({
104
+ pageContainer: {
105
+ paddingHorizontal: 8,
106
+ },
107
+ headerTitle: {
108
+ color: theme?.colors?.typography,
109
+ fontSize: 30,
110
+ fontWeight: "bold",
111
+ marginBottom: 16,
112
+ },
113
+ sessionInfoCard: {
114
+ marginBottom: 24,
115
+ padding: 16,
116
+ borderRadius: 8,
117
+ borderWidth: 1,
118
+ borderColor: theme?.colors?.border,
119
+ },
120
+ sessionUserRow: {
121
+ flexDirection: "row",
122
+ justifyContent: "space-between",
123
+ alignItems: "center",
124
+ marginBottom: 8,
125
+ },
126
+ welcomeText: {
127
+ color: theme?.colors?.typography,
128
+ fontSize: 16,
129
+ },
130
+ userNameText: {
131
+ fontWeight: "500",
132
+ color: theme?.colors?.typography,
133
+ },
134
+ emailText: {
135
+ color: theme?.colors?.typography,
136
+ fontSize: 14,
137
+ marginBottom: 16,
138
+ },
139
+ signOutButton: {
140
+ backgroundColor: theme?.colors?.destructive,
141
+ paddingVertical: 8,
142
+ paddingHorizontal: 16,
143
+ borderRadius: 6,
144
+ alignSelf: "flex-start",
145
+ },
146
+ signOutButtonText: {
147
+ fontWeight: "500",
148
+ },
149
+ apiStatusCard: {
150
+ marginBottom: 24,
151
+ borderRadius: 8,
152
+ borderWidth: 1,
153
+ borderColor: theme?.colors?.border,
154
+ padding: 16,
155
+ },
156
+ cardTitle: {
157
+ marginBottom: 12,
158
+ fontWeight: "500",
159
+ color: theme?.colors?.typography,
160
+ },
161
+ apiStatusRow: {
162
+ flexDirection: "row",
163
+ alignItems: "center",
164
+ gap: 8,
165
+ },
166
+ statusIndicatorDot: {
167
+ height: 12,
168
+ width: 12,
169
+ borderRadius: 9999,
170
+ },
171
+ statusIndicatorGreen: {
172
+ backgroundColor: theme.colors.success,
173
+ },
174
+ statusIndicatorRed: {
175
+ backgroundColor: theme.colors.destructive,
176
+ },
177
+ mutedText: {
178
+ color: theme?.colors?.typography,
179
+ },
180
+ privateDataCard: {
181
+ marginBottom: 24,
182
+ borderRadius: 8,
183
+ borderWidth: 1,
184
+ borderColor: theme?.colors?.border,
185
+ padding: 16,
186
+ },
187
+ }));