@digilogiclabs/create-saas-app 1.4.0 → 1.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (199) hide show
  1. package/CHANGELOG.md +42 -0
  2. package/dist/.tsbuildinfo +1 -1
  3. package/dist/index.js +45 -3
  4. package/dist/index.js.map +1 -1
  5. package/dist/templates/web/base/template/package.json +20 -10
  6. package/dist/templates/web/base/template/src/app/error.tsx +97 -0
  7. package/dist/templates/web/base/template/src/app/layout.tsx +8 -2
  8. package/dist/templates/web/base/template/src/app/loading.tsx +34 -0
  9. package/dist/templates/web/base/template/src/components/__tests__/example.test.tsx +49 -0
  10. package/dist/templates/web/base/template/src/components/providers/app-providers.tsx +6 -2
  11. package/dist/templates/web/base/template/src/components/providers/theme-provider.tsx +94 -0
  12. package/dist/templates/web/base/template/src/components/shared/footer.tsx +36 -0
  13. package/dist/templates/web/base/template/src/components/shared/header.tsx +2 -0
  14. package/dist/templates/web/base/template/src/components/ui/theme-toggle.tsx +34 -0
  15. package/dist/templates/web/base/template/src/lib/auth-server.ts +177 -0
  16. package/dist/templates/web/base/template/src/lib/env.ts +46 -0
  17. package/dist/templates/web/base/template/src/lib/utils.ts +133 -0
  18. package/dist/templates/web/base/template/src/test/setup.ts +79 -0
  19. package/dist/templates/web/base/template/vitest.config.ts +17 -0
  20. package/dist/templates/web/ui-auth/template/package.json +14 -4
  21. package/dist/templates/web/ui-auth/template/src/app/error.tsx +67 -0
  22. package/dist/templates/web/ui-auth/template/src/app/layout.tsx +6 -2
  23. package/dist/templates/web/ui-auth/template/src/app/loading.tsx +20 -0
  24. package/dist/templates/web/ui-auth/template/src/components/__tests__/example.test.tsx +49 -0
  25. package/dist/templates/web/ui-auth/template/src/components/providers/app-providers.tsx +6 -2
  26. package/dist/templates/web/ui-auth/template/src/components/providers/theme-provider.tsx +94 -0
  27. package/dist/templates/web/ui-auth/template/src/components/shared/footer.tsx +36 -0
  28. package/dist/templates/web/ui-auth/template/src/components/shared/header.tsx +2 -0
  29. package/dist/templates/web/ui-auth/template/src/components/ui/theme-toggle.tsx +34 -0
  30. package/dist/templates/web/ui-auth/template/src/lib/env.ts +49 -0
  31. package/dist/templates/web/ui-auth/template/src/lib/utils.ts +133 -0
  32. package/dist/templates/web/ui-auth/template/src/test/setup.ts +79 -0
  33. package/dist/templates/web/ui-auth/template/vitest.config.ts +17 -0
  34. package/dist/templates/web/ui-auth-payments/template/middleware.ts +68 -0
  35. package/dist/templates/web/ui-auth-payments/template/package.json +14 -4
  36. package/dist/templates/web/ui-auth-payments/template/src/app/dashboard/layout.tsx +22 -0
  37. package/dist/templates/web/ui-auth-payments/template/src/app/dashboard/page.tsx +183 -0
  38. package/dist/templates/web/ui-auth-payments/template/src/app/error.tsx +67 -0
  39. package/dist/templates/web/ui-auth-payments/template/src/app/layout.tsx +6 -2
  40. package/dist/templates/web/ui-auth-payments/template/src/app/loading.tsx +20 -0
  41. package/dist/templates/web/ui-auth-payments/template/src/app/login/loading.tsx +38 -0
  42. package/dist/templates/web/ui-auth-payments/template/src/app/signup/loading.tsx +50 -0
  43. package/dist/templates/web/ui-auth-payments/template/src/components/__tests__/example.test.tsx +49 -0
  44. package/dist/templates/web/ui-auth-payments/template/src/components/client/auth-status.tsx +52 -0
  45. package/dist/templates/web/ui-auth-payments/template/src/components/client/login-form.tsx +144 -0
  46. package/dist/templates/web/ui-auth-payments/template/src/components/client/newsletter-signup.tsx +68 -0
  47. package/dist/templates/web/ui-auth-payments/template/src/components/client/signup-form.tsx +185 -0
  48. package/dist/templates/web/ui-auth-payments/template/src/components/providers/app-providers.tsx +6 -2
  49. package/dist/templates/web/ui-auth-payments/template/src/components/providers/theme-provider.tsx +94 -0
  50. package/dist/templates/web/ui-auth-payments/template/src/components/shared/footer.tsx +36 -0
  51. package/dist/templates/web/ui-auth-payments/template/src/components/shared/header.tsx +2 -0
  52. package/dist/templates/web/ui-auth-payments/template/src/components/ui/theme-toggle.tsx +34 -0
  53. package/dist/templates/web/ui-auth-payments/template/src/lib/actions/auth.ts +246 -0
  54. package/dist/templates/web/ui-auth-payments/template/src/lib/actions/index.ts +340 -0
  55. package/dist/templates/web/ui-auth-payments/template/src/lib/auth-server.ts +177 -0
  56. package/dist/templates/web/ui-auth-payments/template/src/lib/env.ts +49 -0
  57. package/dist/templates/web/ui-auth-payments/template/src/lib/utils.ts +133 -0
  58. package/dist/templates/web/ui-auth-payments/template/src/test/setup.ts +79 -0
  59. package/dist/templates/web/ui-auth-payments/template/vitest.config.ts +17 -0
  60. package/dist/templates/web/ui-auth-payments-audio/template/package.json +14 -4
  61. package/dist/templates/web/ui-auth-payments-audio/template/src/app/error.tsx +67 -0
  62. package/dist/templates/web/ui-auth-payments-audio/template/src/app/layout.tsx +8 -2
  63. package/dist/templates/web/ui-auth-payments-audio/template/src/app/loading.tsx +20 -0
  64. package/dist/templates/web/ui-auth-payments-audio/template/src/components/__tests__/example.test.tsx +49 -0
  65. package/dist/templates/web/ui-auth-payments-audio/template/src/components/providers/app-providers.tsx +6 -2
  66. package/dist/templates/web/ui-auth-payments-audio/template/src/components/providers/theme-provider.tsx +94 -0
  67. package/dist/templates/web/ui-auth-payments-audio/template/src/components/shared/footer.tsx +36 -0
  68. package/dist/templates/web/ui-auth-payments-audio/template/src/components/shared/header.tsx +2 -0
  69. package/dist/templates/web/ui-auth-payments-audio/template/src/components/ui/theme-toggle.tsx +34 -0
  70. package/dist/templates/web/ui-auth-payments-audio/template/src/lib/env.ts +49 -0
  71. package/dist/templates/web/ui-auth-payments-audio/template/src/lib/utils.ts +133 -0
  72. package/dist/templates/web/ui-auth-payments-audio/template/src/test/setup.ts +79 -0
  73. package/dist/templates/web/ui-auth-payments-audio/template/vitest.config.ts +17 -0
  74. package/dist/templates/web/ui-auth-payments-video/template/package.json +14 -4
  75. package/dist/templates/web/ui-auth-payments-video/template/src/app/error.tsx +67 -0
  76. package/dist/templates/web/ui-auth-payments-video/template/src/app/layout.tsx +6 -2
  77. package/dist/templates/web/ui-auth-payments-video/template/src/app/loading.tsx +20 -0
  78. package/dist/templates/web/ui-auth-payments-video/template/src/components/__tests__/example.test.tsx +49 -0
  79. package/dist/templates/web/ui-auth-payments-video/template/src/components/providers/app-providers.tsx +6 -2
  80. package/dist/templates/web/ui-auth-payments-video/template/src/components/providers/theme-provider.tsx +94 -0
  81. package/dist/templates/web/ui-auth-payments-video/template/src/components/shared/footer.tsx +36 -0
  82. package/dist/templates/web/ui-auth-payments-video/template/src/components/shared/header.tsx +2 -0
  83. package/dist/templates/web/ui-auth-payments-video/template/src/components/ui/theme-toggle.tsx +34 -0
  84. package/dist/templates/web/ui-auth-payments-video/template/src/lib/env.ts +49 -0
  85. package/dist/templates/web/ui-auth-payments-video/template/src/lib/utils.ts +133 -0
  86. package/dist/templates/web/ui-auth-payments-video/template/src/test/setup.ts +79 -0
  87. package/dist/templates/web/ui-auth-payments-video/template/vitest.config.ts +17 -0
  88. package/dist/templates/web/ui-only/template/package.json +14 -4
  89. package/dist/templates/web/ui-only/template/src/app/error.tsx +67 -0
  90. package/dist/templates/web/ui-only/template/src/app/layout.tsx +6 -2
  91. package/dist/templates/web/ui-only/template/src/app/loading.tsx +20 -0
  92. package/dist/templates/web/ui-only/template/src/components/__tests__/example.test.tsx +49 -0
  93. package/dist/templates/web/ui-only/template/src/components/providers/app-providers.tsx +6 -2
  94. package/dist/templates/web/ui-only/template/src/components/providers/theme-provider.tsx +94 -0
  95. package/dist/templates/web/ui-only/template/src/components/shared/footer.tsx +36 -0
  96. package/dist/templates/web/ui-only/template/src/components/shared/header.tsx +2 -0
  97. package/dist/templates/web/ui-only/template/src/components/ui/theme-toggle.tsx +34 -0
  98. package/dist/templates/web/ui-only/template/src/lib/env.ts +49 -0
  99. package/dist/templates/web/ui-only/template/src/lib/utils.ts +133 -0
  100. package/dist/templates/web/ui-only/template/src/test/setup.ts +79 -0
  101. package/dist/templates/web/ui-only/template/vitest.config.ts +17 -0
  102. package/package.json +1 -1
  103. package/src/templates/web/base/template/package.json +20 -10
  104. package/src/templates/web/base/template/src/app/error.tsx +97 -0
  105. package/src/templates/web/base/template/src/app/layout.tsx +8 -2
  106. package/src/templates/web/base/template/src/app/loading.tsx +34 -0
  107. package/src/templates/web/base/template/src/components/__tests__/example.test.tsx +49 -0
  108. package/src/templates/web/base/template/src/components/providers/app-providers.tsx +6 -2
  109. package/src/templates/web/base/template/src/components/providers/theme-provider.tsx +94 -0
  110. package/src/templates/web/base/template/src/components/shared/footer.tsx +36 -0
  111. package/src/templates/web/base/template/src/components/shared/header.tsx +2 -0
  112. package/src/templates/web/base/template/src/components/ui/theme-toggle.tsx +34 -0
  113. package/src/templates/web/base/template/src/lib/auth-server.ts +177 -0
  114. package/src/templates/web/base/template/src/lib/env.ts +46 -0
  115. package/src/templates/web/base/template/src/lib/utils.ts +133 -0
  116. package/src/templates/web/base/template/src/test/setup.ts +79 -0
  117. package/src/templates/web/base/template/vitest.config.ts +17 -0
  118. package/src/templates/web/ui-auth/template/package.json +14 -4
  119. package/src/templates/web/ui-auth/template/src/app/error.tsx +67 -0
  120. package/src/templates/web/ui-auth/template/src/app/layout.tsx +6 -2
  121. package/src/templates/web/ui-auth/template/src/app/loading.tsx +20 -0
  122. package/src/templates/web/ui-auth/template/src/components/__tests__/example.test.tsx +49 -0
  123. package/src/templates/web/ui-auth/template/src/components/providers/app-providers.tsx +6 -2
  124. package/src/templates/web/ui-auth/template/src/components/providers/theme-provider.tsx +94 -0
  125. package/src/templates/web/ui-auth/template/src/components/shared/footer.tsx +36 -0
  126. package/src/templates/web/ui-auth/template/src/components/shared/header.tsx +2 -0
  127. package/src/templates/web/ui-auth/template/src/components/ui/theme-toggle.tsx +34 -0
  128. package/src/templates/web/ui-auth/template/src/lib/env.ts +49 -0
  129. package/src/templates/web/ui-auth/template/src/lib/utils.ts +133 -0
  130. package/src/templates/web/ui-auth/template/src/test/setup.ts +79 -0
  131. package/src/templates/web/ui-auth/template/vitest.config.ts +17 -0
  132. package/src/templates/web/ui-auth-payments/template/middleware.ts +68 -0
  133. package/src/templates/web/ui-auth-payments/template/package.json +14 -4
  134. package/src/templates/web/ui-auth-payments/template/src/app/dashboard/layout.tsx +22 -0
  135. package/src/templates/web/ui-auth-payments/template/src/app/dashboard/page.tsx +183 -0
  136. package/src/templates/web/ui-auth-payments/template/src/app/error.tsx +67 -0
  137. package/src/templates/web/ui-auth-payments/template/src/app/layout.tsx +6 -2
  138. package/src/templates/web/ui-auth-payments/template/src/app/loading.tsx +20 -0
  139. package/src/templates/web/ui-auth-payments/template/src/app/login/loading.tsx +38 -0
  140. package/src/templates/web/ui-auth-payments/template/src/app/signup/loading.tsx +50 -0
  141. package/src/templates/web/ui-auth-payments/template/src/components/__tests__/example.test.tsx +49 -0
  142. package/src/templates/web/ui-auth-payments/template/src/components/client/auth-status.tsx +52 -0
  143. package/src/templates/web/ui-auth-payments/template/src/components/client/login-form.tsx +144 -0
  144. package/src/templates/web/ui-auth-payments/template/src/components/client/newsletter-signup.tsx +68 -0
  145. package/src/templates/web/ui-auth-payments/template/src/components/client/signup-form.tsx +185 -0
  146. package/src/templates/web/ui-auth-payments/template/src/components/providers/app-providers.tsx +6 -2
  147. package/src/templates/web/ui-auth-payments/template/src/components/providers/theme-provider.tsx +94 -0
  148. package/src/templates/web/ui-auth-payments/template/src/components/shared/footer.tsx +36 -0
  149. package/src/templates/web/ui-auth-payments/template/src/components/shared/header.tsx +2 -0
  150. package/src/templates/web/ui-auth-payments/template/src/components/ui/theme-toggle.tsx +34 -0
  151. package/src/templates/web/ui-auth-payments/template/src/lib/actions/auth.ts +246 -0
  152. package/src/templates/web/ui-auth-payments/template/src/lib/actions/index.ts +340 -0
  153. package/src/templates/web/ui-auth-payments/template/src/lib/auth-server.ts +177 -0
  154. package/src/templates/web/ui-auth-payments/template/src/lib/env.ts +49 -0
  155. package/src/templates/web/ui-auth-payments/template/src/lib/utils.ts +133 -0
  156. package/src/templates/web/ui-auth-payments/template/src/test/setup.ts +79 -0
  157. package/src/templates/web/ui-auth-payments/template/vitest.config.ts +17 -0
  158. package/src/templates/web/ui-auth-payments-audio/template/package.json +14 -4
  159. package/src/templates/web/ui-auth-payments-audio/template/src/app/error.tsx +67 -0
  160. package/src/templates/web/ui-auth-payments-audio/template/src/app/layout.tsx +8 -2
  161. package/src/templates/web/ui-auth-payments-audio/template/src/app/loading.tsx +20 -0
  162. package/src/templates/web/ui-auth-payments-audio/template/src/components/__tests__/example.test.tsx +49 -0
  163. package/src/templates/web/ui-auth-payments-audio/template/src/components/providers/app-providers.tsx +6 -2
  164. package/src/templates/web/ui-auth-payments-audio/template/src/components/providers/theme-provider.tsx +94 -0
  165. package/src/templates/web/ui-auth-payments-audio/template/src/components/shared/footer.tsx +36 -0
  166. package/src/templates/web/ui-auth-payments-audio/template/src/components/shared/header.tsx +2 -0
  167. package/src/templates/web/ui-auth-payments-audio/template/src/components/ui/theme-toggle.tsx +34 -0
  168. package/src/templates/web/ui-auth-payments-audio/template/src/lib/env.ts +49 -0
  169. package/src/templates/web/ui-auth-payments-audio/template/src/lib/utils.ts +133 -0
  170. package/src/templates/web/ui-auth-payments-audio/template/src/test/setup.ts +79 -0
  171. package/src/templates/web/ui-auth-payments-audio/template/vitest.config.ts +17 -0
  172. package/src/templates/web/ui-auth-payments-video/template/package.json +14 -4
  173. package/src/templates/web/ui-auth-payments-video/template/src/app/error.tsx +67 -0
  174. package/src/templates/web/ui-auth-payments-video/template/src/app/layout.tsx +6 -2
  175. package/src/templates/web/ui-auth-payments-video/template/src/app/loading.tsx +20 -0
  176. package/src/templates/web/ui-auth-payments-video/template/src/components/__tests__/example.test.tsx +49 -0
  177. package/src/templates/web/ui-auth-payments-video/template/src/components/providers/app-providers.tsx +6 -2
  178. package/src/templates/web/ui-auth-payments-video/template/src/components/providers/theme-provider.tsx +94 -0
  179. package/src/templates/web/ui-auth-payments-video/template/src/components/shared/footer.tsx +36 -0
  180. package/src/templates/web/ui-auth-payments-video/template/src/components/shared/header.tsx +2 -0
  181. package/src/templates/web/ui-auth-payments-video/template/src/components/ui/theme-toggle.tsx +34 -0
  182. package/src/templates/web/ui-auth-payments-video/template/src/lib/env.ts +49 -0
  183. package/src/templates/web/ui-auth-payments-video/template/src/lib/utils.ts +133 -0
  184. package/src/templates/web/ui-auth-payments-video/template/src/test/setup.ts +79 -0
  185. package/src/templates/web/ui-auth-payments-video/template/vitest.config.ts +17 -0
  186. package/src/templates/web/ui-only/template/package.json +14 -4
  187. package/src/templates/web/ui-only/template/src/app/error.tsx +67 -0
  188. package/src/templates/web/ui-only/template/src/app/layout.tsx +6 -2
  189. package/src/templates/web/ui-only/template/src/app/loading.tsx +20 -0
  190. package/src/templates/web/ui-only/template/src/components/__tests__/example.test.tsx +49 -0
  191. package/src/templates/web/ui-only/template/src/components/providers/app-providers.tsx +6 -2
  192. package/src/templates/web/ui-only/template/src/components/providers/theme-provider.tsx +94 -0
  193. package/src/templates/web/ui-only/template/src/components/shared/footer.tsx +36 -0
  194. package/src/templates/web/ui-only/template/src/components/shared/header.tsx +2 -0
  195. package/src/templates/web/ui-only/template/src/components/ui/theme-toggle.tsx +34 -0
  196. package/src/templates/web/ui-only/template/src/lib/env.ts +49 -0
  197. package/src/templates/web/ui-only/template/src/lib/utils.ts +133 -0
  198. package/src/templates/web/ui-only/template/src/test/setup.ts +79 -0
  199. package/src/templates/web/ui-only/template/vitest.config.ts +17 -0
@@ -0,0 +1,68 @@
1
+ import { NextRequest, NextResponse } from 'next/server'
2
+ import { env } from './src/lib/env'
3
+
4
+ export async function middleware(request: NextRequest) {
5
+ // Get auth token from cookies
6
+ const authToken = request.cookies.get('saas-factory-auth-token')?.value
7
+ const user = request.cookies.get('saas-factory-auth-user')?.value
8
+
9
+ const isAuthenticated = !!(authToken && user)
10
+ const { pathname } = request.nextUrl
11
+
12
+ // Define protected routes
13
+ const protectedRoutes = ['/dashboard', '/profile', '/settings']
14
+ const authRoutes = ['/login', '/signup']
15
+
16
+ // Check if the current path is protected
17
+ const isProtectedRoute = protectedRoutes.some(route =>
18
+ pathname.startsWith(route)
19
+ )
20
+
21
+ // Check if the current path is an auth route
22
+ const isAuthRoute = authRoutes.some(route =>
23
+ pathname.startsWith(route)
24
+ )
25
+
26
+ // Redirect unauthenticated users from protected routes
27
+ if (isProtectedRoute && !isAuthenticated) {
28
+ const loginUrl = new URL('/login', request.url)
29
+ loginUrl.searchParams.set('from', pathname)
30
+ return NextResponse.redirect(loginUrl)
31
+ }
32
+
33
+ // Redirect authenticated users from auth routes
34
+ if (isAuthRoute && isAuthenticated) {
35
+ const redirectUrl = request.nextUrl.searchParams.get('from') || '/'
36
+ return NextResponse.redirect(new URL(redirectUrl, request.url))
37
+ }
38
+
39
+ // Add security headers
40
+ const response = NextResponse.next()
41
+
42
+ // Add CSRF protection headers
43
+ response.headers.set('X-Frame-Options', 'DENY')
44
+ response.headers.set('X-Content-Type-Options', 'nosniff')
45
+ response.headers.set('Referrer-Policy', 'origin-when-cross-origin')
46
+
47
+ // Add CSP header for additional security
48
+ if (env.NODE_ENV === 'production') {
49
+ response.headers.set(
50
+ 'Content-Security-Policy',
51
+ "default-src 'self'; script-src 'self' 'unsafe-eval' 'unsafe-inline' https://js.stripe.com; style-src 'self' 'unsafe-inline' https://fonts.googleapis.com; font-src 'self' https://fonts.gstatic.com; img-src 'self' data: https:; connect-src 'self' https://api.stripe.com"
52
+ )
53
+ }
54
+
55
+ return response
56
+ }
57
+
58
+ export const config = {
59
+ matcher: [
60
+ // Match all request paths except for the ones starting with:
61
+ // - api (API routes)
62
+ // - _next/static (static files)
63
+ // - _next/image (image optimization files)
64
+ // - favicon.ico (favicon file)
65
+ // - public folder files
66
+ '/((?!api|_next/static|_next/image|favicon.ico|.*\\.(?:svg|png|jpg|jpeg|gif|webp)$).*)',
67
+ ],
68
+ }
@@ -4,11 +4,14 @@
4
4
  "description": "{{description}} (with UI Package v0.11.1 + Auth v1.0.0 + Payments)",
5
5
  "private": true,
6
6
  "scripts": {
7
- "dev": "next dev",
7
+ "dev": "next dev --turbo",
8
8
  "build": "next build",
9
9
  "start": "next start",
10
10
  "lint": "next lint",
11
- "type-check": "tsc --noEmit"
11
+ "type-check": "tsc --noEmit",
12
+ "test": "vitest",
13
+ "test:ui": "vitest --ui",
14
+ "test:run": "vitest run"
12
15
  },
13
16
  "dependencies": {
14
17
  "next": "^15.0.0",
@@ -25,7 +28,8 @@
25
28
  "class-variance-authority": "^0.7.0",
26
29
  "tailwind-merge": "^2.0.0",
27
30
  "next-themes": "^0.2.1",
28
- "lucide-react": "^0.542.0"
31
+ "lucide-react": "^0.542.0",
32
+ "zod": "^3.22.4"
29
33
  },
30
34
  "devDependencies": {
31
35
  "typescript": "^5.0.0",
@@ -34,7 +38,13 @@
34
38
  "@types/react-dom": "^19.0.0",
35
39
  "eslint": "^8.0.0",
36
40
  "eslint-config-next": "^15.0.0",
37
- "prettier": "^3.0.0"
41
+ "prettier": "^3.0.0",
42
+ "vitest": "^1.0.0",
43
+ "@vitejs/plugin-react": "^4.0.0",
44
+ "@testing-library/react": "^14.0.0",
45
+ "@testing-library/jest-dom": "^6.0.0",
46
+ "@testing-library/user-event": "^14.0.0",
47
+ "jsdom": "^24.0.0"
38
48
  },
39
49
  "engines": {
40
50
  "node": ">=18.0.0"
@@ -0,0 +1,22 @@
1
+ import { requireAuth } from '@/lib/auth-server'
2
+ import { redirect } from 'next/navigation'
3
+
4
+ export default async function DashboardLayout({
5
+ children,
6
+ }: {
7
+ children: React.ReactNode
8
+ }) {
9
+ // Ensure user is authenticated at the layout level
10
+ try {
11
+ await requireAuth()
12
+ } catch (error) {
13
+ // This will happen if requireAuth redirects, but just in case
14
+ redirect('/login')
15
+ }
16
+
17
+ return (
18
+ <div className="dashboard-layout">
19
+ {children}
20
+ </div>
21
+ )
22
+ }
@@ -0,0 +1,183 @@
1
+ import { Suspense } from 'react'
2
+ import { Button, Card } from '@digilogiclabs/saas-factory-ui'
3
+ import { User, Settings, CreditCard, Activity } from 'lucide-react'
4
+ import { requireAuth } from '@/lib/auth-server'
5
+ import Link from 'next/link'
6
+
7
+ // Example of server component data fetching
8
+ async function getUserStats(userId: string) {
9
+ // Simulate API call
10
+ await new Promise(resolve => setTimeout(resolve, 100))
11
+
12
+ return {
13
+ totalPosts: 42,
14
+ totalViews: 1337,
15
+ totalLikes: 256,
16
+ joinedDate: '2024-01-15'
17
+ }
18
+ }
19
+
20
+ function StatsCard({ title, value, icon: Icon, href }: {
21
+ title: string
22
+ value: string | number
23
+ icon: React.ComponentType<any>
24
+ href?: string
25
+ }) {
26
+ const content = (
27
+ <Card className="p-6 hover:shadow-lg transition-shadow">
28
+ <div className="flex items-center justify-between">
29
+ <div>
30
+ <p className="text-sm font-medium text-gray-600 dark:text-gray-400">{title}</p>
31
+ <p className="text-2xl font-bold text-gray-900 dark:text-white">{value}</p>
32
+ </div>
33
+ <Icon className="h-8 w-8 text-blue-600 dark:text-blue-400" />
34
+ </div>
35
+ </Card>
36
+ )
37
+
38
+ if (href) {
39
+ return <Link href={href}>{content}</Link>
40
+ }
41
+
42
+ return content
43
+ }
44
+
45
+ async function UserStats({ userId }: { userId: string }) {
46
+ const stats = await getUserStats(userId)
47
+
48
+ return (
49
+ <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6 mb-8">
50
+ <StatsCard
51
+ title="Total Posts"
52
+ value={stats.totalPosts}
53
+ icon={Activity}
54
+ />
55
+ <StatsCard
56
+ title="Total Views"
57
+ value={stats.totalViews.toLocaleString()}
58
+ icon={Activity}
59
+ />
60
+ <StatsCard
61
+ title="Total Likes"
62
+ value={stats.totalLikes}
63
+ icon={Activity}
64
+ />
65
+ <StatsCard
66
+ title="Member Since"
67
+ value={new Date(stats.joinedDate).getFullYear()}
68
+ icon={User}
69
+ />
70
+ </div>
71
+ )
72
+ }
73
+
74
+ function UserStatsSkeleton() {
75
+ return (
76
+ <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6 mb-8">
77
+ {[...Array(4)].map((_, i) => (
78
+ <Card key={i} className="p-6">
79
+ <div className="animate-pulse">
80
+ <div className="h-4 bg-gray-200 dark:bg-gray-700 rounded w-1/2 mb-2"></div>
81
+ <div className="h-8 bg-gray-200 dark:bg-gray-700 rounded w-1/3"></div>
82
+ </div>
83
+ </Card>
84
+ ))}
85
+ </div>
86
+ )
87
+ }
88
+
89
+ export default async function DashboardPage() {
90
+ // Server-side authentication requirement
91
+ const user = await requireAuth()
92
+
93
+ return (
94
+ <div className="min-h-screen bg-gray-50 dark:bg-gray-900">
95
+ <div className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
96
+ {/* Header */}
97
+ <div className="mb-8">
98
+ <h1 className="text-3xl font-bold text-gray-900 dark:text-white">
99
+ Welcome back, {user.name || user.email}!
100
+ </h1>
101
+ <p className="text-gray-600 dark:text-gray-300 mt-2">
102
+ Here's what's happening with your account today.
103
+ </p>
104
+ </div>
105
+
106
+ {/* Stats - Streaming with Suspense */}
107
+ <Suspense fallback={<UserStatsSkeleton />}>
108
+ <UserStats userId={user.id} />
109
+ </Suspense>
110
+
111
+ {/* Quick Actions */}
112
+ <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6 mb-8">
113
+ <Card className="p-6">
114
+ <div className="flex items-center mb-4">
115
+ <User className="h-6 w-6 text-blue-600 dark:text-blue-400 mr-3" />
116
+ <h3 className="text-lg font-semibold text-gray-900 dark:text-white">Profile</h3>
117
+ </div>
118
+ <p className="text-gray-600 dark:text-gray-300 mb-4">
119
+ Update your personal information and preferences
120
+ </p>
121
+ <Link href="/profile">
122
+ <Button className="w-full">Manage Profile</Button>
123
+ </Link>
124
+ </Card>
125
+
126
+ <Card className="p-6">
127
+ <div className="flex items-center mb-4">
128
+ <Settings className="h-6 w-6 text-blue-600 dark:text-blue-400 mr-3" />
129
+ <h3 className="text-lg font-semibold text-gray-900 dark:text-white">Settings</h3>
130
+ </div>
131
+ <p className="text-gray-600 dark:text-gray-300 mb-4">
132
+ Configure your account settings and security options
133
+ </p>
134
+ <Link href="/settings">
135
+ <Button variant="outline" className="w-full">Open Settings</Button>
136
+ </Link>
137
+ </Card>
138
+
139
+ <Card className="p-6">
140
+ <div className="flex items-center mb-4">
141
+ <CreditCard className="h-6 w-6 text-blue-600 dark:text-blue-400 mr-3" />
142
+ <h3 className="text-lg font-semibold text-gray-900 dark:text-white">Billing</h3>
143
+ </div>
144
+ <p className="text-gray-600 dark:text-gray-300 mb-4">
145
+ View your subscription and payment information
146
+ </p>
147
+ <Link href="/billing">
148
+ <Button variant="outline" className="w-full">View Billing</Button>
149
+ </Link>
150
+ </Card>
151
+ </div>
152
+
153
+ {/* Recent Activity */}
154
+ <Card className="p-6">
155
+ <h3 className="text-lg font-semibold text-gray-900 dark:text-white mb-4">Recent Activity</h3>
156
+ <div className="space-y-4">
157
+ <div className="flex items-center justify-between py-3 border-b border-gray-200 dark:border-gray-700">
158
+ <div className="flex items-center">
159
+ <div className="w-2 h-2 bg-green-500 rounded-full mr-3"></div>
160
+ <span className="text-gray-900 dark:text-white">Account created</span>
161
+ </div>
162
+ <span className="text-sm text-gray-500 dark:text-gray-400">2 days ago</span>
163
+ </div>
164
+ <div className="flex items-center justify-between py-3 border-b border-gray-200 dark:border-gray-700">
165
+ <div className="flex items-center">
166
+ <div className="w-2 h-2 bg-blue-500 rounded-full mr-3"></div>
167
+ <span className="text-gray-900 dark:text-white">Profile updated</span>
168
+ </div>
169
+ <span className="text-sm text-gray-500 dark:text-gray-400">1 week ago</span>
170
+ </div>
171
+ <div className="flex items-center justify-between py-3">
172
+ <div className="flex items-center">
173
+ <div className="w-2 h-2 bg-purple-500 rounded-full mr-3"></div>
174
+ <span className="text-gray-900 dark:text-white">Subscription started</span>
175
+ </div>
176
+ <span className="text-sm text-gray-500 dark:text-gray-400">2 weeks ago</span>
177
+ </div>
178
+ </div>
179
+ </Card>
180
+ </div>
181
+ </div>
182
+ )
183
+ }
@@ -0,0 +1,67 @@
1
+ 'use client'
2
+
3
+ import { useEffect } from 'react'
4
+ import { Button, Card } from '@digilogiclabs/saas-factory-ui'
5
+ import { RefreshCw, Home, AlertCircle } from 'lucide-react'
6
+
7
+ interface ErrorProps {
8
+ error: Error & { digest?: string }
9
+ reset: () => void
10
+ }
11
+
12
+ export default function Error({ error, reset }: ErrorProps) {
13
+ useEffect(() => {
14
+ // Log the error to an error reporting service
15
+ console.error('Application error:', error)
16
+ }, [error])
17
+
18
+ return (
19
+ <div className="min-h-screen bg-gray-50 dark:bg-gray-900 flex items-center justify-center p-4">
20
+ <Card className="w-full max-w-md p-8 text-center">
21
+ <div className="mb-6">
22
+ <div className="mx-auto w-16 h-16 bg-red-100 dark:bg-red-900 rounded-full flex items-center justify-center mb-4">
23
+ <AlertCircle className="h-8 w-8 text-red-600 dark:text-red-400" />
24
+ </div>
25
+ <h1 className="text-2xl font-bold text-gray-900 dark:text-white mb-2">
26
+ Something went wrong
27
+ </h1>
28
+ <p className="text-gray-600 dark:text-gray-300 mb-6">
29
+ We apologize for the inconvenience. Please try again or return to the home page.
30
+ </p>
31
+ </div>
32
+
33
+ <div className="space-y-3">
34
+ <Button
35
+ onClick={reset}
36
+ className="w-full"
37
+ variant="default"
38
+ >
39
+ <RefreshCw className="w-4 h-4 mr-2" />
40
+ Try Again
41
+ </Button>
42
+
43
+ <Button
44
+ onClick={() => window.location.href = '/'}
45
+ variant="outline"
46
+ className="w-full"
47
+ >
48
+ <Home className="w-4 h-4 mr-2" />
49
+ Go Home
50
+ </Button>
51
+ </div>
52
+
53
+ {process.env.NODE_ENV === 'development' && (
54
+ <details className="mt-6 text-left">
55
+ <summary className="text-sm font-medium text-gray-500 cursor-pointer hover:text-gray-700 dark:hover:text-gray-300">
56
+ Error Details (Development)
57
+ </summary>
58
+ <pre className="mt-2 text-xs bg-gray-100 dark:bg-gray-800 p-3 rounded-md overflow-auto">
59
+ {error.message}
60
+ {error.stack}
61
+ </pre>
62
+ </details>
63
+ )}
64
+ </Card>
65
+ </div>
66
+ )
67
+ }
@@ -3,6 +3,7 @@ import { Inter } from 'next/font/google'
3
3
  import './globals.css'
4
4
  import { AppProviders } from '@/components/providers/app-providers'
5
5
  import { Header } from '@/components/shared/header'
6
+ import { Footer } from '@/components/shared/footer'
6
7
 
7
8
  const inter = Inter({ subsets: ['latin'] })
8
9
 
@@ -18,10 +19,13 @@ export default function RootLayout({
18
19
  }) {
19
20
  return (
20
21
  <html lang="en" suppressHydrationWarning>
21
- <body className={inter.className}>
22
+ <body className={`${inter.className} min-h-screen flex flex-col`}>
22
23
  <AppProviders>
23
24
  <Header />
24
- {children}
25
+ <main className="flex-1">
26
+ {children}
27
+ </main>
28
+ <Footer />
25
29
  </AppProviders>
26
30
  </body>
27
31
  </html>
@@ -0,0 +1,20 @@
1
+ import { Card } from '@digilogiclabs/saas-factory-ui'
2
+ import { Loader2 } from 'lucide-react'
3
+
4
+ export default function Loading() {
5
+ return (
6
+ <div className="min-h-screen bg-gray-50 dark:bg-gray-900 flex items-center justify-center p-4">
7
+ <Card className="w-full max-w-md p-8 text-center">
8
+ <div className="flex flex-col items-center">
9
+ <Loader2 className="h-8 w-8 animate-spin text-blue-600 dark:text-blue-400 mb-4" />
10
+ <h2 className="text-lg font-medium text-gray-900 dark:text-white mb-2">
11
+ Loading
12
+ </h2>
13
+ <p className="text-sm text-gray-600 dark:text-gray-300">
14
+ Please wait while we load your content...
15
+ </p>
16
+ </div>
17
+ </Card>
18
+ </div>
19
+ )
20
+ }
@@ -0,0 +1,38 @@
1
+ import { Card } from '@digilogiclabs/saas-factory-ui'
2
+
3
+ export default function LoginLoading() {
4
+ return (
5
+ <div className="flex items-center justify-center min-h-screen bg-gray-100 dark:bg-gray-900">
6
+ <Card className="w-full max-w-md p-8">
7
+ <div className="animate-pulse space-y-6">
8
+ {/* Title skeleton */}
9
+ <div className="h-8 bg-gray-200 dark:bg-gray-700 rounded-md mx-auto w-32"></div>
10
+
11
+ {/* Form skeleton */}
12
+ <div className="space-y-4">
13
+ {/* Email field */}
14
+ <div>
15
+ <div className="h-4 bg-gray-200 dark:bg-gray-700 rounded w-16 mb-2"></div>
16
+ <div className="h-10 bg-gray-200 dark:bg-gray-700 rounded"></div>
17
+ </div>
18
+
19
+ {/* Password field */}
20
+ <div>
21
+ <div className="h-4 bg-gray-200 dark:bg-gray-700 rounded w-20 mb-2"></div>
22
+ <div className="h-10 bg-gray-200 dark:bg-gray-700 rounded"></div>
23
+ </div>
24
+
25
+ {/* Sign in button */}
26
+ <div className="h-10 bg-blue-200 dark:bg-blue-800 rounded"></div>
27
+
28
+ {/* OAuth button */}
29
+ <div className="h-10 bg-gray-200 dark:bg-gray-700 rounded"></div>
30
+ </div>
31
+
32
+ {/* Sign up link */}
33
+ <div className="h-4 bg-gray-200 dark:bg-gray-700 rounded w-48 mx-auto"></div>
34
+ </div>
35
+ </Card>
36
+ </div>
37
+ )
38
+ }
@@ -0,0 +1,50 @@
1
+ import { Card } from '@digilogiclabs/saas-factory-ui'
2
+
3
+ export default function SignupLoading() {
4
+ return (
5
+ <div className="flex items-center justify-center min-h-screen bg-gray-100 dark:bg-gray-900">
6
+ <Card className="w-full max-w-md p-8">
7
+ <div className="animate-pulse space-y-6">
8
+ {/* Title skeleton */}
9
+ <div className="h-8 bg-gray-200 dark:bg-gray-700 rounded-md mx-auto w-32"></div>
10
+
11
+ {/* Form skeleton */}
12
+ <div className="space-y-4">
13
+ {/* Name field */}
14
+ <div>
15
+ <div className="h-4 bg-gray-200 dark:bg-gray-700 rounded w-12 mb-2"></div>
16
+ <div className="h-10 bg-gray-200 dark:bg-gray-700 rounded"></div>
17
+ </div>
18
+
19
+ {/* Email field */}
20
+ <div>
21
+ <div className="h-4 bg-gray-200 dark:bg-gray-700 rounded w-16 mb-2"></div>
22
+ <div className="h-10 bg-gray-200 dark:bg-gray-700 rounded"></div>
23
+ </div>
24
+
25
+ {/* Password field */}
26
+ <div>
27
+ <div className="h-4 bg-gray-200 dark:bg-gray-700 rounded w-20 mb-2"></div>
28
+ <div className="h-10 bg-gray-200 dark:bg-gray-700 rounded"></div>
29
+ </div>
30
+
31
+ {/* Confirm password field */}
32
+ <div>
33
+ <div className="h-4 bg-gray-200 dark:bg-gray-700 rounded w-32 mb-2"></div>
34
+ <div className="h-10 bg-gray-200 dark:bg-gray-700 rounded"></div>
35
+ </div>
36
+
37
+ {/* Sign up button */}
38
+ <div className="h-10 bg-blue-200 dark:bg-blue-800 rounded"></div>
39
+
40
+ {/* OAuth button */}
41
+ <div className="h-10 bg-gray-200 dark:bg-gray-700 rounded"></div>
42
+ </div>
43
+
44
+ {/* Sign in link */}
45
+ <div className="h-4 bg-gray-200 dark:bg-gray-700 rounded w-44 mx-auto"></div>
46
+ </div>
47
+ </Card>
48
+ </div>
49
+ )
50
+ }
@@ -0,0 +1,49 @@
1
+ /**
2
+ * Example test file demonstrating testing patterns for SaaS Factory components
3
+ *
4
+ * This is optional and serves as a reference for testing your components.
5
+ * You can delete this file if you don't need example tests.
6
+ */
7
+
8
+ import { render, screen } from '@testing-library/react'
9
+ import { Button } from '@digilogiclabs/saas-factory-ui'
10
+
11
+ // Example: Testing UI components
12
+ describe('UI Components', () => {
13
+ it('renders button with correct text', () => {
14
+ render(<Button>Click me</Button>)
15
+ expect(screen.getByRole('button', { name: /click me/i })).toBeInTheDocument()
16
+ })
17
+
18
+ it('button can be disabled', () => {
19
+ render(<Button disabled>Disabled button</Button>)
20
+ expect(screen.getByRole('button')).toBeDisabled()
21
+ })
22
+ })
23
+
24
+ // Example: Testing custom components
25
+ function MockComponent({ title }: { title: string }) {
26
+ return <h1>{title}</h1>
27
+ }
28
+
29
+ describe('Custom Components', () => {
30
+ it('renders with correct title', () => {
31
+ render(<MockComponent title="Test Title" />)
32
+ expect(screen.getByRole('heading', { level: 1 })).toHaveTextContent('Test Title')
33
+ })
34
+ })
35
+
36
+ // Example: Testing utilities
37
+ function formatCurrency(amount: number): string {
38
+ return new Intl.NumberFormat('en-US', {
39
+ style: 'currency',
40
+ currency: 'USD',
41
+ }).format(amount)
42
+ }
43
+
44
+ describe('Utilities', () => {
45
+ it('formats currency correctly', () => {
46
+ expect(formatCurrency(1000)).toBe('$1,000.00')
47
+ expect(formatCurrency(0)).toBe('$0.00')
48
+ })
49
+ })
@@ -0,0 +1,52 @@
1
+ 'use client'
2
+
3
+ import { Button } from '@digilogiclabs/saas-factory-ui'
4
+ import { LogOut, User } from 'lucide-react'
5
+ import { useAuth } from '@digilogiclabs/saas-factory-auth'
6
+ import Link from 'next/link'
7
+
8
+ export function AuthStatus() {
9
+ const { user, signOut, loading } = useAuth()
10
+
11
+ const handleSignOut = async () => {
12
+ try {
13
+ await signOut()
14
+ } catch (err) {
15
+ console.error('Sign out error:', err)
16
+ }
17
+ }
18
+
19
+ if (loading) {
20
+ return (
21
+ <div className="text-sm text-gray-600 dark:text-gray-300">
22
+ Loading...
23
+ </div>
24
+ )
25
+ }
26
+
27
+ if (user) {
28
+ return (
29
+ <div className="flex items-center gap-4">
30
+ <div className="flex items-center gap-2 text-sm text-gray-600 dark:text-gray-300">
31
+ <User className="w-4 h-4" />
32
+ Welcome, {user.email}
33
+ </div>
34
+ <Button variant="outline" size="sm" onClick={handleSignOut}>
35
+ <LogOut className="w-4 h-4 mr-2" />
36
+ Sign Out
37
+ </Button>
38
+ </div>
39
+ )
40
+ }
41
+
42
+ return (
43
+ <div className="flex gap-2">
44
+ <Link href="/login">
45
+ <Button variant="outline" size="sm">Sign In</Button>
46
+ </Link>
47
+ <Link href="/signup">
48
+ <Button size="sm">Sign Up</Button>
49
+ </Link>
50
+ </div>
51
+ )
52
+ }