@nextsparkjs/core 0.1.0-beta.83 → 0.1.0-beta.85

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 (365) hide show
  1. package/dist/styles/classes.json +1 -1
  2. package/dist/templates/app/(auth)/forgot-password/page.tsx +216 -0
  3. package/dist/templates/app/(auth)/layout.tsx +51 -0
  4. package/dist/templates/app/(auth)/login/page.tsx +21 -0
  5. package/dist/templates/app/(auth)/reset-password/page.tsx +212 -0
  6. package/dist/templates/app/(auth)/signup/page.tsx +21 -0
  7. package/dist/templates/app/(auth)/verify-email/page.tsx +190 -0
  8. package/dist/templates/app/(public)/[...slug]/page.tsx +378 -0
  9. package/dist/templates/app/(public)/docs/[section]/[page]/page.tsx +90 -0
  10. package/dist/templates/app/(public)/docs/layout.tsx +25 -0
  11. package/dist/templates/app/(public)/docs/page.tsx +81 -0
  12. package/dist/templates/app/(public)/layout.tsx +41 -0
  13. package/dist/templates/app/(public)/page.tsx +19 -0
  14. package/dist/templates/app/403/page.tsx +89 -0
  15. package/dist/templates/app/api/auth/[...all]/route.ts +78 -0
  16. package/dist/templates/app/api/cron/billing/lifecycle/route.ts +98 -0
  17. package/dist/templates/app/api/csp-report/route.ts +175 -0
  18. package/dist/templates/app/api/devtools/config/entities/route.ts +108 -0
  19. package/dist/templates/app/api/devtools/config/theme/route.ts +66 -0
  20. package/dist/templates/app/api/devtools/tests/[...path]/route.ts +130 -0
  21. package/dist/templates/app/api/devtools/tests/route.ts +134 -0
  22. package/dist/templates/app/api/health/route.ts +29 -0
  23. package/dist/templates/app/api/internal/user-metadata/route.ts +36 -0
  24. package/dist/templates/app/api/superadmin/subscriptions/route.ts +310 -0
  25. package/dist/templates/app/api/superadmin/teams/[teamId]/route.ts +286 -0
  26. package/dist/templates/app/api/superadmin/teams/route.ts +188 -0
  27. package/dist/templates/app/api/superadmin/users/[userId]/route.ts +540 -0
  28. package/dist/templates/app/api/superadmin/users/route.ts +323 -0
  29. package/dist/templates/app/api/user/delete-account/route.ts +55 -0
  30. package/dist/templates/app/api/user/plan-flags/route.ts +283 -0
  31. package/dist/templates/app/api/user/profile/route.ts +133 -0
  32. package/dist/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +210 -0
  33. package/dist/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +331 -0
  34. package/dist/templates/app/api/v1/[entity]/[id]/route.ts +35 -0
  35. package/dist/templates/app/api/v1/[entity]/docs.md +369 -0
  36. package/dist/templates/app/api/v1/[entity]/presets.ts +194 -0
  37. package/dist/templates/app/api/v1/[entity]/route.ts +31 -0
  38. package/dist/templates/app/api/v1/api-keys/[id]/route.ts +303 -0
  39. package/dist/templates/app/api/v1/api-keys/docs.md +101 -0
  40. package/dist/templates/app/api/v1/api-keys/presets.ts +31 -0
  41. package/dist/templates/app/api/v1/api-keys/route.ts +250 -0
  42. package/dist/templates/app/api/v1/auth/docs.md +184 -0
  43. package/dist/templates/app/api/v1/auth/presets.ts +44 -0
  44. package/dist/templates/app/api/v1/auth/signup-with-invite/route.ts +227 -0
  45. package/dist/templates/app/api/v1/billing/cancel/route.ts +206 -0
  46. package/dist/templates/app/api/v1/billing/change-plan/route.ts +97 -0
  47. package/dist/templates/app/api/v1/billing/check-action/route.ts +81 -0
  48. package/dist/templates/app/api/v1/billing/checkout/route.ts +124 -0
  49. package/dist/templates/app/api/v1/billing/docs.md +209 -0
  50. package/dist/templates/app/api/v1/billing/plans/route.ts +85 -0
  51. package/dist/templates/app/api/v1/billing/portal/route.ts +90 -0
  52. package/dist/templates/app/api/v1/billing/presets.ts +121 -0
  53. package/dist/templates/app/api/v1/billing/webhooks/stripe/route.ts +428 -0
  54. package/dist/templates/app/api/v1/blocks/[slug]/route.ts +29 -0
  55. package/dist/templates/app/api/v1/blocks/docs.md +173 -0
  56. package/dist/templates/app/api/v1/blocks/presets.ts +121 -0
  57. package/dist/templates/app/api/v1/blocks/route.ts +45 -0
  58. package/dist/templates/app/api/v1/blocks/validate/route.ts +45 -0
  59. package/dist/templates/app/api/v1/cron/docs.md +116 -0
  60. package/dist/templates/app/api/v1/cron/presets.ts +26 -0
  61. package/dist/templates/app/api/v1/cron/process/route.ts +108 -0
  62. package/dist/templates/app/api/v1/devtools/blocks/route.ts +82 -0
  63. package/dist/templates/app/api/v1/devtools/docs/route.ts +150 -0
  64. package/dist/templates/app/api/v1/devtools/docs.md +204 -0
  65. package/dist/templates/app/api/v1/devtools/features/route.ts +61 -0
  66. package/dist/templates/app/api/v1/devtools/flows/route.ts +61 -0
  67. package/dist/templates/app/api/v1/devtools/presets.ts +113 -0
  68. package/dist/templates/app/api/v1/devtools/scheduled-actions/route.ts +120 -0
  69. package/dist/templates/app/api/v1/devtools/testing/route.ts +82 -0
  70. package/dist/templates/app/api/v1/media/docs.md +117 -0
  71. package/dist/templates/app/api/v1/media/presets.ts +24 -0
  72. package/dist/templates/app/api/v1/media/upload/route.ts +150 -0
  73. package/dist/templates/app/api/v1/patterns/[id]/usages/route.ts +116 -0
  74. package/dist/templates/app/api/v1/plugin/[...path]/route.ts +373 -0
  75. package/dist/templates/app/api/v1/plugin/docs.md +79 -0
  76. package/dist/templates/app/api/v1/plugin/presets.ts +21 -0
  77. package/dist/templates/app/api/v1/plugin/route.ts +96 -0
  78. package/dist/templates/app/api/v1/post-categories/[id]/route.ts +255 -0
  79. package/dist/templates/app/api/v1/post-categories/docs.md +134 -0
  80. package/dist/templates/app/api/v1/post-categories/presets.ts +78 -0
  81. package/dist/templates/app/api/v1/post-categories/route.ts +119 -0
  82. package/dist/templates/app/api/v1/team-invitations/[token]/accept/route.ts +179 -0
  83. package/dist/templates/app/api/v1/team-invitations/[token]/decline/route.ts +120 -0
  84. package/dist/templates/app/api/v1/team-invitations/[token]/route.ts +89 -0
  85. package/dist/templates/app/api/v1/team-invitations/docs.md +88 -0
  86. package/dist/templates/app/api/v1/team-invitations/presets.ts +43 -0
  87. package/dist/templates/app/api/v1/team-invitations/route.ts +114 -0
  88. package/dist/templates/app/api/v1/teams/[teamId]/invitations/route.ts +171 -0
  89. package/dist/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +105 -0
  90. package/dist/templates/app/api/v1/teams/[teamId]/invoices/route.ts +125 -0
  91. package/dist/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +263 -0
  92. package/dist/templates/app/api/v1/teams/[teamId]/members/route.ts +358 -0
  93. package/dist/templates/app/api/v1/teams/[teamId]/route.ts +322 -0
  94. package/dist/templates/app/api/v1/teams/[teamId]/subscription/route.ts +50 -0
  95. package/dist/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +91 -0
  96. package/dist/templates/app/api/v1/teams/docs.md +320 -0
  97. package/dist/templates/app/api/v1/teams/presets.ts +178 -0
  98. package/dist/templates/app/api/v1/teams/route.ts +293 -0
  99. package/dist/templates/app/api/v1/teams/switch/route.ts +88 -0
  100. package/dist/templates/app/api/v1/theme/[...path]/route.ts +361 -0
  101. package/dist/templates/app/api/v1/theme/docs.md +74 -0
  102. package/dist/templates/app/api/v1/theme/presets.ts +21 -0
  103. package/dist/templates/app/api/v1/theme/route.ts +96 -0
  104. package/dist/templates/app/api/v1/users/[id]/meta/[key]/route.ts +363 -0
  105. package/dist/templates/app/api/v1/users/[id]/route.ts +302 -0
  106. package/dist/templates/app/api/v1/users/docs.md +93 -0
  107. package/dist/templates/app/api/v1/users/presets.ts +59 -0
  108. package/dist/templates/app/api/v1/users/route.ts +197 -0
  109. package/dist/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +117 -0
  110. package/dist/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +103 -0
  111. package/dist/templates/app/dashboard/(main)/[entity]/create/page.tsx +95 -0
  112. package/dist/templates/app/dashboard/(main)/[entity]/error.tsx +51 -0
  113. package/dist/templates/app/dashboard/(main)/[entity]/layout.tsx +113 -0
  114. package/dist/templates/app/dashboard/(main)/[entity]/loading.tsx +61 -0
  115. package/dist/templates/app/dashboard/(main)/[entity]/page.tsx +90 -0
  116. package/dist/templates/app/dashboard/(main)/layout.tsx +98 -0
  117. package/dist/templates/app/dashboard/(main)/loading.tsx +5 -0
  118. package/dist/templates/app/dashboard/(main)/page.tsx +201 -0
  119. package/dist/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +114 -0
  120. package/dist/templates/app/dashboard/(main)/patterns/[id]/page.tsx +20 -0
  121. package/dist/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +171 -0
  122. package/dist/templates/app/dashboard/(main)/patterns/create/page.tsx +86 -0
  123. package/dist/templates/app/dashboard/(main)/patterns/page.tsx +444 -0
  124. package/dist/templates/app/dashboard/features/analytics/page.tsx +35 -0
  125. package/dist/templates/app/dashboard/features/automation/page.tsx +35 -0
  126. package/dist/templates/app/dashboard/features/layout.tsx +13 -0
  127. package/dist/templates/app/dashboard/features/loading.tsx +5 -0
  128. package/dist/templates/app/dashboard/features/webhooks/page.tsx +35 -0
  129. package/dist/templates/app/dashboard/layout.tsx +86 -0
  130. package/dist/templates/app/dashboard/permission-denied/page.tsx +29 -0
  131. package/dist/templates/app/dashboard/settings/api-keys/loading.tsx +5 -0
  132. package/dist/templates/app/dashboard/settings/api-keys/page.tsx +513 -0
  133. package/dist/templates/app/dashboard/settings/billing/loading.tsx +5 -0
  134. package/dist/templates/app/dashboard/settings/billing/page.tsx +284 -0
  135. package/dist/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +222 -0
  136. package/dist/templates/app/dashboard/settings/invoices/loading.tsx +5 -0
  137. package/dist/templates/app/dashboard/settings/invoices/page.tsx +82 -0
  138. package/dist/templates/app/dashboard/settings/layout.tsx +151 -0
  139. package/dist/templates/app/dashboard/settings/loading.tsx +5 -0
  140. package/dist/templates/app/dashboard/settings/notifications/loading.tsx +5 -0
  141. package/dist/templates/app/dashboard/settings/notifications/page.tsx +462 -0
  142. package/dist/templates/app/dashboard/settings/page.tsx +92 -0
  143. package/dist/templates/app/dashboard/settings/password/loading.tsx +5 -0
  144. package/dist/templates/app/dashboard/settings/password/page.tsx +306 -0
  145. package/dist/templates/app/dashboard/settings/plans/loading.tsx +5 -0
  146. package/dist/templates/app/dashboard/settings/plans/page.tsx +40 -0
  147. package/dist/templates/app/dashboard/settings/profile/loading.tsx +5 -0
  148. package/dist/templates/app/dashboard/settings/profile/page.tsx +686 -0
  149. package/dist/templates/app/dashboard/settings/security/loading.tsx +5 -0
  150. package/dist/templates/app/dashboard/settings/security/page.tsx +505 -0
  151. package/dist/templates/app/dashboard/settings/teams/loading.tsx +5 -0
  152. package/dist/templates/app/dashboard/settings/teams/page.tsx +272 -0
  153. package/dist/templates/app/dashboard/settings/teams/permissions/page.tsx +92 -0
  154. package/dist/templates/app/devtools/blocks/[slug]/page.tsx +39 -0
  155. package/dist/templates/app/devtools/blocks/page.tsx +31 -0
  156. package/dist/templates/app/devtools/config/page.tsx +31 -0
  157. package/dist/templates/app/devtools/features/page.tsx +31 -0
  158. package/dist/templates/app/devtools/flows/page.tsx +31 -0
  159. package/dist/templates/app/devtools/layout.tsx +58 -0
  160. package/dist/templates/app/devtools/page.tsx +121 -0
  161. package/dist/templates/app/devtools/scheduled-actions/page.tsx +157 -0
  162. package/dist/templates/app/devtools/style/page.tsx +330 -0
  163. package/dist/templates/app/devtools/tags/page.tsx +31 -0
  164. package/dist/templates/app/devtools/tests/[[...path]]/page.tsx +47 -0
  165. package/dist/templates/app/favicon.ico +0 -0
  166. package/dist/templates/app/globals.css +12 -0
  167. package/dist/templates/app/layout.tsx +96 -0
  168. package/dist/templates/app/public/page.tsx +30 -0
  169. package/dist/templates/app/superadmin/docs/[section]/[page]/page.tsx +92 -0
  170. package/dist/templates/app/superadmin/docs/page.tsx +75 -0
  171. package/dist/templates/app/superadmin/layout.tsx +67 -0
  172. package/dist/templates/app/superadmin/page.tsx +149 -0
  173. package/dist/templates/app/superadmin/subscriptions/page.tsx +655 -0
  174. package/dist/templates/app/superadmin/team-roles/page.tsx +493 -0
  175. package/dist/templates/app/superadmin/teams/[teamId]/page.tsx +687 -0
  176. package/dist/templates/app/superadmin/teams/page.tsx +302 -0
  177. package/dist/templates/app/superadmin/users/[userId]/page.tsx +548 -0
  178. package/dist/templates/app/superadmin/users/page.tsx +528 -0
  179. package/package.json +15 -15
  180. package/scripts/build/docs-registry.mjs +0 -0
  181. package/scripts/create-theme.mjs +0 -0
  182. package/scripts/deploy/release-version.mjs +0 -0
  183. package/scripts/deploy/vercel-deploy.mjs +0 -0
  184. package/scripts/dev/watch-plugins.mjs +0 -0
  185. package/scripts/maintenance/update-core.mjs +0 -0
  186. package/scripts/setup/npm-postinstall.mjs +0 -0
  187. package/scripts/setup/setup-claude.mjs +0 -0
  188. package/scripts/validation/check-imports.sh +0 -0
  189. package/templates/app/(auth)/forgot-password/page.tsx +216 -0
  190. package/templates/app/(auth)/layout.tsx +51 -0
  191. package/templates/app/(auth)/login/page.tsx +21 -0
  192. package/templates/app/(auth)/reset-password/page.tsx +212 -0
  193. package/templates/app/(auth)/signup/page.tsx +21 -0
  194. package/templates/app/(auth)/verify-email/page.tsx +190 -0
  195. package/templates/app/(public)/[...slug]/page.tsx +378 -0
  196. package/templates/app/(public)/docs/[section]/[page]/page.tsx +90 -0
  197. package/templates/app/(public)/docs/layout.tsx +25 -0
  198. package/templates/app/(public)/docs/page.tsx +81 -0
  199. package/templates/app/(public)/layout.tsx +41 -0
  200. package/templates/app/(public)/page.tsx +19 -0
  201. package/templates/app/403/page.tsx +89 -0
  202. package/templates/app/api/auth/[...all]/route.ts +78 -0
  203. package/templates/app/api/cron/billing/lifecycle/route.ts +98 -0
  204. package/templates/app/api/csp-report/route.ts +175 -0
  205. package/templates/app/api/devtools/config/entities/route.ts +108 -0
  206. package/templates/app/api/devtools/config/theme/route.ts +66 -0
  207. package/templates/app/api/devtools/tests/[...path]/route.ts +130 -0
  208. package/templates/app/api/devtools/tests/route.ts +134 -0
  209. package/templates/app/api/health/route.ts +29 -0
  210. package/templates/app/api/internal/user-metadata/route.ts +36 -0
  211. package/templates/app/api/superadmin/subscriptions/route.ts +310 -0
  212. package/templates/app/api/superadmin/teams/[teamId]/route.ts +286 -0
  213. package/templates/app/api/superadmin/teams/route.ts +188 -0
  214. package/templates/app/api/superadmin/users/[userId]/route.ts +540 -0
  215. package/templates/app/api/superadmin/users/route.ts +323 -0
  216. package/templates/app/api/user/delete-account/route.ts +55 -0
  217. package/templates/app/api/user/plan-flags/route.ts +283 -0
  218. package/templates/app/api/user/profile/route.ts +133 -0
  219. package/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +210 -0
  220. package/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +331 -0
  221. package/templates/app/api/v1/[entity]/[id]/route.ts +35 -0
  222. package/templates/app/api/v1/[entity]/docs.md +369 -0
  223. package/templates/app/api/v1/[entity]/presets.ts +194 -0
  224. package/templates/app/api/v1/[entity]/route.ts +31 -0
  225. package/templates/app/api/v1/api-keys/[id]/route.ts +303 -0
  226. package/templates/app/api/v1/api-keys/docs.md +101 -0
  227. package/templates/app/api/v1/api-keys/presets.ts +31 -0
  228. package/templates/app/api/v1/api-keys/route.ts +250 -0
  229. package/templates/app/api/v1/auth/docs.md +184 -0
  230. package/templates/app/api/v1/auth/presets.ts +44 -0
  231. package/templates/app/api/v1/auth/signup-with-invite/route.ts +227 -0
  232. package/templates/app/api/v1/billing/cancel/route.ts +206 -0
  233. package/templates/app/api/v1/billing/change-plan/route.ts +97 -0
  234. package/templates/app/api/v1/billing/check-action/route.ts +81 -0
  235. package/templates/app/api/v1/billing/checkout/route.ts +124 -0
  236. package/templates/app/api/v1/billing/docs.md +209 -0
  237. package/templates/app/api/v1/billing/plans/route.ts +85 -0
  238. package/templates/app/api/v1/billing/portal/route.ts +90 -0
  239. package/templates/app/api/v1/billing/presets.ts +121 -0
  240. package/templates/app/api/v1/billing/webhooks/stripe/route.ts +428 -0
  241. package/templates/app/api/v1/blocks/[slug]/route.ts +29 -0
  242. package/templates/app/api/v1/blocks/docs.md +173 -0
  243. package/templates/app/api/v1/blocks/presets.ts +121 -0
  244. package/templates/app/api/v1/blocks/route.ts +45 -0
  245. package/templates/app/api/v1/blocks/validate/route.ts +45 -0
  246. package/templates/app/api/v1/cron/docs.md +116 -0
  247. package/templates/app/api/v1/cron/presets.ts +26 -0
  248. package/templates/app/api/v1/cron/process/route.ts +108 -0
  249. package/templates/app/api/v1/devtools/blocks/route.ts +82 -0
  250. package/templates/app/api/v1/devtools/docs/route.ts +150 -0
  251. package/templates/app/api/v1/devtools/docs.md +204 -0
  252. package/templates/app/api/v1/devtools/features/route.ts +61 -0
  253. package/templates/app/api/v1/devtools/flows/route.ts +61 -0
  254. package/templates/app/api/v1/devtools/presets.ts +113 -0
  255. package/templates/app/api/v1/devtools/scheduled-actions/route.ts +120 -0
  256. package/templates/app/api/v1/devtools/testing/route.ts +82 -0
  257. package/templates/app/api/v1/media/docs.md +117 -0
  258. package/templates/app/api/v1/media/presets.ts +24 -0
  259. package/templates/app/api/v1/media/upload/route.ts +150 -0
  260. package/templates/app/api/v1/patterns/[id]/usages/route.ts +116 -0
  261. package/templates/app/api/v1/plugin/[...path]/route.ts +373 -0
  262. package/templates/app/api/v1/plugin/docs.md +79 -0
  263. package/templates/app/api/v1/plugin/presets.ts +21 -0
  264. package/templates/app/api/v1/plugin/route.ts +96 -0
  265. package/templates/app/api/v1/post-categories/[id]/route.ts +255 -0
  266. package/templates/app/api/v1/post-categories/docs.md +134 -0
  267. package/templates/app/api/v1/post-categories/presets.ts +78 -0
  268. package/templates/app/api/v1/post-categories/route.ts +119 -0
  269. package/templates/app/api/v1/team-invitations/[token]/accept/route.ts +179 -0
  270. package/templates/app/api/v1/team-invitations/[token]/decline/route.ts +120 -0
  271. package/templates/app/api/v1/team-invitations/[token]/route.ts +89 -0
  272. package/templates/app/api/v1/team-invitations/docs.md +88 -0
  273. package/templates/app/api/v1/team-invitations/presets.ts +43 -0
  274. package/templates/app/api/v1/team-invitations/route.ts +114 -0
  275. package/templates/app/api/v1/teams/[teamId]/invitations/route.ts +171 -0
  276. package/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +105 -0
  277. package/templates/app/api/v1/teams/[teamId]/invoices/route.ts +125 -0
  278. package/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +263 -0
  279. package/templates/app/api/v1/teams/[teamId]/members/route.ts +358 -0
  280. package/templates/app/api/v1/teams/[teamId]/route.ts +322 -0
  281. package/templates/app/api/v1/teams/[teamId]/subscription/route.ts +50 -0
  282. package/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +91 -0
  283. package/templates/app/api/v1/teams/docs.md +320 -0
  284. package/templates/app/api/v1/teams/presets.ts +178 -0
  285. package/templates/app/api/v1/teams/route.ts +293 -0
  286. package/templates/app/api/v1/teams/switch/route.ts +88 -0
  287. package/templates/app/api/v1/theme/[...path]/route.ts +361 -0
  288. package/templates/app/api/v1/theme/docs.md +74 -0
  289. package/templates/app/api/v1/theme/presets.ts +21 -0
  290. package/templates/app/api/v1/theme/route.ts +96 -0
  291. package/templates/app/api/v1/users/[id]/meta/[key]/route.ts +363 -0
  292. package/templates/app/api/v1/users/[id]/route.ts +302 -0
  293. package/templates/app/api/v1/users/docs.md +93 -0
  294. package/templates/app/api/v1/users/presets.ts +59 -0
  295. package/templates/app/api/v1/users/route.ts +197 -0
  296. package/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +117 -0
  297. package/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +103 -0
  298. package/templates/app/dashboard/(main)/[entity]/create/page.tsx +95 -0
  299. package/templates/app/dashboard/(main)/[entity]/error.tsx +51 -0
  300. package/templates/app/dashboard/(main)/[entity]/layout.tsx +113 -0
  301. package/templates/app/dashboard/(main)/[entity]/loading.tsx +61 -0
  302. package/templates/app/dashboard/(main)/[entity]/page.tsx +90 -0
  303. package/templates/app/dashboard/(main)/layout.tsx +98 -0
  304. package/templates/app/dashboard/(main)/loading.tsx +5 -0
  305. package/templates/app/dashboard/(main)/page.tsx +201 -0
  306. package/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +114 -0
  307. package/templates/app/dashboard/(main)/patterns/[id]/page.tsx +20 -0
  308. package/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +171 -0
  309. package/templates/app/dashboard/(main)/patterns/create/page.tsx +86 -0
  310. package/templates/app/dashboard/(main)/patterns/page.tsx +444 -0
  311. package/templates/app/dashboard/features/analytics/page.tsx +35 -0
  312. package/templates/app/dashboard/features/automation/page.tsx +35 -0
  313. package/templates/app/dashboard/features/layout.tsx +13 -0
  314. package/templates/app/dashboard/features/loading.tsx +5 -0
  315. package/templates/app/dashboard/features/webhooks/page.tsx +35 -0
  316. package/templates/app/dashboard/layout.tsx +86 -0
  317. package/templates/app/dashboard/permission-denied/page.tsx +29 -0
  318. package/templates/app/dashboard/settings/api-keys/loading.tsx +5 -0
  319. package/templates/app/dashboard/settings/api-keys/page.tsx +513 -0
  320. package/templates/app/dashboard/settings/billing/loading.tsx +5 -0
  321. package/templates/app/dashboard/settings/billing/page.tsx +284 -0
  322. package/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +222 -0
  323. package/templates/app/dashboard/settings/invoices/loading.tsx +5 -0
  324. package/templates/app/dashboard/settings/invoices/page.tsx +82 -0
  325. package/templates/app/dashboard/settings/layout.tsx +151 -0
  326. package/templates/app/dashboard/settings/loading.tsx +5 -0
  327. package/templates/app/dashboard/settings/notifications/loading.tsx +5 -0
  328. package/templates/app/dashboard/settings/notifications/page.tsx +462 -0
  329. package/templates/app/dashboard/settings/page.tsx +92 -0
  330. package/templates/app/dashboard/settings/password/loading.tsx +5 -0
  331. package/templates/app/dashboard/settings/password/page.tsx +306 -0
  332. package/templates/app/dashboard/settings/plans/loading.tsx +5 -0
  333. package/templates/app/dashboard/settings/plans/page.tsx +40 -0
  334. package/templates/app/dashboard/settings/profile/loading.tsx +5 -0
  335. package/templates/app/dashboard/settings/profile/page.tsx +686 -0
  336. package/templates/app/dashboard/settings/security/loading.tsx +5 -0
  337. package/templates/app/dashboard/settings/security/page.tsx +505 -0
  338. package/templates/app/dashboard/settings/teams/loading.tsx +5 -0
  339. package/templates/app/dashboard/settings/teams/page.tsx +272 -0
  340. package/templates/app/dashboard/settings/teams/permissions/page.tsx +92 -0
  341. package/templates/app/devtools/blocks/[slug]/page.tsx +39 -0
  342. package/templates/app/devtools/blocks/page.tsx +31 -0
  343. package/templates/app/devtools/config/page.tsx +31 -0
  344. package/templates/app/devtools/features/page.tsx +31 -0
  345. package/templates/app/devtools/flows/page.tsx +31 -0
  346. package/templates/app/devtools/layout.tsx +58 -0
  347. package/templates/app/devtools/page.tsx +121 -0
  348. package/templates/app/devtools/scheduled-actions/page.tsx +157 -0
  349. package/templates/app/devtools/style/page.tsx +330 -0
  350. package/templates/app/devtools/tags/page.tsx +31 -0
  351. package/templates/app/devtools/tests/[[...path]]/page.tsx +47 -0
  352. package/templates/app/favicon.ico +0 -0
  353. package/templates/app/globals.css +12 -0
  354. package/templates/app/layout.tsx +96 -0
  355. package/templates/app/public/page.tsx +30 -0
  356. package/templates/app/superadmin/docs/[section]/[page]/page.tsx +92 -0
  357. package/templates/app/superadmin/docs/page.tsx +75 -0
  358. package/templates/app/superadmin/layout.tsx +67 -0
  359. package/templates/app/superadmin/page.tsx +149 -0
  360. package/templates/app/superadmin/subscriptions/page.tsx +655 -0
  361. package/templates/app/superadmin/team-roles/page.tsx +493 -0
  362. package/templates/app/superadmin/teams/[teamId]/page.tsx +687 -0
  363. package/templates/app/superadmin/teams/page.tsx +302 -0
  364. package/templates/app/superadmin/users/[userId]/page.tsx +548 -0
  365. package/templates/app/superadmin/users/page.tsx +528 -0
@@ -0,0 +1,655 @@
1
+ "use client";
2
+
3
+ import { useQuery, keepPreviousData } from "@tanstack/react-query";
4
+ import { useState } from "react";
5
+ import { format } from "date-fns";
6
+ import { sel } from "@nextsparkjs/core/selectors";
7
+ import {
8
+ Card,
9
+ CardContent,
10
+ CardDescription,
11
+ CardHeader,
12
+ CardTitle,
13
+ } from "@nextsparkjs/core/components/ui/card";
14
+ import { Button } from "@nextsparkjs/core/components/ui/button";
15
+ import { Badge } from "@nextsparkjs/core/components/ui/badge";
16
+ import {
17
+ Table,
18
+ TableBody,
19
+ TableCell,
20
+ TableHead,
21
+ TableHeader,
22
+ TableRow,
23
+ } from "@nextsparkjs/core/components/ui/table";
24
+ import {
25
+ Select,
26
+ SelectContent,
27
+ SelectItem,
28
+ SelectTrigger,
29
+ SelectValue,
30
+ } from "@nextsparkjs/core/components/ui/select";
31
+ import {
32
+ ArrowLeft,
33
+ CreditCard,
34
+ DollarSign,
35
+ TrendingUp,
36
+ AlertCircle,
37
+ CheckCircle,
38
+ Clock,
39
+ XCircle,
40
+ Loader2,
41
+ AlertTriangle,
42
+ RefreshCw,
43
+ ExternalLink,
44
+ } from "lucide-react";
45
+ import Link from "next/link";
46
+ import { getTemplateOrDefaultClient } from "@nextsparkjs/registries/template-registry.client";
47
+ import {
48
+ SearchInput,
49
+ PaginationControls,
50
+ } from "@nextsparkjs/core/components/superadmin/filters";
51
+
52
+ interface SubscriptionPlan {
53
+ id: string;
54
+ slug: string;
55
+ name: string;
56
+ type: string;
57
+ priceMonthly: number | null;
58
+ priceYearly: number | null;
59
+ }
60
+
61
+ interface SubscriptionTeam {
62
+ id: string;
63
+ name: string;
64
+ owner: {
65
+ name: string;
66
+ email: string;
67
+ };
68
+ }
69
+
70
+ interface Subscription {
71
+ id: string;
72
+ team: SubscriptionTeam;
73
+ plan: SubscriptionPlan;
74
+ billingInterval: 'monthly' | 'yearly';
75
+ status: string;
76
+ currentPeriodStart: string;
77
+ currentPeriodEnd: string;
78
+ trialEndsAt: string | null;
79
+ canceledAt: string | null;
80
+ cancelAtPeriodEnd: boolean;
81
+ externalSubscriptionId: string | null;
82
+ createdAt: string;
83
+ }
84
+
85
+ interface PlanDistribution {
86
+ slug: string;
87
+ name: string;
88
+ count: number;
89
+ }
90
+
91
+ interface SubscriptionsData {
92
+ stats: {
93
+ total: number;
94
+ active: number;
95
+ trialing: number;
96
+ canceled: number;
97
+ pastDue: number;
98
+ free: number;
99
+ paid: number;
100
+ monthly: number;
101
+ yearly: number;
102
+ mrr: number;
103
+ mrrFormatted: string;
104
+ arr: number;
105
+ arrFormatted: string;
106
+ };
107
+ planDistribution: PlanDistribution[];
108
+ subscriptions: Subscription[];
109
+ pagination: {
110
+ page: number;
111
+ limit: number;
112
+ total: number;
113
+ totalPages: number;
114
+ hasMore: boolean;
115
+ };
116
+ metadata: {
117
+ requestedBy: string;
118
+ requestedAt: string;
119
+ source: string;
120
+ };
121
+ }
122
+
123
+ const statusConfig = {
124
+ active: {
125
+ label: "Active",
126
+ icon: CheckCircle,
127
+ variant: "default" as const,
128
+ className: "bg-green-100 text-green-800 border-green-200",
129
+ },
130
+ trialing: {
131
+ label: "Trial",
132
+ icon: Clock,
133
+ variant: "secondary" as const,
134
+ className: "bg-blue-100 text-blue-800 border-blue-200",
135
+ },
136
+ past_due: {
137
+ label: "Past Due",
138
+ icon: AlertCircle,
139
+ variant: "destructive" as const,
140
+ className: "bg-yellow-100 text-yellow-800 border-yellow-200",
141
+ },
142
+ canceled: {
143
+ label: "Canceled",
144
+ icon: XCircle,
145
+ variant: "outline" as const,
146
+ className: "bg-gray-100 text-gray-800 border-gray-200",
147
+ },
148
+ expired: {
149
+ label: "Expired",
150
+ icon: XCircle,
151
+ variant: "outline" as const,
152
+ className: "bg-red-100 text-red-800 border-red-200",
153
+ },
154
+ paused: {
155
+ label: "Paused",
156
+ icon: Clock,
157
+ variant: "secondary" as const,
158
+ className: "bg-orange-100 text-orange-800 border-orange-200",
159
+ },
160
+ };
161
+
162
+ /**
163
+ * Subscriptions Management Page
164
+ *
165
+ * Comprehensive subscription overview for superadmins.
166
+ * Shows stats, MRR, plan distribution, and subscription list.
167
+ */
168
+ function SubscriptionsPage() {
169
+ const [statusFilter, setStatusFilter] = useState<string>("all");
170
+ const [planFilter, setPlanFilter] = useState<string>("all");
171
+ const [intervalFilter, setIntervalFilter] = useState<string>("all");
172
+ const [searchQuery, setSearchQuery] = useState("");
173
+ const [page, setPage] = useState(1);
174
+ const [limit, setLimit] = useState(20);
175
+
176
+ // Build query params
177
+ const getQueryParams = () => {
178
+ const params = new URLSearchParams();
179
+ if (searchQuery) params.set("search", searchQuery);
180
+ if (statusFilter !== "all") params.set("status", statusFilter);
181
+ if (planFilter !== "all") params.set("plan", planFilter);
182
+ if (intervalFilter !== "all") params.set("interval", intervalFilter);
183
+ params.set("page", String(page));
184
+ params.set("limit", String(limit));
185
+ return params.toString();
186
+ };
187
+
188
+ // Handle filter changes
189
+ const handleStatusChange = (status: string) => {
190
+ setStatusFilter(status);
191
+ setPage(1);
192
+ };
193
+
194
+ const handlePlanChange = (plan: string) => {
195
+ setPlanFilter(plan);
196
+ setPage(1);
197
+ };
198
+
199
+ const handleIntervalChange = (interval: string) => {
200
+ setIntervalFilter(interval);
201
+ setPage(1);
202
+ };
203
+
204
+ const handleSearchChange = (value: string) => {
205
+ setSearchQuery(value);
206
+ setPage(1);
207
+ };
208
+
209
+ const handleLimitChange = (newLimit: number) => {
210
+ setLimit(newLimit);
211
+ setPage(1);
212
+ };
213
+
214
+ const hasActiveFilters = searchQuery || statusFilter !== "all" || planFilter !== "all" || intervalFilter !== "all";
215
+
216
+ const clearAllFilters = () => {
217
+ setSearchQuery("");
218
+ setStatusFilter("all");
219
+ setPlanFilter("all");
220
+ setIntervalFilter("all");
221
+ setPage(1);
222
+ };
223
+
224
+ // Fetch subscriptions data
225
+ const { data, isLoading, isFetching, error, refetch } = useQuery<SubscriptionsData>({
226
+ queryKey: ["superadmin-subscriptions", statusFilter, planFilter, intervalFilter, searchQuery, page, limit],
227
+ queryFn: async () => {
228
+ const queryString = getQueryParams();
229
+ const url = `/api/superadmin/subscriptions${queryString ? `?${queryString}` : ""}`;
230
+ const response = await fetch(url);
231
+ if (!response.ok) {
232
+ throw new Error("Failed to fetch subscriptions data");
233
+ }
234
+ return response.json();
235
+ },
236
+ retry: 2,
237
+ staleTime: 30000,
238
+ placeholderData: keepPreviousData,
239
+ });
240
+
241
+ // Loading state
242
+ if (isLoading) {
243
+ return (
244
+ <div className="space-y-6">
245
+ <div className="flex items-center gap-4">
246
+ <Button variant="outline" size="sm" asChild>
247
+ <Link href="/superadmin" className="flex items-center gap-2">
248
+ <ArrowLeft className="h-4 w-4" />
249
+ Back to Super Admin
250
+ </Link>
251
+ </Button>
252
+ <div>
253
+ <h1 className="text-3xl font-bold tracking-tight">Subscriptions</h1>
254
+ <p className="text-muted-foreground">Loading subscription data...</p>
255
+ </div>
256
+ </div>
257
+
258
+ <div className="flex items-center justify-center py-12">
259
+ <Loader2 className="h-8 w-8 animate-spin" />
260
+ </div>
261
+ </div>
262
+ );
263
+ }
264
+
265
+ // Error state
266
+ if (error) {
267
+ return (
268
+ <div className="space-y-6">
269
+ <div className="flex items-center gap-4">
270
+ <Button variant="outline" size="sm" asChild>
271
+ <Link href="/superadmin" className="flex items-center gap-2">
272
+ <ArrowLeft className="h-4 w-4" />
273
+ Back to Super Admin
274
+ </Link>
275
+ </Button>
276
+ <div>
277
+ <h1 className="text-3xl font-bold tracking-tight">Subscriptions</h1>
278
+ <p className="text-muted-foreground">Error loading data</p>
279
+ </div>
280
+ </div>
281
+
282
+ <Card>
283
+ <CardHeader>
284
+ <CardTitle className="flex items-center gap-2">
285
+ <AlertTriangle className="h-5 w-5 text-destructive" />
286
+ Error Loading Subscriptions
287
+ </CardTitle>
288
+ <CardDescription>
289
+ {error instanceof Error ? error.message : "Failed to load subscription data"}
290
+ </CardDescription>
291
+ </CardHeader>
292
+ <CardContent>
293
+ <Button
294
+ onClick={() => refetch()}
295
+ variant="outline"
296
+ className="flex items-center gap-2"
297
+ >
298
+ <RefreshCw className="h-4 w-4" />
299
+ Retry
300
+ </Button>
301
+ </CardContent>
302
+ </Card>
303
+ </div>
304
+ );
305
+ }
306
+
307
+ const stats = data?.stats;
308
+ const subscriptions = data?.subscriptions || [];
309
+
310
+ return (
311
+ <div className="space-y-6">
312
+ {/* Header */}
313
+ <div className="flex items-center gap-4">
314
+ <Button variant="outline" size="sm" asChild>
315
+ <Link href="/superadmin" className="flex items-center gap-2">
316
+ <ArrowLeft className="h-4 w-4" />
317
+ Back to Super Admin
318
+ </Link>
319
+ </Button>
320
+ <div className="flex-1">
321
+ <h1 className="text-3xl font-bold tracking-tight">Subscriptions</h1>
322
+ <p className="text-muted-foreground">
323
+ Manage billing and subscription overview
324
+ </p>
325
+ </div>
326
+ <Button
327
+ onClick={() => refetch()}
328
+ variant="outline"
329
+ size="sm"
330
+ className="flex items-center gap-2"
331
+ >
332
+ <RefreshCw className="h-4 w-4" />
333
+ Refresh
334
+ </Button>
335
+ </div>
336
+
337
+ {/* Stats Cards */}
338
+ <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-5 gap-4" data-cy={sel('superadmin.subscriptions.container')}>
339
+ {/* MRR Card */}
340
+ <Card className="border-green-200 bg-green-50/50" data-cy={sel('superadmin.subscriptions.stats.mrr')}>
341
+ <CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
342
+ <CardTitle className="text-sm font-medium">Monthly Recurring Revenue</CardTitle>
343
+ <DollarSign className="h-4 w-4 text-green-600" />
344
+ </CardHeader>
345
+ <CardContent>
346
+ <div className="text-2xl font-bold text-green-700">
347
+ {stats?.mrrFormatted || "$0.00"}
348
+ </div>
349
+ <p className="text-xs text-muted-foreground">
350
+ From {stats?.monthly || 0} monthly subscriptions
351
+ </p>
352
+ </CardContent>
353
+ </Card>
354
+
355
+ {/* ARR Card */}
356
+ <Card className="border-blue-200 bg-blue-50/50">
357
+ <CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
358
+ <CardTitle className="text-sm font-medium">Annual Recurring Revenue</CardTitle>
359
+ <TrendingUp className="h-4 w-4 text-blue-600" />
360
+ </CardHeader>
361
+ <CardContent>
362
+ <div className="text-2xl font-bold text-blue-700">
363
+ {stats?.arrFormatted || "$0.00"}
364
+ </div>
365
+ <p className="text-xs text-muted-foreground">
366
+ From {stats?.yearly || 0} yearly subscriptions
367
+ </p>
368
+ </CardContent>
369
+ </Card>
370
+
371
+ {/* Active Subscriptions */}
372
+ <Card data-cy={sel('superadmin.subscriptions.stats.activeCount')}>
373
+ <CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
374
+ <CardTitle className="text-sm font-medium">Active Subscriptions</CardTitle>
375
+ <CheckCircle className="h-4 w-4 text-green-600" />
376
+ </CardHeader>
377
+ <CardContent>
378
+ <div className="text-2xl font-bold">{stats?.active || 0}</div>
379
+ <p className="text-xs text-muted-foreground">
380
+ {stats?.trialing || 0} in trial
381
+ </p>
382
+ </CardContent>
383
+ </Card>
384
+
385
+ {/* Total Subscriptions */}
386
+ <Card>
387
+ <CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
388
+ <CardTitle className="text-sm font-medium">Total Subscriptions</CardTitle>
389
+ <CreditCard className="h-4 w-4 text-muted-foreground" />
390
+ </CardHeader>
391
+ <CardContent>
392
+ <div className="text-2xl font-bold">{stats?.total || 0}</div>
393
+ <p className="text-xs text-muted-foreground">
394
+ {stats?.free || 0} free, {stats?.paid || 0} paid
395
+ </p>
396
+ </CardContent>
397
+ </Card>
398
+
399
+ {/* Issues */}
400
+ <Card className={stats?.pastDue && stats.pastDue > 0 ? "border-yellow-200 bg-yellow-50/50" : ""}>
401
+ <CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
402
+ <CardTitle className="text-sm font-medium">Needs Attention</CardTitle>
403
+ <AlertCircle className={`h-4 w-4 ${stats?.pastDue && stats.pastDue > 0 ? "text-yellow-600" : "text-muted-foreground"}`} />
404
+ </CardHeader>
405
+ <CardContent>
406
+ <div className="text-2xl font-bold">{stats?.pastDue || 0}</div>
407
+ <p className="text-xs text-muted-foreground">
408
+ Past due payments
409
+ </p>
410
+ </CardContent>
411
+ </Card>
412
+ </div>
413
+
414
+ {/* Plan Distribution */}
415
+ {data?.planDistribution && data.planDistribution.length > 0 && (
416
+ <Card data-cy={sel('superadmin.subscriptions.stats.planDistribution')}>
417
+ <CardHeader>
418
+ <CardTitle className="flex items-center gap-2">
419
+ <TrendingUp className="h-5 w-5" />
420
+ Plan Distribution
421
+ </CardTitle>
422
+ <CardDescription>
423
+ Active subscriptions by plan
424
+ </CardDescription>
425
+ </CardHeader>
426
+ <CardContent>
427
+ <div className="flex flex-wrap gap-4">
428
+ {data.planDistribution.map((plan) => (
429
+ <div
430
+ key={plan.slug}
431
+ className="flex items-center gap-3 px-4 py-2 rounded-lg border bg-muted/30"
432
+ >
433
+ <div className="font-medium">{plan.name}</div>
434
+ <Badge variant="secondary">{plan.count}</Badge>
435
+ </div>
436
+ ))}
437
+ </div>
438
+ </CardContent>
439
+ </Card>
440
+ )}
441
+
442
+ {/* Filters */}
443
+ <div className="flex flex-col sm:flex-row flex-wrap gap-4 items-start sm:items-center">
444
+ <SearchInput
445
+ value={searchQuery}
446
+ onChange={handleSearchChange}
447
+ placeholder="Search by team name or email..."
448
+ className="w-full sm:max-w-xs"
449
+ data-cy="subscriptions-search-input"
450
+ />
451
+ <Select value={statusFilter} onValueChange={handleStatusChange}>
452
+ <SelectTrigger className="w-full sm:w-[140px]" data-cy="status-filter">
453
+ <SelectValue placeholder="Status" />
454
+ </SelectTrigger>
455
+ <SelectContent>
456
+ <SelectItem value="all">All Statuses</SelectItem>
457
+ <SelectItem value="active">Active</SelectItem>
458
+ <SelectItem value="trialing">Trialing</SelectItem>
459
+ <SelectItem value="past_due">Past Due</SelectItem>
460
+ <SelectItem value="canceled">Canceled</SelectItem>
461
+ </SelectContent>
462
+ </Select>
463
+ <Select value={planFilter} onValueChange={handlePlanChange}>
464
+ <SelectTrigger className="w-full sm:w-[140px]" data-cy="plan-filter">
465
+ <SelectValue placeholder="Plan" />
466
+ </SelectTrigger>
467
+ <SelectContent>
468
+ <SelectItem value="all">All Plans</SelectItem>
469
+ <SelectItem value="free">Free</SelectItem>
470
+ <SelectItem value="pro">Pro</SelectItem>
471
+ <SelectItem value="enterprise">Enterprise</SelectItem>
472
+ </SelectContent>
473
+ </Select>
474
+ <Select value={intervalFilter} onValueChange={handleIntervalChange}>
475
+ <SelectTrigger className="w-full sm:w-[140px]" data-cy="interval-filter">
476
+ <SelectValue placeholder="Interval" />
477
+ </SelectTrigger>
478
+ <SelectContent>
479
+ <SelectItem value="all">All Intervals</SelectItem>
480
+ <SelectItem value="monthly">Monthly</SelectItem>
481
+ <SelectItem value="yearly">Yearly</SelectItem>
482
+ </SelectContent>
483
+ </Select>
484
+ {hasActiveFilters && (
485
+ <Button
486
+ variant="ghost"
487
+ size="sm"
488
+ onClick={clearAllFilters}
489
+ data-cy="clear-filters-btn"
490
+ >
491
+ Clear Filters
492
+ </Button>
493
+ )}
494
+ {isFetching && !isLoading && (
495
+ <Loader2 className="h-4 w-4 animate-spin text-muted-foreground" />
496
+ )}
497
+ </div>
498
+
499
+ {/* Subscriptions Table */}
500
+ <Card>
501
+ <CardHeader>
502
+ <CardTitle className="flex items-center gap-2">
503
+ <CreditCard className="h-5 w-5" />
504
+ Subscriptions ({data?.pagination.total || 0})
505
+ </CardTitle>
506
+ <CardDescription>
507
+ All team subscriptions and their billing status
508
+ </CardDescription>
509
+ </CardHeader>
510
+ <CardContent>
511
+ {subscriptions.length === 0 ? (
512
+ <div className="text-center py-8">
513
+ <CreditCard className="mx-auto h-12 w-12 text-muted-foreground" />
514
+ <h3 className="mt-4 text-lg font-medium">No subscriptions found</h3>
515
+ <p className="mt-2 text-sm text-muted-foreground">
516
+ {hasActiveFilters
517
+ ? "Try adjusting your filters"
518
+ : "No subscriptions have been created yet"}
519
+ </p>
520
+ </div>
521
+ ) : (
522
+ <div className="rounded-md border">
523
+ <Table>
524
+ <TableHeader>
525
+ <TableRow>
526
+ <TableHead>Team</TableHead>
527
+ <TableHead>Plan</TableHead>
528
+ <TableHead>Status</TableHead>
529
+ <TableHead>Price</TableHead>
530
+ <TableHead>Period End</TableHead>
531
+ <TableHead>Actions</TableHead>
532
+ </TableRow>
533
+ </TableHeader>
534
+ <TableBody>
535
+ {subscriptions.map((sub) => {
536
+ const statusInfo = statusConfig[sub.status as keyof typeof statusConfig] || statusConfig.active;
537
+ const StatusIcon = statusInfo.icon;
538
+
539
+ return (
540
+ <TableRow key={sub.id}>
541
+ <TableCell>
542
+ <div>
543
+ <div className="font-medium">{sub.team.name}</div>
544
+ <div className="text-sm text-muted-foreground">
545
+ {sub.team.owner.email}
546
+ </div>
547
+ </div>
548
+ </TableCell>
549
+ <TableCell>
550
+ <Badge variant={sub.plan.type === "paid" ? "default" : "secondary"}>
551
+ {sub.plan.name}
552
+ </Badge>
553
+ </TableCell>
554
+ <TableCell>
555
+ <Badge className={statusInfo.className}>
556
+ <StatusIcon className="h-3 w-3 mr-1" />
557
+ {statusInfo.label}
558
+ </Badge>
559
+ {sub.cancelAtPeriodEnd && (
560
+ <div className="text-xs text-muted-foreground mt-1">
561
+ Cancels at period end
562
+ </div>
563
+ )}
564
+ </TableCell>
565
+ <TableCell>
566
+ {(() => {
567
+ const price = sub.billingInterval === 'yearly'
568
+ ? sub.plan.priceYearly
569
+ : sub.plan.priceMonthly;
570
+ const suffix = sub.billingInterval === 'yearly' ? '/yr' : '/mo';
571
+ return price && price > 0
572
+ ? `$${(price / 100).toFixed(2)}${suffix}`
573
+ : "Free";
574
+ })()}
575
+ </TableCell>
576
+ <TableCell>
577
+ <div className="text-sm">
578
+ {format(new Date(sub.currentPeriodEnd), "MMM dd, yyyy")}
579
+ </div>
580
+ {sub.trialEndsAt && new Date(sub.trialEndsAt) > new Date() && (
581
+ <div className="text-xs text-blue-600">
582
+ Trial ends {format(new Date(sub.trialEndsAt), "MMM dd")}
583
+ </div>
584
+ )}
585
+ </TableCell>
586
+ <TableCell>
587
+ <div className="flex items-center gap-2">
588
+ <Button variant="ghost" size="sm" asChild>
589
+ <Link href={`/superadmin/teams/${sub.team.id}`}>
590
+ View Team
591
+ </Link>
592
+ </Button>
593
+ {sub.externalSubscriptionId && (
594
+ <Button
595
+ variant="ghost"
596
+ size="sm"
597
+ asChild
598
+ className="text-muted-foreground"
599
+ >
600
+ <a
601
+ href={`https://dashboard.stripe.com/test/subscriptions/${sub.externalSubscriptionId}`}
602
+ target="_blank"
603
+ rel="noopener noreferrer"
604
+ >
605
+ <ExternalLink className="h-4 w-4" />
606
+ </a>
607
+ </Button>
608
+ )}
609
+ </div>
610
+ </TableCell>
611
+ </TableRow>
612
+ );
613
+ })}
614
+ </TableBody>
615
+ </Table>
616
+ </div>
617
+ )}
618
+
619
+ {/* Pagination */}
620
+ {data?.pagination && data.pagination.totalPages > 1 && (
621
+ <div className="mt-4">
622
+ <PaginationControls
623
+ page={data.pagination.page}
624
+ totalPages={data.pagination.totalPages}
625
+ total={data.pagination.total}
626
+ limit={data.pagination.limit}
627
+ onPageChange={setPage}
628
+ onLimitChange={handleLimitChange}
629
+ context="subscriptions"
630
+ />
631
+ </div>
632
+ )}
633
+ </CardContent>
634
+ </Card>
635
+
636
+ {/* Metadata Footer */}
637
+ {data?.metadata && (
638
+ <Card className="bg-muted/30">
639
+ <CardContent className="pt-4">
640
+ <div className="text-xs text-muted-foreground space-y-1">
641
+ <p>
642
+ Last updated:{" "}
643
+ {new Date(data.metadata.requestedAt).toLocaleString()}
644
+ </p>
645
+ <p>Requested by: {data.metadata.requestedBy}</p>
646
+ <p>Source: {data.metadata.source}</p>
647
+ </div>
648
+ </CardContent>
649
+ </Card>
650
+ )}
651
+ </div>
652
+ );
653
+ }
654
+
655
+ export default getTemplateOrDefaultClient("app/superadmin/subscriptions/page.tsx", SubscriptionsPage);