@digilogiclabs/create-saas-app 1.5.2 → 1.5.4

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 (223) hide show
  1. package/dist/.tsbuildinfo +1 -1
  2. package/dist/templates/mobile/base/template/.env.example +15 -0
  3. package/dist/templates/mobile/base/template/App.tsx +88 -0
  4. package/dist/templates/mobile/base/template/app/(auth)/login.tsx +44 -0
  5. package/dist/templates/mobile/base/template/app/(auth)/signup.tsx +43 -0
  6. package/dist/templates/mobile/base/template/app/checkout.tsx +20 -0
  7. package/dist/templates/mobile/base/template/package.json +38 -0
  8. package/dist/templates/shared/auth/firebase/web/config.ts +23 -0
  9. package/dist/templates/shared/auth/supabase/web/config.ts +8 -0
  10. package/dist/templates/web/base/template/.env.example +15 -0
  11. package/dist/templates/web/base/template/.eslintrc.js +8 -0
  12. package/dist/templates/web/base/template/README.md +68 -0
  13. package/dist/templates/web/base/template/next.config.js +15 -0
  14. package/dist/templates/web/base/template/package.json +58 -0
  15. package/dist/templates/web/base/template/postcss.config.js +7 -0
  16. package/dist/templates/web/base/template/src/app/auth/callback/route.ts +18 -0
  17. package/dist/templates/web/base/template/src/app/checkout/page.tsx +28 -0
  18. package/dist/templates/web/base/template/src/app/error.tsx +97 -0
  19. package/dist/templates/web/base/template/src/app/globals.css +60 -0
  20. package/dist/templates/web/base/template/src/app/layout.tsx +35 -0
  21. package/dist/templates/web/base/template/src/app/loading.tsx +34 -0
  22. package/dist/templates/web/base/template/src/app/login/page.tsx +39 -0
  23. package/dist/templates/web/base/template/src/app/page.tsx +132 -0
  24. package/dist/templates/web/base/template/src/app/signup/page.tsx +39 -0
  25. package/dist/templates/web/base/template/src/components/__tests__/example.test.tsx +49 -0
  26. package/dist/templates/web/base/template/src/components/providers/app-providers.tsx +33 -0
  27. package/dist/templates/web/base/template/src/components/providers/theme-provider.tsx +94 -0
  28. package/dist/templates/web/base/template/src/components/shared/footer.tsx +36 -0
  29. package/dist/templates/web/base/template/src/components/shared/header.tsx +44 -0
  30. package/dist/templates/web/base/template/src/components/ui/badge.tsx +36 -0
  31. package/dist/templates/web/base/template/src/components/ui/button.tsx +56 -0
  32. package/dist/templates/web/base/template/src/components/ui/card.tsx +71 -0
  33. package/dist/templates/web/base/template/src/components/ui/theme-toggle.tsx +34 -0
  34. package/dist/templates/web/base/template/src/lib/auth-server.ts +177 -0
  35. package/dist/templates/web/base/template/src/lib/env.ts +46 -0
  36. package/dist/templates/web/base/template/src/lib/utils.ts +140 -0
  37. package/dist/templates/web/base/template/src/test/setup.ts +79 -0
  38. package/dist/templates/web/base/template/tailwind.config.js +77 -0
  39. package/dist/templates/web/base/template/tsconfig.json +33 -0
  40. package/dist/templates/web/base/template/vitest.config.ts +17 -0
  41. package/dist/templates/web/base/template.backup/.env.example +15 -0
  42. package/dist/templates/web/base/template.backup.20250817/.env.example +15 -0
  43. package/dist/templates/web/ui-auth/template/.env.example +15 -0
  44. package/dist/templates/web/ui-auth/template/.eslintrc.js +8 -0
  45. package/dist/templates/web/ui-auth/template/README.md +68 -0
  46. package/dist/templates/web/ui-auth/template/next.config.js +12 -0
  47. package/dist/templates/web/ui-auth/template/package.json +50 -0
  48. package/dist/templates/web/ui-auth/template/postcss.config.js +7 -0
  49. package/dist/templates/web/ui-auth/template/src/app/auth/callback/route.ts +12 -0
  50. package/dist/templates/web/ui-auth/template/src/app/checkout/page.tsx +25 -0
  51. package/dist/templates/web/ui-auth/template/src/app/error.tsx +67 -0
  52. package/dist/templates/web/ui-auth/template/src/app/globals.css +42 -0
  53. package/dist/templates/web/ui-auth/template/src/app/layout.tsx +33 -0
  54. package/dist/templates/web/ui-auth/template/src/app/loading.tsx +20 -0
  55. package/dist/templates/web/ui-auth/template/src/app/login/page.tsx +109 -0
  56. package/dist/templates/web/ui-auth/template/src/app/page.tsx +129 -0
  57. package/dist/templates/web/ui-auth/template/src/app/signup/page.tsx +128 -0
  58. package/dist/templates/web/ui-auth/template/src/components/__tests__/example.test.tsx +49 -0
  59. package/dist/templates/web/ui-auth/template/src/components/providers/app-providers.tsx +29 -0
  60. package/dist/templates/web/ui-auth/template/src/components/providers/theme-provider.tsx +94 -0
  61. package/dist/templates/web/ui-auth/template/src/components/shared/footer.tsx +36 -0
  62. package/dist/templates/web/ui-auth/template/src/components/shared/header.tsx +53 -0
  63. package/dist/templates/web/ui-auth/template/src/components/ui/badge.tsx +36 -0
  64. package/dist/templates/web/ui-auth/template/src/components/ui/theme-toggle.tsx +34 -0
  65. package/dist/templates/web/ui-auth/template/src/lib/env.ts +49 -0
  66. package/dist/templates/web/ui-auth/template/src/lib/utils.ts +140 -0
  67. package/dist/templates/web/ui-auth/template/src/test/setup.ts +79 -0
  68. package/dist/templates/web/ui-auth/template/tailwind.config.js +77 -0
  69. package/dist/templates/web/ui-auth/template/tsconfig.json +33 -0
  70. package/dist/templates/web/ui-auth/template/vitest.config.ts +17 -0
  71. package/dist/templates/web/ui-auth/template.backup/.env.example +15 -0
  72. package/dist/templates/web/ui-auth/template.backup.20250817/.env.example +15 -0
  73. package/dist/templates/web/ui-auth-payments/template/.env.example +15 -0
  74. package/dist/templates/web/ui-auth-payments/template/README.md +165 -0
  75. package/dist/templates/web/ui-auth-payments/template/middleware.ts +68 -0
  76. package/dist/templates/web/ui-auth-payments/template/next.config.js +12 -0
  77. package/dist/templates/web/ui-auth-payments/template/package-lock.json +12240 -0
  78. package/dist/templates/web/ui-auth-payments/template/package.json +52 -0
  79. package/dist/templates/web/ui-auth-payments/template/postcss.config.js +7 -0
  80. package/dist/templates/web/ui-auth-payments/template/src/app/auth/callback/route.ts +12 -0
  81. package/dist/templates/web/ui-auth-payments/template/src/app/billing/page.tsx +211 -0
  82. package/dist/templates/web/ui-auth-payments/template/src/app/checkout/page.tsx +142 -0
  83. package/dist/templates/web/ui-auth-payments/template/src/app/dashboard/layout.tsx +22 -0
  84. package/dist/templates/web/ui-auth-payments/template/src/app/dashboard/page.tsx +183 -0
  85. package/dist/templates/web/ui-auth-payments/template/src/app/error.tsx +67 -0
  86. package/dist/templates/web/ui-auth-payments/template/src/app/globals.css +42 -0
  87. package/dist/templates/web/ui-auth-payments/template/src/app/layout.tsx +33 -0
  88. package/dist/templates/web/ui-auth-payments/template/src/app/loading.tsx +20 -0
  89. package/dist/templates/web/ui-auth-payments/template/src/app/login/loading.tsx +38 -0
  90. package/dist/templates/web/ui-auth-payments/template/src/app/login/page.tsx +109 -0
  91. package/dist/templates/web/ui-auth-payments/template/src/app/page.tsx +143 -0
  92. package/dist/templates/web/ui-auth-payments/template/src/app/signup/loading.tsx +50 -0
  93. package/dist/templates/web/ui-auth-payments/template/src/app/signup/page.tsx +128 -0
  94. package/dist/templates/web/ui-auth-payments/template/src/components/__tests__/example.test.tsx +49 -0
  95. package/dist/templates/web/ui-auth-payments/template/src/components/client/auth-status.tsx +52 -0
  96. package/dist/templates/web/ui-auth-payments/template/src/components/client/login-form.tsx +144 -0
  97. package/dist/templates/web/ui-auth-payments/template/src/components/client/newsletter-signup.tsx +68 -0
  98. package/dist/templates/web/ui-auth-payments/template/src/components/client/signup-form.tsx +185 -0
  99. package/dist/templates/web/ui-auth-payments/template/src/components/providers/app-providers.tsx +32 -0
  100. package/dist/templates/web/ui-auth-payments/template/src/components/providers/theme-provider.tsx +94 -0
  101. package/dist/templates/web/ui-auth-payments/template/src/components/shared/footer.tsx +36 -0
  102. package/dist/templates/web/ui-auth-payments/template/src/components/shared/header.tsx +62 -0
  103. package/dist/templates/web/ui-auth-payments/template/src/components/ui/badge.tsx +36 -0
  104. package/dist/templates/web/ui-auth-payments/template/src/components/ui/theme-toggle.tsx +34 -0
  105. package/dist/templates/web/ui-auth-payments/template/src/lib/actions/auth.ts +246 -0
  106. package/dist/templates/web/ui-auth-payments/template/src/lib/actions/index.ts +340 -0
  107. package/dist/templates/web/ui-auth-payments/template/src/lib/auth-server.ts +177 -0
  108. package/dist/templates/web/ui-auth-payments/template/src/lib/env.ts +49 -0
  109. package/dist/templates/web/ui-auth-payments/template/src/lib/utils.ts +140 -0
  110. package/dist/templates/web/ui-auth-payments/template/src/test/setup.ts +79 -0
  111. package/dist/templates/web/ui-auth-payments/template/tailwind.config.js +77 -0
  112. package/dist/templates/web/ui-auth-payments/template/tsconfig.json +33 -0
  113. package/dist/templates/web/ui-auth-payments/template/tsconfig.tsbuildinfo +1 -0
  114. package/dist/templates/web/ui-auth-payments/template/vitest.config.ts +17 -0
  115. package/dist/templates/web/ui-auth-payments-audio/template/.env.example +15 -0
  116. package/dist/templates/web/ui-auth-payments-audio/template/README.md +187 -0
  117. package/dist/templates/web/ui-auth-payments-audio/template/middleware.ts +68 -0
  118. package/dist/templates/web/ui-auth-payments-audio/template/next.config.js +12 -0
  119. package/dist/templates/web/ui-auth-payments-audio/template/package-lock.json +12241 -0
  120. package/dist/templates/web/ui-auth-payments-audio/template/package.json +53 -0
  121. package/dist/templates/web/ui-auth-payments-audio/template/postcss.config.js +7 -0
  122. package/dist/templates/web/ui-auth-payments-audio/template/src/app/auth/callback/route.ts +12 -0
  123. package/dist/templates/web/ui-auth-payments-audio/template/src/app/billing/page.tsx +211 -0
  124. package/dist/templates/web/ui-auth-payments-audio/template/src/app/checkout/page.tsx +142 -0
  125. package/dist/templates/web/ui-auth-payments-audio/template/src/app/dashboard/layout.tsx +22 -0
  126. package/dist/templates/web/ui-auth-payments-audio/template/src/app/dashboard/page.tsx +183 -0
  127. package/dist/templates/web/ui-auth-payments-audio/template/src/app/error.tsx +67 -0
  128. package/dist/templates/web/ui-auth-payments-audio/template/src/app/globals.css +42 -0
  129. package/dist/templates/web/ui-auth-payments-audio/template/src/app/layout.tsx +35 -0
  130. package/dist/templates/web/ui-auth-payments-audio/template/src/app/loading.tsx +20 -0
  131. package/dist/templates/web/ui-auth-payments-audio/template/src/app/login/page.tsx +6 -0
  132. package/dist/templates/web/ui-auth-payments-audio/template/src/app/page.tsx +181 -0
  133. package/dist/templates/web/ui-auth-payments-audio/template/src/app/signup/page.tsx +6 -0
  134. package/dist/templates/web/ui-auth-payments-audio/template/src/components/__tests__/example.test.tsx +49 -0
  135. package/dist/templates/web/ui-auth-payments-audio/template/src/components/client/auth-status.tsx +52 -0
  136. package/dist/templates/web/ui-auth-payments-audio/template/src/components/client/login-form.tsx +144 -0
  137. package/dist/templates/web/ui-auth-payments-audio/template/src/components/client/signup-form.tsx +185 -0
  138. package/dist/templates/web/ui-auth-payments-audio/template/src/components/providers/app-providers.tsx +32 -0
  139. package/dist/templates/web/ui-auth-payments-audio/template/src/components/providers/theme-provider.tsx +94 -0
  140. package/dist/templates/web/ui-auth-payments-audio/template/src/components/shared/footer.tsx +36 -0
  141. package/dist/templates/web/ui-auth-payments-audio/template/src/components/shared/header.tsx +62 -0
  142. package/dist/templates/web/ui-auth-payments-audio/template/src/components/ui/badge.tsx +36 -0
  143. package/dist/templates/web/ui-auth-payments-audio/template/src/components/ui/theme-toggle.tsx +34 -0
  144. package/dist/templates/web/ui-auth-payments-audio/template/src/lib/actions/auth.ts +246 -0
  145. package/dist/templates/web/ui-auth-payments-audio/template/src/lib/actions/index.ts +14 -0
  146. package/dist/templates/web/ui-auth-payments-audio/template/src/lib/auth-server.ts +177 -0
  147. package/dist/templates/web/ui-auth-payments-audio/template/src/lib/env.ts +49 -0
  148. package/dist/templates/web/ui-auth-payments-audio/template/src/lib/utils.ts +140 -0
  149. package/dist/templates/web/ui-auth-payments-audio/template/src/test/setup.ts +79 -0
  150. package/dist/templates/web/ui-auth-payments-audio/template/tailwind.config.js +77 -0
  151. package/dist/templates/web/ui-auth-payments-audio/template/tsconfig.json +33 -0
  152. package/dist/templates/web/ui-auth-payments-audio/template/tsconfig.tsbuildinfo +1 -0
  153. package/dist/templates/web/ui-auth-payments-audio/template/vitest.config.ts +17 -0
  154. package/dist/templates/web/ui-auth-payments-video/template/.env.example +15 -0
  155. package/dist/templates/web/ui-auth-payments-video/template/README.md +190 -0
  156. package/dist/templates/web/ui-auth-payments-video/template/next.config.js +12 -0
  157. package/dist/templates/web/ui-auth-payments-video/template/package.json +53 -0
  158. package/dist/templates/web/ui-auth-payments-video/template/postcss.config.js +7 -0
  159. package/dist/templates/web/ui-auth-payments-video/template/src/app/auth/callback/route.ts +12 -0
  160. package/dist/templates/web/ui-auth-payments-video/template/src/app/billing/page.tsx +211 -0
  161. package/dist/templates/web/ui-auth-payments-video/template/src/app/checkout/page.tsx +142 -0
  162. package/dist/templates/web/ui-auth-payments-video/template/src/app/error.tsx +67 -0
  163. package/dist/templates/web/ui-auth-payments-video/template/src/app/globals.css +42 -0
  164. package/dist/templates/web/ui-auth-payments-video/template/src/app/layout.tsx +33 -0
  165. package/dist/templates/web/ui-auth-payments-video/template/src/app/loading.tsx +20 -0
  166. package/dist/templates/web/ui-auth-payments-video/template/src/app/login/page.tsx +109 -0
  167. package/dist/templates/web/ui-auth-payments-video/template/src/app/page.tsx +187 -0
  168. package/dist/templates/web/ui-auth-payments-video/template/src/app/signup/page.tsx +128 -0
  169. package/dist/templates/web/ui-auth-payments-video/template/src/components/__tests__/example.test.tsx +49 -0
  170. package/dist/templates/web/ui-auth-payments-video/template/src/components/providers/app-providers.tsx +32 -0
  171. package/dist/templates/web/ui-auth-payments-video/template/src/components/providers/theme-provider.tsx +94 -0
  172. package/dist/templates/web/ui-auth-payments-video/template/src/components/shared/footer.tsx +36 -0
  173. package/dist/templates/web/ui-auth-payments-video/template/src/components/shared/header.tsx +62 -0
  174. package/dist/templates/web/ui-auth-payments-video/template/src/components/ui/badge.tsx +36 -0
  175. package/dist/templates/web/ui-auth-payments-video/template/src/components/ui/theme-toggle.tsx +34 -0
  176. package/dist/templates/web/ui-auth-payments-video/template/src/lib/env.ts +49 -0
  177. package/dist/templates/web/ui-auth-payments-video/template/src/lib/utils.ts +140 -0
  178. package/dist/templates/web/ui-auth-payments-video/template/src/test/setup.ts +79 -0
  179. package/dist/templates/web/ui-auth-payments-video/template/tailwind.config.js +77 -0
  180. package/dist/templates/web/ui-auth-payments-video/template/tsconfig.json +33 -0
  181. package/dist/templates/web/ui-auth-payments-video/template/vitest.config.ts +17 -0
  182. package/dist/templates/web/ui-only/template/.env.example +15 -0
  183. package/dist/templates/web/ui-only/template/.eslintrc.js +8 -0
  184. package/dist/templates/web/ui-only/template/README.md +68 -0
  185. package/dist/templates/web/ui-only/template/next.config.js +12 -0
  186. package/dist/templates/web/ui-only/template/package.json +49 -0
  187. package/dist/templates/web/ui-only/template/postcss.config.js +7 -0
  188. package/dist/templates/web/ui-only/template/src/app/auth/callback/route.ts +12 -0
  189. package/dist/templates/web/ui-only/template/src/app/checkout/page.tsx +25 -0
  190. package/dist/templates/web/ui-only/template/src/app/error.tsx +67 -0
  191. package/dist/templates/web/ui-only/template/src/app/globals.css +42 -0
  192. package/dist/templates/web/ui-only/template/src/app/layout.tsx +33 -0
  193. package/dist/templates/web/ui-only/template/src/app/loading.tsx +20 -0
  194. package/dist/templates/web/ui-only/template/src/app/login/page.tsx +63 -0
  195. package/dist/templates/web/ui-only/template/src/app/page.tsx +91 -0
  196. package/dist/templates/web/ui-only/template/src/app/signup/page.tsx +79 -0
  197. package/dist/templates/web/ui-only/template/src/components/__tests__/example.test.tsx +49 -0
  198. package/dist/templates/web/ui-only/template/src/components/providers/app-providers.tsx +26 -0
  199. package/dist/templates/web/ui-only/template/src/components/providers/theme-provider.tsx +94 -0
  200. package/dist/templates/web/ui-only/template/src/components/shared/footer.tsx +36 -0
  201. package/dist/templates/web/ui-only/template/src/components/shared/header.tsx +53 -0
  202. package/dist/templates/web/ui-only/template/src/components/ui/badge.tsx +36 -0
  203. package/dist/templates/web/ui-only/template/src/components/ui/theme-toggle.tsx +34 -0
  204. package/dist/templates/web/ui-only/template/src/lib/env.ts +49 -0
  205. package/dist/templates/web/ui-only/template/src/lib/utils.ts +140 -0
  206. package/dist/templates/web/ui-only/template/src/test/setup.ts +79 -0
  207. package/dist/templates/web/ui-only/template/tailwind.config.js +77 -0
  208. package/dist/templates/web/ui-only/template/tsconfig.json +33 -0
  209. package/dist/templates/web/ui-only/template/vitest.config.ts +17 -0
  210. package/dist/templates/web/ui-only/template.backup/.env.example +15 -0
  211. package/dist/templates/web/ui-only/template.backup.20250817/.env.example +15 -0
  212. package/dist/templates/web/ui-package-test/template/next-env.d.ts +5 -0
  213. package/dist/templates/web/ui-package-test/template/package.json +42 -0
  214. package/dist/templates/web/ui-package-test/template/src/app/page.tsx +106 -0
  215. package/dist/templates/web/ui-package-test/template/tsconfig.json +41 -0
  216. package/package.json +3 -2
  217. package/src/templates/mobile/base/template/package.json +38 -38
  218. package/src/templates/web/base/template/package.json +1 -1
  219. package/src/templates/web/ui-auth/template/package.json +1 -1
  220. package/src/templates/web/ui-auth-payments/template/package.json +1 -1
  221. package/src/templates/web/ui-auth-payments-audio/template/package.json +1 -1
  222. package/src/templates/web/ui-auth-payments-video/template/package.json +1 -1
  223. package/src/templates/web/ui-only/template/package.json +1 -1
@@ -0,0 +1,71 @@
1
+ import * as React from "react"
2
+ import { cn } from "@/lib/utils"
3
+
4
+ const Card = React.forwardRef<
5
+ HTMLDivElement,
6
+ React.HTMLAttributes<HTMLDivElement>
7
+ >(({ className, ...props }, ref) => (
8
+ <div
9
+ ref={ref}
10
+ className={cn(
11
+ "rounded-lg border bg-card text-card-foreground shadow-sm",
12
+ className
13
+ )}
14
+ {...props}
15
+ />
16
+ ))
17
+ Card.displayName = "Card"
18
+
19
+ const CardHeader = React.forwardRef<
20
+ HTMLDivElement,
21
+ React.HTMLAttributes<HTMLDivElement>
22
+ >(({ className, ...props }, ref) => (
23
+ <div ref={ref} className={cn("flex flex-col space-y-1.5 p-6", className)} {...props} />
24
+ ))
25
+ CardHeader.displayName = "CardHeader"
26
+
27
+ const CardTitle = React.forwardRef<
28
+ HTMLParagraphElement,
29
+ React.HTMLAttributes<HTMLHeadingElement>
30
+ >(({ className, ...props }, ref) => (
31
+ <h3
32
+ ref={ref}
33
+ className={cn(
34
+ "text-2xl font-semibold leading-none tracking-tight",
35
+ className
36
+ )}
37
+ {...props}
38
+ />
39
+ ))
40
+ CardTitle.displayName = "CardTitle"
41
+
42
+ const CardDescription = React.forwardRef<
43
+ HTMLParagraphElement,
44
+ React.HTMLAttributes<HTMLParagraphElement>
45
+ >(({ className, ...props }, ref) => (
46
+ <p
47
+ ref={ref}
48
+ className={cn("text-sm text-muted-foreground", className)}
49
+ {...props}
50
+ />
51
+ ))
52
+ CardDescription.displayName = "CardDescription"
53
+
54
+ const CardContent = React.forwardRef<
55
+ HTMLDivElement,
56
+ React.HTMLAttributes<HTMLDivElement>
57
+ >(({ className, ...props }, ref) => (
58
+ <div ref={ref} className={cn("p-6 pt-0", className)} {...props} />
59
+ ))
60
+ CardContent.displayName = "CardContent"
61
+
62
+ const CardFooter = React.forwardRef<
63
+ HTMLDivElement,
64
+ React.HTMLAttributes<HTMLDivElement>
65
+ >(({ className, ...props }, ref) => (
66
+ <div ref={ref} className={cn("flex items-center p-6 pt-0", className)} {...props} />
67
+ ))
68
+ CardFooter.displayName = "CardFooter"
69
+
70
+ export { Card, CardHeader, CardFooter, CardTitle, CardDescription, CardContent }
71
+
@@ -0,0 +1,34 @@
1
+ 'use client'
2
+
3
+ import * as React from "react"
4
+ import { Moon, Sun } from "lucide-react"
5
+ import { useTheme } from "next-themes"
6
+
7
+ import { Button } from "@/components/ui/button"
8
+
9
+ export function ThemeToggle() {
10
+ const { theme, setTheme } = useTheme()
11
+
12
+ const toggleTheme = () => {
13
+ if (theme === 'light') {
14
+ setTheme('dark')
15
+ } else if (theme === 'dark') {
16
+ setTheme('system')
17
+ } else {
18
+ setTheme('light')
19
+ }
20
+ }
21
+
22
+ return (
23
+ <Button
24
+ variant="outline"
25
+ size="icon"
26
+ onClick={toggleTheme}
27
+ className="h-9 w-9"
28
+ >
29
+ <Sun className="h-4 w-4 rotate-0 scale-100 transition-all dark:-rotate-90 dark:scale-0" />
30
+ <Moon className="absolute h-4 w-4 rotate-90 scale-0 transition-all dark:rotate-0 dark:scale-100" />
31
+ <span className="sr-only">Toggle theme</span>
32
+ </Button>
33
+ )
34
+ }
@@ -0,0 +1,177 @@
1
+ import { cookies } from 'next/headers'
2
+ import { redirect } from 'next/navigation'
3
+
4
+ // Server-side auth utilities that work with @digilogiclabs/saas-factory-auth
5
+ // These functions provide server-side access to auth state
6
+
7
+ export type User = {
8
+ id: string
9
+ email: string
10
+ name?: string
11
+ avatar?: string
12
+ [key: string]: any
13
+ }
14
+
15
+ /**
16
+ * Get the current user from server-side cookies
17
+ * This function should be called in Server Components, Server Actions, or Route Handlers
18
+ */
19
+ export async function getCurrentUser(): Promise<User | null> {
20
+ try {
21
+ const cookieStore = await cookies()
22
+
23
+ // Check for auth token in cookies set by @digilogiclabs/saas-factory-auth
24
+ const authToken = cookieStore.get('saas-factory-auth-token')?.value
25
+ const userCookie = cookieStore.get('saas-factory-auth-user')?.value
26
+
27
+ if (!authToken || !userCookie) {
28
+ return null
29
+ }
30
+
31
+ // Parse user data from cookie
32
+ try {
33
+ const userData = JSON.parse(userCookie)
34
+ return userData
35
+ } catch {
36
+ return null
37
+ }
38
+ } catch (error) {
39
+ console.error('Error getting current user:', error)
40
+ return null
41
+ }
42
+ }
43
+
44
+ /**
45
+ * Require authentication - redirects to login if not authenticated
46
+ * Use this in Server Components that require authentication
47
+ */
48
+ export async function requireAuth(): Promise<User> {
49
+ const user = await getCurrentUser()
50
+
51
+ if (!user) {
52
+ redirect('/login')
53
+ }
54
+
55
+ return user
56
+ }
57
+
58
+ /**
59
+ * Check if user is authenticated (without redirecting)
60
+ * Returns boolean for conditional rendering
61
+ */
62
+ export async function isAuthenticated(): Promise<boolean> {
63
+ const user = await getCurrentUser()
64
+ return !!user
65
+ }
66
+
67
+ /**
68
+ * Redirect if already authenticated
69
+ * Use this on login/signup pages
70
+ */
71
+ export async function redirectIfAuthenticated(redirectTo: string = '/') {
72
+ const user = await getCurrentUser()
73
+
74
+ if (user) {
75
+ redirect(redirectTo)
76
+ }
77
+ }
78
+
79
+ /**
80
+ * Get user session data for server components
81
+ * Returns both user and loading state info
82
+ */
83
+ export async function getServerSession() {
84
+ const user = await getCurrentUser()
85
+
86
+ return {
87
+ user,
88
+ isAuthenticated: !!user,
89
+ isLoading: false, // Server-side is never loading
90
+ }
91
+ }
92
+
93
+ /**
94
+ * Server-side auth check for API routes
95
+ */
96
+ export async function getAuthFromRequest(request?: Request): Promise<User | null> {
97
+ try {
98
+ // If request is provided, extract cookies from it
99
+ if (request) {
100
+ const cookie = request.headers.get('cookie')
101
+ if (!cookie) return null
102
+
103
+ // Parse cookies manually from request
104
+ const cookies = cookie.split(';').reduce((acc, cookie) => {
105
+ const [key, value] = cookie.trim().split('=')
106
+ acc[key] = value
107
+ return acc
108
+ }, {} as Record<string, string>)
109
+
110
+ const authToken = cookies['saas-factory-auth-token']
111
+ const userCookie = cookies['saas-factory-auth-user']
112
+
113
+ if (!authToken || !userCookie) {
114
+ return null
115
+ }
116
+
117
+ try {
118
+ const userData = JSON.parse(decodeURIComponent(userCookie))
119
+ return userData
120
+ } catch {
121
+ return null
122
+ }
123
+ }
124
+
125
+ // Fallback to cookies() API
126
+ return await getCurrentUser()
127
+ } catch (error) {
128
+ console.error('Error getting auth from request:', error)
129
+ return null
130
+ }
131
+ }
132
+
133
+ /**
134
+ * Create server action wrapper that requires authentication
135
+ */
136
+ export function withAuth<T extends any[], R>(
137
+ action: (user: User, ...args: T) => Promise<R>
138
+ ) {
139
+ return async (...args: T): Promise<R> => {
140
+ const user = await requireAuth()
141
+ return action(user, ...args)
142
+ }
143
+ }
144
+
145
+ /**
146
+ * Server-side utility to check specific roles/permissions
147
+ * Extend this based on your user data structure
148
+ */
149
+ export async function hasRole(role: string): Promise<boolean> {
150
+ const user = await getCurrentUser()
151
+
152
+ if (!user) return false
153
+
154
+ // Assuming role is stored in user.role or user.roles
155
+ if (Array.isArray(user.roles)) {
156
+ return user.roles.includes(role)
157
+ }
158
+
159
+ return user.role === role
160
+ }
161
+
162
+ /**
163
+ * Server-side utility to require specific roles
164
+ */
165
+ export async function requireRole(role: string): Promise<User> {
166
+ const user = await requireAuth()
167
+
168
+ const hasRequiredRole = Array.isArray(user.roles)
169
+ ? user.roles.includes(role)
170
+ : user.role === role
171
+
172
+ if (!hasRequiredRole) {
173
+ redirect('/unauthorized')
174
+ }
175
+
176
+ return user
177
+ }
@@ -0,0 +1,46 @@
1
+ import { z } from 'zod'
2
+
3
+ const envSchema = z.object({
4
+ // Next.js environment
5
+ NODE_ENV: z.enum(['development', 'production', 'test']).default('development'),
6
+
7
+ // Supabase Configuration (if needed for custom auth)
8
+ NEXT_PUBLIC_SUPABASE_URL: z.string().url().optional(),
9
+ NEXT_PUBLIC_SUPABASE_ANON_KEY: z.string().optional(),
10
+ SUPABASE_SERVICE_ROLE_KEY: z.string().optional(),
11
+
12
+ // Stripe Configuration (from @digilogiclabs/saas-factory-payments)
13
+ NEXT_PUBLIC_STRIPE_PUBLISHABLE_KEY: z.string().optional(),
14
+ STRIPE_SECRET_KEY: z.string().optional(),
15
+ STRIPE_WEBHOOK_SECRET: z.string().optional(),
16
+
17
+ // App Configuration
18
+ NEXT_PUBLIC_APP_URL: z.string().url().default('http://localhost:3000'),
19
+
20
+ // Database (if using custom database)
21
+ DATABASE_URL: z.string().optional(),
22
+ })
23
+
24
+ export type Env = z.infer<typeof envSchema>
25
+
26
+ const parsedEnv = envSchema.safeParse(process.env)
27
+
28
+ if (!parsedEnv.success) {
29
+ console.error('❌ Invalid environment variables:', parsedEnv.error.format())
30
+ throw new Error('Invalid environment variables')
31
+ }
32
+
33
+ export const env = parsedEnv.data
34
+
35
+ // Export for easier access in server components
36
+ export const {
37
+ NODE_ENV,
38
+ NEXT_PUBLIC_SUPABASE_URL,
39
+ NEXT_PUBLIC_SUPABASE_ANON_KEY,
40
+ SUPABASE_SERVICE_ROLE_KEY,
41
+ NEXT_PUBLIC_STRIPE_PUBLISHABLE_KEY,
42
+ STRIPE_SECRET_KEY,
43
+ STRIPE_WEBHOOK_SECRET,
44
+ NEXT_PUBLIC_APP_URL,
45
+ DATABASE_URL,
46
+ } = env
@@ -0,0 +1,140 @@
1
+ /**
2
+ * Enhanced utility functions for {{titleCaseName}}
3
+ *
4
+ * This extends the basic utils with additional functionality while
5
+ * maintaining compatibility with existing templates.
6
+ */
7
+
8
+ import { type ClassValue, clsx } from "clsx"
9
+ import { twMerge } from "tailwind-merge"
10
+
11
+ // Core utility function (keeps existing functionality)
12
+ export function cn(...inputs: ClassValue[]) {
13
+ return twMerge(clsx(inputs))
14
+ }
15
+
16
+ // Additional utilities for enhanced templates
17
+
18
+ /**
19
+ * Format currency with proper locale support
20
+ */
21
+ export function formatCurrency(
22
+ amount: number,
23
+ currency: string = 'USD',
24
+ locale: string = 'en-US'
25
+ ): string {
26
+ return new Intl.NumberFormat(locale, {
27
+ style: 'currency',
28
+ currency,
29
+ }).format(amount)
30
+ }
31
+
32
+ /**
33
+ * Format date with relative time support
34
+ */
35
+ export function formatDate(
36
+ date: Date | string,
37
+ options?: Intl.DateTimeFormatOptions
38
+ ): string {
39
+ const dateObj = typeof date === 'string' ? new Date(date) : date
40
+
41
+ const defaultOptions: Intl.DateTimeFormatOptions = {
42
+ year: 'numeric',
43
+ month: 'short',
44
+ day: 'numeric',
45
+ }
46
+
47
+ return new Intl.DateTimeFormat('en-US', options || defaultOptions).format(dateObj)
48
+ }
49
+
50
+ /**
51
+ * Get relative time string (e.g., "2 hours ago")
52
+ */
53
+ export function getRelativeTime(date: Date | string): string {
54
+ const dateObj = typeof date === 'string' ? new Date(date) : date
55
+ const now = new Date()
56
+ const diffInMs = now.getTime() - dateObj.getTime()
57
+ const diffInSeconds = Math.floor(diffInMs / 1000)
58
+ const diffInMinutes = Math.floor(diffInSeconds / 60)
59
+ const diffInHours = Math.floor(diffInMinutes / 60)
60
+ const diffInDays = Math.floor(diffInHours / 24)
61
+
62
+ if (diffInSeconds < 60) return 'just now'
63
+ if (diffInMinutes < 60) return `${diffInMinutes} minute${diffInMinutes > 1 ? 's' : ''} ago`
64
+ if (diffInHours < 24) return `${diffInHours} hour${diffInHours > 1 ? 's' : ''} ago`
65
+ if (diffInDays < 7) return `${diffInDays} day${diffInDays > 1 ? 's' : ''} ago`
66
+
67
+ return formatDate(dateObj)
68
+ }
69
+
70
+ /**
71
+ * Truncate text to specified length with ellipsis
72
+ */
73
+ export function truncate(text: string, length: number = 50): string {
74
+ if (text.length <= length) return text
75
+ return text.slice(0, length).trim() + '...'
76
+ }
77
+
78
+ /**
79
+ * Generate initials from a name
80
+ */
81
+ export function getInitials(name: string): string {
82
+ return name
83
+ .split(' ')
84
+ .map(word => word.charAt(0).toUpperCase())
85
+ .slice(0, 2)
86
+ .join('')
87
+ }
88
+
89
+ /**
90
+ * Validate email format
91
+ */
92
+ export function isValidEmail(email: string): boolean {
93
+ const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/
94
+ return emailRegex.test(email)
95
+ }
96
+
97
+ /**
98
+ * Generate a random ID
99
+ */
100
+ export function generateId(prefix: string = ''): string {
101
+ const timestamp = Date.now().toString(36)
102
+ const randomString = Math.random().toString(36).substring(2)
103
+ return prefix ? `${prefix}-${timestamp}-${randomString}` : `${timestamp}-${randomString}`
104
+ }
105
+
106
+ /**
107
+ * Debounce function
108
+ */
109
+ export function debounce<T extends (...args: any[]) => any>(
110
+ func: T,
111
+ delay: number
112
+ ): (...args: Parameters<T>) => void {
113
+ let timeoutId: NodeJS.Timeout | null = null
114
+
115
+ return (...args: Parameters<T>) => {
116
+ if (timeoutId !== null) {
117
+ clearTimeout(timeoutId)
118
+ }
119
+ timeoutId = setTimeout(() => func(...args), delay)
120
+ }
121
+ }
122
+
123
+ /**
124
+ * Sleep/delay function for async operations
125
+ */
126
+ export function sleep(ms: number): Promise<void> {
127
+ return new Promise(resolve => setTimeout(resolve, ms))
128
+ }
129
+
130
+ /**
131
+ * Safe JSON parse with fallback
132
+ */
133
+ export function safeJsonParse<T>(json: string, fallback: T): T {
134
+ try {
135
+ return JSON.parse(json)
136
+ } catch {
137
+ return fallback
138
+ }
139
+ }
140
+
@@ -0,0 +1,79 @@
1
+ import '@testing-library/jest-dom'
2
+ import { vi, beforeEach, afterAll } from 'vitest'
3
+
4
+ // Mock Next.js router
5
+ vi.mock('next/navigation', () => ({
6
+ useRouter: () => ({
7
+ push: vi.fn(),
8
+ back: vi.fn(),
9
+ forward: vi.fn(),
10
+ refresh: vi.fn(),
11
+ replace: vi.fn(),
12
+ prefetch: vi.fn(),
13
+ }),
14
+ useSearchParams: () => new URLSearchParams(),
15
+ usePathname: () => '/',
16
+ }))
17
+
18
+ // Mock SaaS Factory Auth (optional - only if testing auth components)
19
+ vi.mock('@digilogiclabs/saas-factory-auth', () => ({
20
+ useAuth: () => ({
21
+ user: null,
22
+ loading: false,
23
+ error: null,
24
+ signIn: vi.fn(),
25
+ signUp: vi.fn(),
26
+ signOut: vi.fn(),
27
+ signInWithOAuth: vi.fn(),
28
+ }),
29
+ AuthProvider: ({ children }: { children: React.ReactNode }) => children,
30
+ }))
31
+
32
+ // Mock SaaS Factory Payments (optional - only if testing payment components)
33
+ vi.mock('@digilogiclabs/saas-factory-payments', () => ({
34
+ usePayments: () => ({
35
+ createCheckoutSession: vi.fn(),
36
+ createPortalSession: vi.fn(),
37
+ subscription: null,
38
+ loading: false,
39
+ }),
40
+ PaymentsProvider: ({ children }: { children: React.ReactNode }) => children,
41
+ }))
42
+
43
+ // Global test utilities
44
+ global.ResizeObserver = vi.fn().mockImplementation(() => ({
45
+ observe: vi.fn(),
46
+ unobserve: vi.fn(),
47
+ disconnect: vi.fn(),
48
+ }))
49
+
50
+ // Suppress console warnings in tests unless needed
51
+ const originalConsoleError = console.error
52
+ const originalConsoleWarn = console.warn
53
+
54
+ beforeEach(() => {
55
+ console.error = (...args: any[]) => {
56
+ if (
57
+ typeof args[0] === 'string' &&
58
+ args[0].includes('Warning: ReactDOM.render is no longer supported')
59
+ ) {
60
+ return
61
+ }
62
+ originalConsoleError.call(console, ...args)
63
+ }
64
+
65
+ console.warn = (...args: any[]) => {
66
+ if (
67
+ typeof args[0] === 'string' &&
68
+ args[0].includes('useLayoutEffect does nothing on the server')
69
+ ) {
70
+ return
71
+ }
72
+ originalConsoleWarn.call(console, ...args)
73
+ }
74
+ })
75
+
76
+ afterAll(() => {
77
+ console.error = originalConsoleError
78
+ console.warn = originalConsoleWarn
79
+ })
@@ -0,0 +1,77 @@
1
+ /** @type {import('tailwindcss').Config} */
2
+ module.exports = {
3
+ darkMode: ["class"],
4
+ content: [
5
+ './pages/**/*.{ts,tsx}',
6
+ './components/**/*.{ts,tsx}',
7
+ './app/**/*.{ts,tsx}',
8
+ './src/**/*.{ts,tsx}',
9
+ '../../node_modules/@digilogiclabs/saas-factory-ui/dist/**/*.{js,ts,jsx,tsx}',
10
+ ],
11
+ theme: {
12
+ container: {
13
+ center: true,
14
+ padding: "2rem",
15
+ screens: {
16
+ "2xl": "1400px",
17
+ },
18
+ },
19
+ extend: {
20
+ colors: {
21
+ border: "hsl(var(--border))",
22
+ input: "hsl(var(--input))",
23
+ ring: "hsl(var(--ring))",
24
+ background: "hsl(var(--background))",
25
+ foreground: "hsl(var(--foreground))",
26
+ primary: {
27
+ DEFAULT: "hsl(var(--primary))",
28
+ foreground: "hsl(var(--primary-foreground))",
29
+ },
30
+ secondary: {
31
+ DEFAULT: "hsl(var(--secondary))",
32
+ foreground: "hsl(var(--secondary-foreground))",
33
+ },
34
+ destructive: {
35
+ DEFAULT: "hsl(var(--destructive))",
36
+ foreground: "hsl(var(--destructive-foreground))",
37
+ },
38
+ muted: {
39
+ DEFAULT: "hsl(var(--muted))",
40
+ foreground: "hsl(var(--muted-foreground))",
41
+ },
42
+ accent: {
43
+ DEFAULT: "hsl(var(--accent))",
44
+ foreground: "hsl(var(--accent-foreground))",
45
+ },
46
+ popover: {
47
+ DEFAULT: "hsl(var(--popover))",
48
+ foreground: "hsl(var(--popover-foreground))",
49
+ },
50
+ card: {
51
+ DEFAULT: "hsl(var(--card))",
52
+ foreground: "hsl(var(--card-foreground))",
53
+ },
54
+ },
55
+ borderRadius: {
56
+ lg: "var(--radius)",
57
+ md: "calc(var(--radius) - 2px)",
58
+ sm: "calc(var(--radius) - 4px)",
59
+ },
60
+ keyframes: {
61
+ "accordion-down": {
62
+ from: { height: 0 },
63
+ to: { height: "var(--radix-accordion-content-height)" },
64
+ },
65
+ "accordion-up": {
66
+ from: { height: "var(--radix-accordion-content-height)" },
67
+ to: { height: 0 },
68
+ },
69
+ },
70
+ animation: {
71
+ "accordion-down": "accordion-down 0.2s ease-out",
72
+ "accordion-up": "accordion-up 0.2s ease-out",
73
+ },
74
+ },
75
+ },
76
+ plugins: [require("tailwindcss-animate")],
77
+ }
@@ -0,0 +1,33 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "es5",
4
+ "lib": ["dom", "dom.iterable", "es6"],
5
+ "allowJs": true,
6
+ "skipLibCheck": true,
7
+ "strict": true,
8
+ "noEmit": true,
9
+ "esModuleInterop": true,
10
+ "module": "esnext",
11
+ "moduleResolution": "bundler",
12
+ "resolveJsonModule": true,
13
+ "isolatedModules": true,
14
+ "jsx": "preserve",
15
+ "incremental": true,
16
+ "plugins": [
17
+ {
18
+ "name": "next"
19
+ }
20
+ ],
21
+ "baseUrl": ".",
22
+ "paths": {
23
+ "@/*": ["./src/*"],
24
+ "@/components/*": ["./src/components/*"],
25
+ "@/lib/*": ["./src/lib/*"],
26
+ "@/hooks/*": ["./src/hooks/*"],
27
+ "@/types/*": ["./src/types/*"]
28
+ }
29
+ },
30
+ "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts"],
31
+ "exclude": ["node_modules"]
32
+ }
33
+
@@ -0,0 +1,17 @@
1
+ import { defineConfig } from 'vitest/config'
2
+ import react from '@vitejs/plugin-react'
3
+ import path from 'path'
4
+
5
+ export default defineConfig({
6
+ plugins: [react()],
7
+ test: {
8
+ environment: 'jsdom',
9
+ setupFiles: ['./src/test/setup.ts'],
10
+ globals: true,
11
+ },
12
+ resolve: {
13
+ alias: {
14
+ '@': path.resolve(__dirname, './src'),
15
+ },
16
+ },
17
+ })
@@ -0,0 +1,15 @@
1
+ # Auth Configuration
2
+ NEXT_PUBLIC_AUTH_PROVIDER=supabase|firebase
3
+
4
+ # Supabase
5
+ NEXT_PUBLIC_SUPABASE_URL=
6
+ NEXT_PUBLIC_SUPABASE_ANON_KEY=
7
+
8
+ # Firebase
9
+ NEXT_PUBLIC_FIREBASE_API_KEY=
10
+ NEXT_PUBLIC_FIREBASE_AUTH_DOMAIN=
11
+ NEXT_PUBLIC_FIREBASE_PROJECT_ID=
12
+
13
+ # Payments
14
+ NEXT_PUBLIC_STRIPE_PUBLISHABLE_KEY=
15
+ STRIPE_SECRET_KEY=