@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,94 @@
1
+ 'use client'
2
+
3
+ import React, { createContext, useContext, useEffect } from 'react'
4
+
5
+ interface ThemeContextType {
6
+ themeColor: string
7
+ }
8
+
9
+ const ThemeContext = createContext<ThemeContextType | undefined>(undefined)
10
+
11
+ interface AppThemeProviderProps {
12
+ children: React.ReactNode
13
+ themeColor: string
14
+ }
15
+
16
+ const THEME_COLORS = {
17
+ blue: {
18
+ primary: '221.2 83.2% 53.3%',
19
+ primaryForeground: '210 40% 98%',
20
+ secondary: '210 40% 96%',
21
+ accent: '210 40% 96%',
22
+ ring: '221.2 83.2% 53.3%',
23
+ },
24
+ green: {
25
+ primary: '142.1 76.2% 36.3%',
26
+ primaryForeground: '355.7 100% 97.3%',
27
+ secondary: '138.5 76.2% 96.7%',
28
+ accent: '138.5 76.2% 96.7%',
29
+ ring: '142.1 76.2% 36.3%',
30
+ },
31
+ purple: {
32
+ primary: '262.1 83.3% 57.8%',
33
+ primaryForeground: '210 40% 98%',
34
+ secondary: '270 3.18% 96.5%',
35
+ accent: '270 3.18% 96.5%',
36
+ ring: '262.1 83.3% 57.8%',
37
+ },
38
+ orange: {
39
+ primary: '24.6 95% 53.1%',
40
+ primaryForeground: '60 9.1% 97.8%',
41
+ secondary: '60 4.8% 95.9%',
42
+ accent: '60 4.8% 95.9%',
43
+ ring: '24.6 95% 53.1%',
44
+ },
45
+ red: {
46
+ primary: '0 72.2% 50.6%',
47
+ primaryForeground: '0 85.7% 97.3%',
48
+ secondary: '0 0% 96.3%',
49
+ accent: '0 0% 96.3%',
50
+ ring: '0 72.2% 50.6%',
51
+ },
52
+ slate: {
53
+ primary: '215 27.9% 16.9%',
54
+ primaryForeground: '0 0% 98%',
55
+ secondary: '210 40% 96%',
56
+ accent: '210 40% 96%',
57
+ ring: '215 27.9% 16.9%',
58
+ },
59
+ }
60
+
61
+ export function AppThemeProvider({ children, themeColor }: AppThemeProviderProps) {
62
+ useEffect(() => {
63
+ const colorScheme = THEME_COLORS[themeColor as keyof typeof THEME_COLORS] || THEME_COLORS.blue
64
+
65
+ // Apply theme colors to CSS custom properties
66
+ const root = document.documentElement
67
+ root.style.setProperty('--primary', colorScheme.primary)
68
+ root.style.setProperty('--primary-foreground', colorScheme.primaryForeground)
69
+ root.style.setProperty('--secondary', colorScheme.secondary)
70
+ root.style.setProperty('--accent', colorScheme.accent)
71
+ root.style.setProperty('--ring', colorScheme.ring)
72
+
73
+ // Store theme color preference
74
+ localStorage.setItem('{{packageName}}-theme-color', themeColor)
75
+ }, [themeColor])
76
+
77
+ const contextValue: ThemeContextType = {
78
+ themeColor,
79
+ }
80
+
81
+ return (
82
+ <ThemeContext.Provider value={contextValue}>
83
+ {children}
84
+ </ThemeContext.Provider>
85
+ )
86
+ }
87
+
88
+ export function useAppTheme() {
89
+ const context = useContext(ThemeContext)
90
+ if (context === undefined) {
91
+ throw new Error('useAppTheme must be used within a AppThemeProvider')
92
+ }
93
+ return context
94
+ }
@@ -0,0 +1,36 @@
1
+ 'use client'
2
+
3
+ import React from 'react'
4
+
5
+ interface FooterProps {
6
+ className?: string
7
+ }
8
+
9
+ export function Footer({ className = '' }: FooterProps) {
10
+ return (
11
+ <footer className={`border-t bg-background py-4 ${className}`}>
12
+ <div className="container mx-auto px-4">
13
+ <div className="flex flex-col sm:flex-row justify-between items-center gap-4 text-sm text-muted-foreground">
14
+ <div className="flex items-center gap-2">
15
+ <span>&copy; {new Date().getFullYear()} {{titleCaseName}}</span>
16
+ </div>
17
+
18
+ <div className="flex items-center gap-2 text-xs">
19
+ <span>Generated with</span>
20
+ <a
21
+ href="https://docs.digilogiclabs.com"
22
+ target="_blank"
23
+ rel="noopener noreferrer"
24
+ className="hover:text-foreground transition-colors"
25
+ >
26
+ Digi Logic Labs
27
+ </a>
28
+ <span className="text-muted-foreground/60">
29
+ • UI v{{uiVersion}} • Auth v{{authVersion}} • Payments v{{paymentsVersion}} • Video v{{videoVersion}} • {{generatedDate}}
30
+ </span>
31
+ </div>
32
+ </div>
33
+ </div>
34
+ </footer>
35
+ )
36
+ }
@@ -0,0 +1,62 @@
1
+ 'use client';
2
+
3
+ import React from 'react';
4
+ import Link from 'next/link';
5
+ import { LogOut, CreditCard } from 'lucide-react';
6
+ import { useAuth } from '@digilogiclabs/saas-factory-auth';
7
+ import { ThemeToggle } from '../ui/theme-toggle';
8
+
9
+ export function Header() {
10
+ const { user, signOut } = useAuth();
11
+
12
+ const handleSignOut = async () => {
13
+ try {
14
+ await signOut();
15
+ } catch (error) {
16
+ console.error('Sign out error:', error);
17
+ }
18
+ };
19
+
20
+ return (
21
+ <header className="bg-white dark:bg-gray-800 shadow-md">
22
+ <div className="container mx-auto px-4 py-4 flex justify-between items-center">
23
+ <Link href="/" className="text-2xl font-bold text-gray-900 dark:text-white">
24
+ {{titleCaseName}}
25
+ </Link>
26
+ <nav className="flex items-center gap-4">
27
+ <ThemeToggle />
28
+ {user ? (
29
+ <>
30
+ <Link href="/dashboard" className="text-gray-600 dark:text-gray-300 hover:text-gray-900 dark:hover:text-white">
31
+ Dashboard
32
+ </Link>
33
+ <Link href="/billing" className="flex items-center gap-2 text-gray-600 dark:text-gray-300 hover:text-gray-900 dark:hover:text-white">
34
+ <CreditCard className="h-4 w-4" />
35
+ Billing
36
+ </Link>
37
+ <button
38
+ onClick={handleSignOut}
39
+ className="p-2 text-gray-600 dark:text-gray-300 hover:text-gray-900 dark:hover:text-white"
40
+ >
41
+ <LogOut className="h-5 w-5" />
42
+ </button>
43
+ </>
44
+ ) : (
45
+ <>
46
+ <Link href="/login" className="text-gray-600 dark:text-gray-300 hover:text-gray-900 dark:hover:text-white">
47
+ Login
48
+ </Link>
49
+ <Link
50
+ href="/signup"
51
+ className="px-4 py-2 bg-blue-600 text-white rounded hover:bg-blue-700"
52
+ >
53
+ Sign Up
54
+ </Link>
55
+ </>
56
+ )}
57
+ </nav>
58
+ </div>
59
+ </header>
60
+ );
61
+ }
62
+
@@ -0,0 +1,36 @@
1
+ import * as React from "react"
2
+ import { cva, type VariantProps } from "class-variance-authority"
3
+ import { cn } from "@/lib/utils"
4
+
5
+ const badgeVariants = cva(
6
+ "inline-flex items-center rounded-full border px-2.5 py-0.5 text-xs font-semibold transition-colors focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2",
7
+ {
8
+ variants: {
9
+ variant: {
10
+ default:
11
+ "border-transparent bg-primary text-primary-foreground hover:bg-primary/80",
12
+ secondary:
13
+ "border-transparent bg-secondary text-secondary-foreground hover:bg-secondary/80",
14
+ destructive:
15
+ "border-transparent bg-destructive text-destructive-foreground hover:bg-destructive/80",
16
+ outline: "text-foreground",
17
+ },
18
+ },
19
+ defaultVariants: {
20
+ variant: "default",
21
+ },
22
+ }
23
+ )
24
+
25
+ export interface BadgeProps
26
+ extends React.HTMLAttributes<HTMLDivElement>,
27
+ VariantProps<typeof badgeVariants> {}
28
+
29
+ function Badge({ className, variant, ...props }: BadgeProps) {
30
+ return (
31
+ <div className={cn(badgeVariants({ variant }), className)} {...props} />
32
+ )
33
+ }
34
+
35
+ export { Badge, badgeVariants }
36
+
@@ -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 "@digilogiclabs/saas-factory-ui"
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,49 @@
1
+ /**
2
+ * Optional Environment Configuration with Zod validation
3
+ *
4
+ * This provides type-safe environment variable validation.
5
+ * Usage is optional - existing templates will continue to work without this.
6
+ *
7
+ * To use: import { env } from '@/lib/env' instead of process.env
8
+ */
9
+
10
+ import { z } from 'zod'
11
+
12
+ const envSchema = z.object({
13
+ // Next.js built-in
14
+ NODE_ENV: z.enum(['development', 'production', 'test']).default('development'),
15
+
16
+ // App configuration
17
+ NEXT_PUBLIC_APP_URL: z.string().url().optional(),
18
+ NEXT_PUBLIC_APP_NAME: z.string().default('{{titleCaseName}}'),
19
+
20
+ // Auth configuration (if using server-side auth)
21
+ SUPABASE_URL: z.string().url().optional(),
22
+ SUPABASE_ANON_KEY: z.string().optional(),
23
+ SUPABASE_SERVICE_ROLE_KEY: z.string().optional(),
24
+
25
+ // Payment configuration
26
+ STRIPE_SECRET_KEY: z.string().optional(),
27
+ STRIPE_WEBHOOK_SECRET: z.string().optional(),
28
+
29
+ // Optional analytics
30
+ ANALYTICS_ID: z.string().optional(),
31
+ })
32
+
33
+ // Parse with fallback to process.env for backward compatibility
34
+ function parseEnv() {
35
+ try {
36
+ return envSchema.parse(process.env)
37
+ } catch (error) {
38
+ if (process.env.NODE_ENV === 'development') {
39
+ console.warn('Environment validation failed. Using process.env fallback:', error)
40
+ }
41
+ // Fallback to process.env for backward compatibility
42
+ return process.env as any
43
+ }
44
+ }
45
+
46
+ export const env = parseEnv()
47
+
48
+ // Type helper for environment variables
49
+ export type Env = z.infer<typeof envSchema>
@@ -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: [],
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=
@@ -0,0 +1,8 @@
1
+ /** @type {import('eslint').Linter.Config} */
2
+ module.exports = {
3
+ extends: ['next/core-web-vitals'],
4
+ rules: {
5
+ // Add any custom rules here
6
+ },
7
+ }
8
+