@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,190 @@
1
+ "use client";
2
+
3
+ import { AlertCircle, CheckCircle, Loader2 } from "lucide-react";
4
+ import { useSearchParams, useRouter } from "next/navigation";
5
+ import { Suspense, useEffect, useState } from "react";
6
+ import { Button } from '@nextsparkjs/core/components/ui/button';
7
+ import { Alert, AlertDescription } from '@nextsparkjs/core/components/ui/alert';
8
+ import { getTemplateOrDefaultClient } from '@nextsparkjs/registries/template-registry.client'
9
+ import { sel } from '@nextsparkjs/core/selectors'
10
+
11
+ function VerifyEmailContent() {
12
+ const searchParams = useSearchParams();
13
+ const router = useRouter();
14
+ const token = searchParams.get("token") || "";
15
+
16
+ const [status, setStatus] = useState<'loading' | 'success' | 'error'>('loading');
17
+ const [error, setError] = useState<string | null>(null);
18
+
19
+ useEffect(() => {
20
+ const handleVerification = async () => {
21
+ if (!token) {
22
+ setError("No verification token provided");
23
+ setStatus('error');
24
+ return;
25
+ }
26
+
27
+ try {
28
+ // Call the Better Auth verify endpoint with special header to avoid redirect loop
29
+ const response = await fetch(`/api/auth/verify-email?token=${token}`, {
30
+ method: 'GET',
31
+ credentials: 'include',
32
+ headers: {
33
+ 'x-verify-from-ui': 'true'
34
+ }
35
+ });
36
+
37
+ if (!response.ok) {
38
+ const errorText = await response.text();
39
+ setError(errorText || "Email verification failed");
40
+ setStatus('error');
41
+ } else {
42
+ setStatus('success');
43
+
44
+ // Create default metadata after successful verification
45
+ try {
46
+ const response = await fetch('/api/auth/session', {
47
+ method: 'GET',
48
+ credentials: 'include'
49
+ });
50
+
51
+ if (response.ok) {
52
+ const session = await response.json();
53
+ if (session?.user?.id) {
54
+ // Create default metadata for the newly verified user
55
+ await fetch('/api/internal/user-metadata', {
56
+ method: 'POST',
57
+ headers: {
58
+ 'Content-Type': 'application/json',
59
+ },
60
+ body: JSON.stringify({
61
+ userId: session.user.id,
62
+ metadata: {
63
+ uiPreferences: {
64
+ theme: "light",
65
+ sidebarCollapsed: false
66
+ },
67
+ securityPreferences: {
68
+ twoFactorEnabled: false,
69
+ loginAlertsEnabled: true
70
+ },
71
+ notificationsPreferences: {
72
+ pushEnabled: true,
73
+ loginAlertsEmail: true,
74
+ loginAlertsPush: true,
75
+ passwordChangesEmail: true,
76
+ passwordChangesPush: true,
77
+ suspiciousActivityEmail: true,
78
+ suspiciousActivityPush: true,
79
+ mentionsEmail: true,
80
+ mentionsPush: true,
81
+ projectUpdatesEmail: true,
82
+ projectUpdatesPush: false,
83
+ teamInvitesEmail: true,
84
+ teamInvitesPush: true,
85
+ newsletterEmail: false,
86
+ newsletterPush: false,
87
+ promotionsEmail: false,
88
+ promotionsPush: false,
89
+ featureAnnouncementsEmail: true,
90
+ featureAnnouncementsPush: false
91
+ }
92
+ }
93
+ })
94
+ });
95
+ console.log('Default metadata created for verified user:', session.user.id);
96
+ }
97
+ }
98
+ } catch (metaError) {
99
+ console.error('Error creating default metadata:', metaError);
100
+ // Don't fail the verification if metadata creation fails
101
+ }
102
+
103
+ // Redirect after successful verification
104
+ setTimeout(() => {
105
+ router.push('/dashboard');
106
+ }, 2000);
107
+ }
108
+ } catch {
109
+ setError("An unexpected error occurred");
110
+ setStatus('error');
111
+ }
112
+ };
113
+
114
+ handleVerification();
115
+ }, [token, router]);
116
+
117
+ if (status === 'loading') {
118
+ return (
119
+ <div className="space-y-4" data-cy={sel('auth.verifyEmail.loading')}>
120
+ <div className="text-center">
121
+ <h2 className="text-xl font-semibold">Verifying Email</h2>
122
+ <p className="text-sm text-muted-foreground mt-2">Please wait while we verify your email address</p>
123
+ </div>
124
+ <div className="flex justify-center py-6">
125
+ <Loader2 className="h-8 w-8 animate-spin text-muted-foreground" />
126
+ </div>
127
+ </div>
128
+ );
129
+ }
130
+
131
+ if (status === 'success') {
132
+ return (
133
+ <div className="space-y-4" data-cy={sel('auth.verifyEmail.success')}>
134
+ <div className="text-center">
135
+ <div className="flex justify-center mb-2">
136
+ <CheckCircle className="h-12 w-12 text-primary" />
137
+ </div>
138
+ <h2 className="text-xl font-semibold">Email Verified</h2>
139
+ <p className="text-sm text-muted-foreground mt-2">Your email has been successfully verified</p>
140
+ </div>
141
+ <Alert>
142
+ <AlertDescription>
143
+ Redirecting you to the dashboard...
144
+ </AlertDescription>
145
+ </Alert>
146
+ </div>
147
+ );
148
+ }
149
+
150
+ return (
151
+ <div className="space-y-4" data-cy={sel('auth.verifyEmail.error')}>
152
+ <div className="text-center">
153
+ <div className="flex justify-center mb-2">
154
+ <AlertCircle className="h-12 w-12 text-destructive" />
155
+ </div>
156
+ <h2 className="text-xl font-semibold">Verification Failed</h2>
157
+ <p className="text-sm text-muted-foreground mt-2">We couldn&apos;t verify your email address</p>
158
+ </div>
159
+ {error && (
160
+ <Alert variant="destructive" data-cy={sel('auth.verifyEmail.errorMessage')}>
161
+ <AlertDescription>{error}</AlertDescription>
162
+ </Alert>
163
+ )}
164
+ <div className="flex gap-2">
165
+ <Button onClick={() => router.push('/signup')} variant="outline" className="flex-1" data-cy={sel('auth.verifyEmail.backSignup')}>
166
+ Back to Sign Up
167
+ </Button>
168
+ <Button onClick={() => router.push('/login')} className="flex-1" data-cy={sel('auth.verifyEmail.goLogin')}>
169
+ Go to Login
170
+ </Button>
171
+ </div>
172
+ </div>
173
+ );
174
+ }
175
+
176
+ function VerifyEmailPage() {
177
+ return (
178
+ <Suspense fallback={
179
+ <div className="flex justify-center py-8">
180
+ <Loader2 className="h-8 w-8 animate-spin text-muted-foreground" />
181
+ </div>
182
+ }>
183
+ <VerifyEmailContent />
184
+ </Suspense>
185
+ );
186
+ }
187
+
188
+ export const dynamic = 'force-dynamic';
189
+
190
+ export default getTemplateOrDefaultClient('app/(auth)/verify-email/page.tsx', VerifyEmailPage)
@@ -0,0 +1,378 @@
1
+ /**
2
+ * Public Dynamic Catch-All Route
3
+ *
4
+ * Handles all public URLs for builder-enabled entities based on access.basePath configuration.
5
+ *
6
+ * Resolution strategy (longest-match-first):
7
+ * - /blog/my-post → posts (basePath: '/blog')
8
+ * - /about → pages (basePath: '/')
9
+ * - /blog → posts archive (exact basePath match)
10
+ *
11
+ * Template System:
12
+ * 1. Checks for theme template override first
13
+ * 2. Falls back to default PageRenderer if no template
14
+ */
15
+
16
+ import { notFound } from 'next/navigation'
17
+ import { query } from '@nextsparkjs/core/lib/db'
18
+ import { PageRenderer } from '@nextsparkjs/core/components/public/pageBuilder'
19
+ import {
20
+ matchPathToEntity,
21
+ getEntityBasePath,
22
+ } from '@nextsparkjs/core/lib/entities/schema-generator'
23
+ import { getEntityRegistry, setEntityRegistry } from '@nextsparkjs/core/lib/entities/queries'
24
+ import { TemplateService } from '@nextsparkjs/core/lib/services/template.service'
25
+ import { resolvePublicEntityFromUrl } from '@nextsparkjs/core/lib/api/entity/public-resolver'
26
+ import { PublicEntityGrid } from '@nextsparkjs/core/components/public/entities/PublicEntityGrid'
27
+ import type { EntityConfig } from '@nextsparkjs/core/lib/entities/types'
28
+ import type { Metadata } from 'next'
29
+ // Pattern resolution imports
30
+ import { PatternsResolverService } from '@nextsparkjs/core/lib/blocks/patterns-resolver.service'
31
+ import { extractPatternIds, resolvePatternReferences } from '@nextsparkjs/core/lib/blocks/pattern-resolver'
32
+ import type { BlockInstance } from '@nextsparkjs/core/types/blocks'
33
+ import type { PatternReference } from '@nextsparkjs/core/types/pattern-reference'
34
+ // Import registry directly - webpack resolves @nextsparkjs/registries alias at compile time
35
+ import { ENTITY_REGISTRY, ENTITY_METADATA } from '@nextsparkjs/registries/entity-registry'
36
+
37
+ // Initialize registry at module load time (before any component renders)
38
+ // This ensures the registry is available even if this page loads before the layout
39
+ setEntityRegistry(ENTITY_REGISTRY, ENTITY_METADATA)
40
+
41
+ /**
42
+ * Convert entity registry to format expected by matchPathToEntity
43
+ */
44
+ function getEntityConfigs(): Record<string, EntityConfig> {
45
+ const registry = getEntityRegistry()
46
+ const configs: Record<string, EntityConfig> = {}
47
+ for (const [key, entry] of Object.entries(registry)) {
48
+ configs[key] = entry.config as EntityConfig
49
+ }
50
+ return configs
51
+ }
52
+
53
+ // Enable ISR with 1 hour revalidation
54
+ export const revalidate = 3600
55
+
56
+ interface PageProps {
57
+ params: Promise<{ slug: string[] }>
58
+ searchParams: Promise<{ [key: string]: string | string[] | undefined }>
59
+ }
60
+
61
+ /**
62
+ * Base fields that all builder-enabled entities have (from migrations)
63
+ * These are the minimum fields needed for public page rendering
64
+ */
65
+ const BASE_PUBLIC_FIELDS = ['id', 'slug', 'title', 'status', 'blocks', 'locale', 'createdAt', 'userId']
66
+
67
+ /**
68
+ * Optional SEO fields that may exist on builder entities
69
+ */
70
+ const SEO_FIELDS = ['seoTitle', 'seoDescription', 'ogImage']
71
+
72
+ /**
73
+ * Common optional fields for content entities (posts, articles, etc.)
74
+ * These are checked dynamically - only included if they exist in entity.fields
75
+ */
76
+ const OPTIONAL_CONTENT_FIELDS = ['excerpt', 'featuredImage']
77
+
78
+ interface PublishedItem {
79
+ id: string
80
+ slug: string
81
+ title: string
82
+ status: string
83
+ blocks: Array<{
84
+ id: string
85
+ blockSlug: string
86
+ props: Record<string, unknown>
87
+ }>
88
+ excerpt?: string
89
+ featuredImage?: string
90
+ seoTitle?: string
91
+ seoDescription?: string
92
+ ogImage?: string
93
+ locale?: string
94
+ createdAt?: string
95
+ userId?: string
96
+ }
97
+
98
+ /**
99
+ * Quote a field name for PostgreSQL (handles camelCase)
100
+ */
101
+ function quoteField(field: string): string {
102
+ return /[A-Z]/.test(field) ? `"${field}"` : field
103
+ }
104
+
105
+ /**
106
+ * Build SELECT clause dynamically based on entity configuration
107
+ * Only includes fields that actually exist in the entity's schema
108
+ */
109
+ function buildPublicSelectClause(entity: EntityConfig): string {
110
+ const fields = new Set<string>(BASE_PUBLIC_FIELDS)
111
+
112
+ // Add SEO fields (these are standard for builder entities)
113
+ SEO_FIELDS.forEach(f => fields.add(f))
114
+
115
+ // Check entity.fields for optional content fields
116
+ const entityFieldNames = entity.fields?.map(f => f.name) || []
117
+ OPTIONAL_CONTENT_FIELDS.forEach(f => {
118
+ if (entityFieldNames.includes(f)) {
119
+ fields.add(f)
120
+ }
121
+ })
122
+
123
+ return Array.from(fields).map(quoteField).join(', ')
124
+ }
125
+
126
+ /**
127
+ * Resolve pattern references in blocks array
128
+ *
129
+ * Fetches all referenced patterns and expands them inline.
130
+ * This ensures public pages show the actual pattern content.
131
+ *
132
+ * @param blocks - Blocks array which may contain pattern references
133
+ * @returns Resolved blocks array with patterns expanded
134
+ */
135
+ async function getResolvedBlocks(
136
+ blocks: (BlockInstance | PatternReference)[]
137
+ ): Promise<BlockInstance[]> {
138
+ // Extract pattern IDs from blocks array
139
+ const patternIds = extractPatternIds(blocks)
140
+
141
+ // If no patterns referenced, return blocks as-is
142
+ if (patternIds.length === 0) {
143
+ return blocks as BlockInstance[]
144
+ }
145
+
146
+ try {
147
+ // Batch fetch all referenced patterns (only published ones)
148
+ const patterns = await PatternsResolverService.getByIds(patternIds)
149
+
150
+ // Build pattern cache (Map for O(1) lookup)
151
+ const patternCache = new Map(patterns.map((p) => [p.id, p]))
152
+
153
+ // Resolve pattern references and return flattened blocks
154
+ return resolvePatternReferences(blocks, patternCache)
155
+ } catch (error) {
156
+ console.error('[getResolvedBlocks] Failed to resolve patterns:', error)
157
+ // Fallback: Return blocks without pattern resolution
158
+ // Pattern references will be skipped gracefully
159
+ return blocks.filter((block) => !('type' in block && block.type === 'pattern')) as BlockInstance[]
160
+ }
161
+ }
162
+
163
+ /**
164
+ * Build the template path for an entity based on its basePath
165
+ */
166
+ function buildTemplatePath(entity: EntityConfig): string {
167
+ const basePath = getEntityBasePath(entity) || '/'
168
+ if (basePath === '/') {
169
+ return 'app/(public)/[entity]/page.tsx'
170
+ }
171
+ return `app/(public)${basePath}/[slug]/page.tsx`
172
+ }
173
+
174
+ /**
175
+ * Fetch a published item from the database
176
+ * Dynamically builds SELECT clause based on entity configuration
177
+ * to avoid querying non-existent columns
178
+ */
179
+ async function fetchPublishedItem(
180
+ entity: EntityConfig,
181
+ slug: string
182
+ ): Promise<PublishedItem | null> {
183
+ try {
184
+ const tableName = entity.tableName || entity.slug
185
+ const selectClause = buildPublicSelectClause(entity)
186
+
187
+ const result = await query<PublishedItem>(
188
+ `SELECT ${selectClause} FROM "${tableName}" WHERE slug = $1 AND status = 'published'`,
189
+ [slug]
190
+ )
191
+ return result.rows[0] || null
192
+ } catch (error) {
193
+ console.error(`[fetchPublishedItem] Error fetching ${entity.slug}:`, error)
194
+ return null
195
+ }
196
+ }
197
+
198
+ /**
199
+ * Generate metadata for the page
200
+ */
201
+ export async function generateMetadata({
202
+ params,
203
+ }: PageProps): Promise<Metadata> {
204
+ const slugParts = (await params).slug
205
+ const fullPath = '/' + slugParts.join('/')
206
+
207
+ // Get entity configs from registry
208
+ const registry = getEntityConfigs()
209
+
210
+ // Match path to builder entity
211
+ const match = matchPathToEntity(fullPath, registry)
212
+
213
+ if (match) {
214
+ const { entity, slug, isArchive } = match
215
+
216
+ // Archive page metadata
217
+ if (isArchive) {
218
+ return {
219
+ title: `${entity.names.plural} | Boilerplate`,
220
+ description: `Browse all ${entity.names.plural.toLowerCase()}`,
221
+ }
222
+ }
223
+
224
+ // Metadata from database
225
+ const item = await fetchPublishedItem(entity, slug)
226
+ if (item) {
227
+ return {
228
+ title: item.seoTitle || `${item.title} | Boilerplate`,
229
+ description: item.seoDescription || item.excerpt || undefined,
230
+ openGraph: {
231
+ title: item.seoTitle || item.title,
232
+ description: item.seoDescription || item.excerpt || undefined,
233
+ images: item.ogImage
234
+ ? [item.ogImage]
235
+ : item.featuredImage
236
+ ? [item.featuredImage]
237
+ : [],
238
+ type: 'article',
239
+ },
240
+ }
241
+ }
242
+ }
243
+
244
+ return {
245
+ title: 'Not Found',
246
+ }
247
+ }
248
+
249
+ /**
250
+ * Main catch-all page component
251
+ */
252
+ export default async function DynamicPublicPage({
253
+ params,
254
+ searchParams,
255
+ }: PageProps) {
256
+ const slugParts = (await params).slug
257
+ const resolvedSearchParams = await searchParams
258
+ const fullPath = '/' + slugParts.join('/')
259
+
260
+ // Get entity configs from registry
261
+ const registry = getEntityConfigs()
262
+
263
+ // Match path to builder entity using longest-match strategy
264
+ const match = matchPathToEntity(fullPath, registry)
265
+
266
+ if (match) {
267
+ const { entity, slug, isArchive } = match
268
+
269
+ // === ARCHIVE PAGE ===
270
+ // Handle archive pages (e.g., /blog without slug)
271
+ if (isArchive) {
272
+ // Check if entity has archive page configured
273
+ if (!entity.ui?.public?.hasArchivePage) {
274
+ notFound()
275
+ }
276
+
277
+ // Check for custom archive template (future enhancement)
278
+ // For now, use PublicEntityGrid
279
+ return (
280
+ <div className="container mx-auto px-4 py-8" data-cy="public-archive-page">
281
+ <div className="mb-8">
282
+ <h1 className="text-4xl font-bold text-foreground mb-4">
283
+ {entity.names.plural}
284
+ </h1>
285
+ <p className="text-lg text-muted-foreground">
286
+ Browse all {entity.names.plural.toLowerCase()}
287
+ </p>
288
+ </div>
289
+
290
+ <PublicEntityGrid
291
+ entityType={entity.slug}
292
+ entitySlug={entity.slug}
293
+ searchParams={resolvedSearchParams}
294
+ />
295
+ </div>
296
+ )
297
+ }
298
+
299
+ // === SINGLE ITEM PAGE ===
300
+ // Check for theme template override first
301
+ const templatePath = buildTemplatePath(entity)
302
+ if (TemplateService.hasOverride(templatePath)) {
303
+ const Template = TemplateService.getComponent(templatePath)
304
+ if (Template) {
305
+ // Template handles its own data fetching and rendering
306
+ // Pass params in the format expected by the template
307
+ return <Template params={Promise.resolve({ slug })} searchParams={searchParams} />
308
+ }
309
+ }
310
+
311
+ // No template override - use default rendering
312
+ const item = await fetchPublishedItem(entity, slug)
313
+
314
+ if (!item) {
315
+ notFound()
316
+ }
317
+
318
+ // Resolve pattern references before rendering
319
+ // This expands pattern blocks inline for public display
320
+ const resolvedBlocks = await getResolvedBlocks(
321
+ item.blocks as (BlockInstance | PatternReference)[]
322
+ )
323
+
324
+ // Default rendering with PageRenderer
325
+ return (
326
+ <main
327
+ className="min-h-screen bg-background"
328
+ data-cy="public-entity-page"
329
+ data-entity={entity.slug}
330
+ data-slug={slug}
331
+ >
332
+ <PageRenderer
333
+ page={{
334
+ id: item.id,
335
+ title: item.title,
336
+ slug: item.slug,
337
+ blocks: resolvedBlocks,
338
+ locale: item.locale || 'en',
339
+ }}
340
+ />
341
+ </main>
342
+ )
343
+ }
344
+
345
+ // === FALLBACK: Try legacy entity archive resolution ===
346
+ // This handles entity archives like /products, /clients when they're not using basePath
347
+ const resolution = await resolvePublicEntityFromUrl(fullPath)
348
+
349
+ if (
350
+ resolution.isValidPublicEntity &&
351
+ resolution.hasArchivePage &&
352
+ resolution.entityConfig
353
+ ) {
354
+ const entityConfig = resolution.entityConfig
355
+
356
+ return (
357
+ <div className="container mx-auto px-4 py-8" data-cy="public-archive-page">
358
+ <div className="mb-8">
359
+ <h1 className="text-4xl font-bold text-foreground mb-4">
360
+ {entityConfig.names.plural}
361
+ </h1>
362
+ <p className="text-lg text-muted-foreground">
363
+ Browse all {entityConfig.names.plural.toLowerCase()}
364
+ </p>
365
+ </div>
366
+
367
+ <PublicEntityGrid
368
+ entityType={entityConfig.slug}
369
+ entitySlug={entityConfig.slug}
370
+ searchParams={resolvedSearchParams}
371
+ />
372
+ </div>
373
+ )
374
+ }
375
+
376
+ // No match found
377
+ notFound()
378
+ }
@@ -0,0 +1,90 @@
1
+ import { notFound } from 'next/navigation'
2
+ import { DOCS_REGISTRY } from '@nextsparkjs/registries/docs-registry'
3
+ import { parseMarkdownFile } from '@nextsparkjs/core/lib/docs/parser'
4
+ import { DocsBreadcrumbs } from '@nextsparkjs/core/components/docs/docs-breadcrumbs'
5
+ import { DocsContent } from '@nextsparkjs/core/components/docs/docs-content'
6
+ import { sel } from '@nextsparkjs/core/selectors'
7
+ import { getTranslations } from 'next-intl/server'
8
+ import path from 'path'
9
+ import type { Metadata } from 'next'
10
+
11
+ interface DocsPageProps {
12
+ params: Promise<{
13
+ section: string
14
+ page: string
15
+ }>
16
+ }
17
+
18
+ export async function generateStaticParams() {
19
+ const params = []
20
+
21
+ // Generate params for public docs only
22
+ for (const section of DOCS_REGISTRY.public) {
23
+ for (const page of section.pages) {
24
+ params.push({
25
+ section: section.slug,
26
+ page: page.slug
27
+ })
28
+ }
29
+ }
30
+
31
+ return params
32
+ }
33
+
34
+ export async function generateMetadata({ params }: DocsPageProps): Promise<Metadata> {
35
+ const resolvedParams = await params
36
+ const { section: sectionSlug, page: pageSlug } = resolvedParams
37
+
38
+ const section = DOCS_REGISTRY.public.find(s => s.slug === sectionSlug)
39
+ if (!section) return { title: 'Page Not Found' }
40
+
41
+ const page = section.pages.find(p => p.slug === pageSlug)
42
+ if (!page) return { title: 'Page Not Found' }
43
+
44
+ const filePath = path.join(process.cwd(), page.path)
45
+ const { metadata } = await parseMarkdownFile(filePath)
46
+
47
+ return {
48
+ title: `${metadata.title} | Documentation`,
49
+ description: metadata.description || `Documentation for ${metadata.title}`
50
+ }
51
+ }
52
+
53
+ export default async function DocsPage({ params }: DocsPageProps) {
54
+ const resolvedParams = await params
55
+ const { section: sectionSlug, page: pageSlug } = resolvedParams
56
+ const t = await getTranslations('docs')
57
+
58
+ // Find section in public docs
59
+ const section = DOCS_REGISTRY.public.find(s => s.slug === sectionSlug)
60
+ if (!section) notFound()
61
+
62
+ // Find page in section
63
+ const page = section.pages.find(p => p.slug === pageSlug)
64
+ if (!page) notFound()
65
+
66
+ // Parse markdown file
67
+ const filePath = path.join(process.cwd(), page.path)
68
+ const { metadata, html } = await parseMarkdownFile(filePath)
69
+
70
+ return (
71
+ <div className="max-w-4xl" data-cy={sel('public.docs.pageDetail')}>
72
+ <DocsBreadcrumbs
73
+ items={[
74
+ { label: t('breadcrumbs.home'), href: '/docs' },
75
+ { label: section.title },
76
+ { label: metadata.title }
77
+ ]}
78
+ />
79
+
80
+ <article className="mt-8 prose prose-slate dark:prose-invert max-w-none">
81
+ <h1 className="text-4xl font-bold mb-2">{metadata.title}</h1>
82
+ {metadata.description && (
83
+ <p className="text-xl text-muted-foreground mb-8">{metadata.description}</p>
84
+ )}
85
+
86
+ <DocsContent html={html} />
87
+ </article>
88
+ </div>
89
+ )
90
+ }
@@ -0,0 +1,25 @@
1
+ import { Suspense } from 'react'
2
+ import { DocsLayout } from '@nextsparkjs/core/components/docs/docs-layout'
3
+ import { DocsSidebar } from '@nextsparkjs/core/components/docs/docs-sidebar'
4
+ import { DOCS_REGISTRY } from '@nextsparkjs/registries/docs-registry'
5
+ import { sel } from '@nextsparkjs/core/selectors'
6
+
7
+ export default function DocsLayoutPage({
8
+ children
9
+ }: {
10
+ children: React.ReactNode
11
+ }) {
12
+ // Only show public docs in this layout
13
+ const sections = DOCS_REGISTRY.public
14
+
15
+ return (
16
+ <DocsLayout>
17
+ <DocsSidebar sections={sections} />
18
+ <main className="flex-1 p-6 lg:p-8" data-cy={sel('public.docs.mainContent')}>
19
+ <Suspense fallback={<div className="animate-pulse">Loading...</div>}>
20
+ {children}
21
+ </Suspense>
22
+ </main>
23
+ </DocsLayout>
24
+ )
25
+ }