@nextsparkjs/core 0.1.0-beta.84 → 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 (356) 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 +2 -2
  180. package/templates/app/(auth)/forgot-password/page.tsx +216 -0
  181. package/templates/app/(auth)/layout.tsx +51 -0
  182. package/templates/app/(auth)/login/page.tsx +21 -0
  183. package/templates/app/(auth)/reset-password/page.tsx +212 -0
  184. package/templates/app/(auth)/signup/page.tsx +21 -0
  185. package/templates/app/(auth)/verify-email/page.tsx +190 -0
  186. package/templates/app/(public)/[...slug]/page.tsx +378 -0
  187. package/templates/app/(public)/docs/[section]/[page]/page.tsx +90 -0
  188. package/templates/app/(public)/docs/layout.tsx +25 -0
  189. package/templates/app/(public)/docs/page.tsx +81 -0
  190. package/templates/app/(public)/layout.tsx +41 -0
  191. package/templates/app/(public)/page.tsx +19 -0
  192. package/templates/app/403/page.tsx +89 -0
  193. package/templates/app/api/auth/[...all]/route.ts +78 -0
  194. package/templates/app/api/cron/billing/lifecycle/route.ts +98 -0
  195. package/templates/app/api/csp-report/route.ts +175 -0
  196. package/templates/app/api/devtools/config/entities/route.ts +108 -0
  197. package/templates/app/api/devtools/config/theme/route.ts +66 -0
  198. package/templates/app/api/devtools/tests/[...path]/route.ts +130 -0
  199. package/templates/app/api/devtools/tests/route.ts +134 -0
  200. package/templates/app/api/health/route.ts +29 -0
  201. package/templates/app/api/internal/user-metadata/route.ts +36 -0
  202. package/templates/app/api/superadmin/subscriptions/route.ts +310 -0
  203. package/templates/app/api/superadmin/teams/[teamId]/route.ts +286 -0
  204. package/templates/app/api/superadmin/teams/route.ts +188 -0
  205. package/templates/app/api/superadmin/users/[userId]/route.ts +540 -0
  206. package/templates/app/api/superadmin/users/route.ts +323 -0
  207. package/templates/app/api/user/delete-account/route.ts +55 -0
  208. package/templates/app/api/user/plan-flags/route.ts +283 -0
  209. package/templates/app/api/user/profile/route.ts +133 -0
  210. package/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +210 -0
  211. package/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +331 -0
  212. package/templates/app/api/v1/[entity]/[id]/route.ts +35 -0
  213. package/templates/app/api/v1/[entity]/docs.md +369 -0
  214. package/templates/app/api/v1/[entity]/presets.ts +194 -0
  215. package/templates/app/api/v1/[entity]/route.ts +31 -0
  216. package/templates/app/api/v1/api-keys/[id]/route.ts +303 -0
  217. package/templates/app/api/v1/api-keys/docs.md +101 -0
  218. package/templates/app/api/v1/api-keys/presets.ts +31 -0
  219. package/templates/app/api/v1/api-keys/route.ts +250 -0
  220. package/templates/app/api/v1/auth/docs.md +184 -0
  221. package/templates/app/api/v1/auth/presets.ts +44 -0
  222. package/templates/app/api/v1/auth/signup-with-invite/route.ts +227 -0
  223. package/templates/app/api/v1/billing/cancel/route.ts +206 -0
  224. package/templates/app/api/v1/billing/change-plan/route.ts +97 -0
  225. package/templates/app/api/v1/billing/check-action/route.ts +81 -0
  226. package/templates/app/api/v1/billing/checkout/route.ts +124 -0
  227. package/templates/app/api/v1/billing/docs.md +209 -0
  228. package/templates/app/api/v1/billing/plans/route.ts +85 -0
  229. package/templates/app/api/v1/billing/portal/route.ts +90 -0
  230. package/templates/app/api/v1/billing/presets.ts +121 -0
  231. package/templates/app/api/v1/billing/webhooks/stripe/route.ts +428 -0
  232. package/templates/app/api/v1/blocks/[slug]/route.ts +29 -0
  233. package/templates/app/api/v1/blocks/docs.md +173 -0
  234. package/templates/app/api/v1/blocks/presets.ts +121 -0
  235. package/templates/app/api/v1/blocks/route.ts +45 -0
  236. package/templates/app/api/v1/blocks/validate/route.ts +45 -0
  237. package/templates/app/api/v1/cron/docs.md +116 -0
  238. package/templates/app/api/v1/cron/presets.ts +26 -0
  239. package/templates/app/api/v1/cron/process/route.ts +108 -0
  240. package/templates/app/api/v1/devtools/blocks/route.ts +82 -0
  241. package/templates/app/api/v1/devtools/docs/route.ts +150 -0
  242. package/templates/app/api/v1/devtools/docs.md +204 -0
  243. package/templates/app/api/v1/devtools/features/route.ts +61 -0
  244. package/templates/app/api/v1/devtools/flows/route.ts +61 -0
  245. package/templates/app/api/v1/devtools/presets.ts +113 -0
  246. package/templates/app/api/v1/devtools/scheduled-actions/route.ts +120 -0
  247. package/templates/app/api/v1/devtools/testing/route.ts +82 -0
  248. package/templates/app/api/v1/media/docs.md +117 -0
  249. package/templates/app/api/v1/media/presets.ts +24 -0
  250. package/templates/app/api/v1/media/upload/route.ts +150 -0
  251. package/templates/app/api/v1/patterns/[id]/usages/route.ts +116 -0
  252. package/templates/app/api/v1/plugin/[...path]/route.ts +373 -0
  253. package/templates/app/api/v1/plugin/docs.md +79 -0
  254. package/templates/app/api/v1/plugin/presets.ts +21 -0
  255. package/templates/app/api/v1/plugin/route.ts +96 -0
  256. package/templates/app/api/v1/post-categories/[id]/route.ts +255 -0
  257. package/templates/app/api/v1/post-categories/docs.md +134 -0
  258. package/templates/app/api/v1/post-categories/presets.ts +78 -0
  259. package/templates/app/api/v1/post-categories/route.ts +119 -0
  260. package/templates/app/api/v1/team-invitations/[token]/accept/route.ts +179 -0
  261. package/templates/app/api/v1/team-invitations/[token]/decline/route.ts +120 -0
  262. package/templates/app/api/v1/team-invitations/[token]/route.ts +89 -0
  263. package/templates/app/api/v1/team-invitations/docs.md +88 -0
  264. package/templates/app/api/v1/team-invitations/presets.ts +43 -0
  265. package/templates/app/api/v1/team-invitations/route.ts +114 -0
  266. package/templates/app/api/v1/teams/[teamId]/invitations/route.ts +171 -0
  267. package/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +105 -0
  268. package/templates/app/api/v1/teams/[teamId]/invoices/route.ts +125 -0
  269. package/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +263 -0
  270. package/templates/app/api/v1/teams/[teamId]/members/route.ts +358 -0
  271. package/templates/app/api/v1/teams/[teamId]/route.ts +322 -0
  272. package/templates/app/api/v1/teams/[teamId]/subscription/route.ts +50 -0
  273. package/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +91 -0
  274. package/templates/app/api/v1/teams/docs.md +320 -0
  275. package/templates/app/api/v1/teams/presets.ts +178 -0
  276. package/templates/app/api/v1/teams/route.ts +293 -0
  277. package/templates/app/api/v1/teams/switch/route.ts +88 -0
  278. package/templates/app/api/v1/theme/[...path]/route.ts +361 -0
  279. package/templates/app/api/v1/theme/docs.md +74 -0
  280. package/templates/app/api/v1/theme/presets.ts +21 -0
  281. package/templates/app/api/v1/theme/route.ts +96 -0
  282. package/templates/app/api/v1/users/[id]/meta/[key]/route.ts +363 -0
  283. package/templates/app/api/v1/users/[id]/route.ts +302 -0
  284. package/templates/app/api/v1/users/docs.md +93 -0
  285. package/templates/app/api/v1/users/presets.ts +59 -0
  286. package/templates/app/api/v1/users/route.ts +197 -0
  287. package/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +117 -0
  288. package/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +103 -0
  289. package/templates/app/dashboard/(main)/[entity]/create/page.tsx +95 -0
  290. package/templates/app/dashboard/(main)/[entity]/error.tsx +51 -0
  291. package/templates/app/dashboard/(main)/[entity]/layout.tsx +113 -0
  292. package/templates/app/dashboard/(main)/[entity]/loading.tsx +61 -0
  293. package/templates/app/dashboard/(main)/[entity]/page.tsx +90 -0
  294. package/templates/app/dashboard/(main)/layout.tsx +98 -0
  295. package/templates/app/dashboard/(main)/loading.tsx +5 -0
  296. package/templates/app/dashboard/(main)/page.tsx +201 -0
  297. package/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +114 -0
  298. package/templates/app/dashboard/(main)/patterns/[id]/page.tsx +20 -0
  299. package/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +171 -0
  300. package/templates/app/dashboard/(main)/patterns/create/page.tsx +86 -0
  301. package/templates/app/dashboard/(main)/patterns/page.tsx +444 -0
  302. package/templates/app/dashboard/features/analytics/page.tsx +35 -0
  303. package/templates/app/dashboard/features/automation/page.tsx +35 -0
  304. package/templates/app/dashboard/features/layout.tsx +13 -0
  305. package/templates/app/dashboard/features/loading.tsx +5 -0
  306. package/templates/app/dashboard/features/webhooks/page.tsx +35 -0
  307. package/templates/app/dashboard/layout.tsx +86 -0
  308. package/templates/app/dashboard/permission-denied/page.tsx +29 -0
  309. package/templates/app/dashboard/settings/api-keys/loading.tsx +5 -0
  310. package/templates/app/dashboard/settings/api-keys/page.tsx +513 -0
  311. package/templates/app/dashboard/settings/billing/loading.tsx +5 -0
  312. package/templates/app/dashboard/settings/billing/page.tsx +284 -0
  313. package/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +222 -0
  314. package/templates/app/dashboard/settings/invoices/loading.tsx +5 -0
  315. package/templates/app/dashboard/settings/invoices/page.tsx +82 -0
  316. package/templates/app/dashboard/settings/layout.tsx +151 -0
  317. package/templates/app/dashboard/settings/loading.tsx +5 -0
  318. package/templates/app/dashboard/settings/notifications/loading.tsx +5 -0
  319. package/templates/app/dashboard/settings/notifications/page.tsx +462 -0
  320. package/templates/app/dashboard/settings/page.tsx +92 -0
  321. package/templates/app/dashboard/settings/password/loading.tsx +5 -0
  322. package/templates/app/dashboard/settings/password/page.tsx +306 -0
  323. package/templates/app/dashboard/settings/plans/loading.tsx +5 -0
  324. package/templates/app/dashboard/settings/plans/page.tsx +40 -0
  325. package/templates/app/dashboard/settings/profile/loading.tsx +5 -0
  326. package/templates/app/dashboard/settings/profile/page.tsx +686 -0
  327. package/templates/app/dashboard/settings/security/loading.tsx +5 -0
  328. package/templates/app/dashboard/settings/security/page.tsx +505 -0
  329. package/templates/app/dashboard/settings/teams/loading.tsx +5 -0
  330. package/templates/app/dashboard/settings/teams/page.tsx +272 -0
  331. package/templates/app/dashboard/settings/teams/permissions/page.tsx +92 -0
  332. package/templates/app/devtools/blocks/[slug]/page.tsx +39 -0
  333. package/templates/app/devtools/blocks/page.tsx +31 -0
  334. package/templates/app/devtools/config/page.tsx +31 -0
  335. package/templates/app/devtools/features/page.tsx +31 -0
  336. package/templates/app/devtools/flows/page.tsx +31 -0
  337. package/templates/app/devtools/layout.tsx +58 -0
  338. package/templates/app/devtools/page.tsx +121 -0
  339. package/templates/app/devtools/scheduled-actions/page.tsx +157 -0
  340. package/templates/app/devtools/style/page.tsx +330 -0
  341. package/templates/app/devtools/tags/page.tsx +31 -0
  342. package/templates/app/devtools/tests/[[...path]]/page.tsx +47 -0
  343. package/templates/app/favicon.ico +0 -0
  344. package/templates/app/globals.css +12 -0
  345. package/templates/app/layout.tsx +96 -0
  346. package/templates/app/public/page.tsx +30 -0
  347. package/templates/app/superadmin/docs/[section]/[page]/page.tsx +92 -0
  348. package/templates/app/superadmin/docs/page.tsx +75 -0
  349. package/templates/app/superadmin/layout.tsx +67 -0
  350. package/templates/app/superadmin/page.tsx +149 -0
  351. package/templates/app/superadmin/subscriptions/page.tsx +655 -0
  352. package/templates/app/superadmin/team-roles/page.tsx +493 -0
  353. package/templates/app/superadmin/teams/[teamId]/page.tsx +687 -0
  354. package/templates/app/superadmin/teams/page.tsx +302 -0
  355. package/templates/app/superadmin/users/[userId]/page.tsx +548 -0
  356. package/templates/app/superadmin/users/page.tsx +528 -0
@@ -0,0 +1,93 @@
1
+ # Users API
2
+
3
+ Manage user profiles and account settings.
4
+
5
+ ## Overview
6
+
7
+ The Users API allows you to view and update user profile information. Users can update their own profile, while admins can manage other users within their team.
8
+
9
+ ## Authentication
10
+
11
+ All endpoints require authentication via:
12
+ - **Session cookie** (for browser-based requests)
13
+ - **API Key** header (for server-to-server requests)
14
+
15
+ ## Endpoints
16
+
17
+ ### Get Current User
18
+ `GET /api/v1/users/me`
19
+
20
+ Returns the currently authenticated user's profile.
21
+
22
+ **Example Response:**
23
+ ```json
24
+ {
25
+ "id": "user_123",
26
+ "name": "John Doe",
27
+ "email": "john@example.com",
28
+ "image": "/uploads/avatar.png",
29
+ "createdAt": "2024-01-15T10:30:00Z",
30
+ "teams": [
31
+ {
32
+ "id": "team_456",
33
+ "name": "Acme Corp",
34
+ "role": "admin"
35
+ }
36
+ ]
37
+ }
38
+ ```
39
+
40
+ ### Update Current User
41
+ `PATCH /api/v1/users/me`
42
+
43
+ Update the current user's profile.
44
+
45
+ **Request Body:**
46
+ ```json
47
+ {
48
+ "name": "John Smith",
49
+ "image": "/uploads/new-avatar.png"
50
+ }
51
+ ```
52
+
53
+ ### List Team Members
54
+ `GET /api/v1/users`
55
+
56
+ Returns all users in the current team.
57
+
58
+ **Query Parameters:**
59
+ - `limit` (number, optional): Maximum records to return. Default: 20
60
+ - `offset` (number, optional): Number of records to skip. Default: 0
61
+ - `search` (string, optional): Search by name or email
62
+
63
+ ### Get User by ID
64
+ `GET /api/v1/users/[id]`
65
+
66
+ Returns a specific user's profile (within the same team).
67
+
68
+ **Path Parameters:**
69
+ - `id` (string, required): User ID
70
+
71
+ ## User Roles
72
+
73
+ Team-level roles determine permissions:
74
+ - `member` - Basic team member
75
+ - `admin` - Can manage team settings and members
76
+ - `owner` - Full control of the team
77
+
78
+ ## Error Responses
79
+
80
+ | Status | Description |
81
+ |--------|-------------|
82
+ | 400 | Bad Request - Invalid parameters |
83
+ | 401 | Unauthorized - Missing or invalid auth |
84
+ | 403 | Forbidden - Insufficient permissions |
85
+ | 404 | Not Found - User doesn't exist |
86
+ | 422 | Validation Error - Invalid data |
87
+
88
+ ## Related APIs
89
+
90
+ - **[Auth](/api/v1/auth)** - Authentication and session management
91
+ - **[Teams](/api/v1/teams)** - Team membership and roles
92
+ - **[Team Invitations](/api/v1/team-invitations)** - Invite users to teams
93
+ - **[API Keys](/api/v1/api-keys)** - Manage programmatic access
@@ -0,0 +1,59 @@
1
+ /**
2
+ * API Presets for Users
3
+ *
4
+ * These presets appear in the DevTools API Explorer's "Presets" tab.
5
+ */
6
+
7
+ import { defineApiEndpoint } from '@nextsparkjs/core/types/api-presets'
8
+
9
+ export default defineApiEndpoint({
10
+ endpoint: '/api/v1/users',
11
+ summary: 'Manage user profiles and team members',
12
+ presets: [
13
+ {
14
+ id: 'get-me',
15
+ title: 'Get Current User',
16
+ description: 'Fetch the currently authenticated user profile',
17
+ method: 'GET',
18
+ pathParams: {
19
+ id: 'me'
20
+ },
21
+ tags: ['read', 'profile']
22
+ },
23
+ {
24
+ id: 'list-team-members',
25
+ title: 'List Team Members',
26
+ description: 'Fetch all users in the current team',
27
+ method: 'GET',
28
+ params: {
29
+ limit: 20,
30
+ offset: 0
31
+ },
32
+ tags: ['read', 'list']
33
+ },
34
+ {
35
+ id: 'update-profile',
36
+ title: 'Update My Profile',
37
+ description: 'Update current user name',
38
+ method: 'PATCH',
39
+ pathParams: {
40
+ id: 'me'
41
+ },
42
+ payload: {
43
+ name: 'Updated Name'
44
+ },
45
+ tags: ['write', 'profile']
46
+ },
47
+ {
48
+ id: 'search-users',
49
+ title: 'Search Team Members',
50
+ description: 'Search users by name or email',
51
+ method: 'GET',
52
+ params: {
53
+ search: 'john',
54
+ limit: 10
55
+ },
56
+ tags: ['read', 'search']
57
+ }
58
+ ]
59
+ })
@@ -0,0 +1,197 @@
1
+ import { NextRequest, NextResponse } from 'next/server';
2
+ import { queryWithRLS, mutateWithRLS } from '@nextsparkjs/core/lib/db';
3
+ import {
4
+ createApiResponse,
5
+ createApiError,
6
+ parsePaginationParams,
7
+ createPaginationMeta,
8
+ parseFilters,
9
+ withApiLogging,
10
+ handleCorsPreflightRequest,
11
+ addCorsHeaders,
12
+ parseMetaParams,
13
+ includeEntityMetadata,
14
+ handleEntityMetadataInResponse,
15
+ processEntityMetadata
16
+ } from '@nextsparkjs/core/lib/api/helpers';
17
+ import { authenticateRequest } from '@nextsparkjs/core/lib/api/auth/dual-auth';
18
+ import { hasAdminPermission } from '@nextsparkjs/core/lib/api/auth/permissions';
19
+ import { z } from 'zod';
20
+
21
+ const createUserSchema = z.object({
22
+ email: z.string().email('Invalid email format'),
23
+ firstName: z.string().min(1, 'First name is required'),
24
+ lastName: z.string().min(1, 'Last name is required'),
25
+ country: z.string().min(2, 'Country is required (minimum 2 characters)'),
26
+ image: z.string().url('Invalid image URL format').optional(),
27
+ language: z.string().optional().default('en'),
28
+ timezone: z.string().optional().default('UTC'),
29
+ role: z.enum(['member', 'colaborator']).default('member'),
30
+ metas: z.record(z.string(), z.any()).optional()
31
+ });
32
+
33
+
34
+
35
+ // Handle CORS preflight
36
+ export async function OPTIONS() {
37
+ return handleCorsPreflightRequest();
38
+ }
39
+
40
+ // GET /api/v1/users - List users with dual auth
41
+ export const GET = withApiLogging(async (req: NextRequest): Promise<NextResponse> => {
42
+ try {
43
+ // Authenticate using dual auth
44
+ const authResult = await authenticateRequest(req);
45
+
46
+ if (!authResult.success) {
47
+ return NextResponse.json(
48
+ { success: false, error: 'Authentication required', code: 'AUTHENTICATION_FAILED' },
49
+ { status: 401 }
50
+ );
51
+ }
52
+
53
+ if (authResult.rateLimitResponse) {
54
+ return authResult.rateLimitResponse as NextResponse;
55
+ }
56
+
57
+ // SECURITY: Only superadmins can list all users
58
+ if (!hasAdminPermission(authResult, 'users:read')) {
59
+ const response = createApiError('Insufficient permissions. Superadmin access required.', 403);
60
+ return addCorsHeaders(response);
61
+ }
62
+
63
+ const metaParams = parseMetaParams(req);
64
+ const { page, limit, offset } = parsePaginationParams(req);
65
+ const filters = parseFilters(req);
66
+
67
+ // Build WHERE clause based on filters
68
+ let whereClause = 'WHERE 1=1';
69
+ const queryParams: (string | number)[] = [];
70
+ let paramCount = 1;
71
+
72
+ if (filters.role) {
73
+ whereClause += ` AND role = $${paramCount}`;
74
+ queryParams.push(filters.role);
75
+ paramCount++;
76
+ }
77
+
78
+ // Add pagination params
79
+ queryParams.push(limit, offset);
80
+
81
+ const users = await queryWithRLS(
82
+ `SELECT id, email, name, "firstName", "lastName", image, country, timezone, language, role, "emailVerified", "createdAt", "updatedAt"
83
+ FROM "users"
84
+ ${whereClause}
85
+ ORDER BY "createdAt" DESC
86
+ LIMIT $${paramCount++} OFFSET $${paramCount++}`,
87
+ queryParams,
88
+ authResult.user!.id
89
+ );
90
+
91
+ // Get total count for pagination
92
+ const totalResult = await queryWithRLS<{ count: number }>(
93
+ `SELECT COUNT(*) as count FROM "users" ${whereClause}`,
94
+ queryParams.slice(0, -2), // Remove limit and offset
95
+ authResult.user!.id
96
+ );
97
+
98
+ const total = totalResult[0]?.count || 0;
99
+ const paginationMeta = createPaginationMeta(page, limit, total);
100
+
101
+ // Include metadata if requested (usando helper compartido)
102
+ const usersWithMeta = await includeEntityMetadata('user', users as { id: string }[], metaParams, authResult.user!.id);
103
+
104
+ const response = createApiResponse(usersWithMeta, paginationMeta);
105
+ return addCorsHeaders(response);
106
+ } catch (error) {
107
+ console.error('Error fetching users:', error);
108
+ const response = createApiError('Internal server error', 500);
109
+ return addCorsHeaders(response);
110
+ }
111
+ });
112
+
113
+ // POST /api/v1/users - Create user with dual auth
114
+ export const POST = withApiLogging(async (req: NextRequest): Promise<NextResponse> => {
115
+ try {
116
+ // Authenticate using dual auth
117
+ const authResult = await authenticateRequest(req);
118
+
119
+ if (!authResult.success) {
120
+ return NextResponse.json(
121
+ { success: false, error: 'Authentication required', code: 'AUTHENTICATION_FAILED' },
122
+ { status: 401 }
123
+ );
124
+ }
125
+
126
+ if (authResult.rateLimitResponse) {
127
+ return authResult.rateLimitResponse as NextResponse;
128
+ }
129
+
130
+ // SECURITY: Only superadmins can create users
131
+ if (!hasAdminPermission(authResult, 'users:write')) {
132
+ const response = createApiError('Insufficient permissions. Superadmin access required.', 403);
133
+ return addCorsHeaders(response);
134
+ }
135
+
136
+ const body = await req.json();
137
+ const { metas, ...userData } = body;
138
+ const validatedData = createUserSchema.parse(userData);
139
+
140
+ // Check if email already exists
141
+ const existingUser = await queryWithRLS(
142
+ 'SELECT id FROM "users" WHERE email = $1',
143
+ [validatedData.email],
144
+ authResult.user!.id
145
+ );
146
+
147
+ if (existingUser.length > 0) {
148
+ const response = createApiError('Email already exists', 409, null, 'EMAIL_EXISTS');
149
+ return addCorsHeaders(response);
150
+ }
151
+
152
+ const newUserId = globalThis.crypto.randomUUID();
153
+
154
+ // Create user
155
+ const result = await mutateWithRLS(
156
+ `INSERT INTO "users" (id, email, "firstName", "lastName", country, image, language, timezone, role, "emailVerified")
157
+ VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, true) RETURNING *`,
158
+ [
159
+ newUserId,
160
+ validatedData.email,
161
+ validatedData.firstName,
162
+ validatedData.lastName,
163
+ validatedData.country,
164
+ validatedData.image || null,
165
+ validatedData.language,
166
+ validatedData.timezone,
167
+ validatedData.role
168
+ ],
169
+ authResult.user!.id
170
+ );
171
+
172
+ const createdUser = result.rows[0];
173
+
174
+ // Handle metadata if provided (usando helper compartido)
175
+ const metadataWasProvided = metas && typeof metas === 'object' && Object.keys(metas).length > 0;
176
+
177
+ if (metadataWasProvided) {
178
+ await processEntityMetadata('user', newUserId, metas, authResult.user!.id);
179
+ }
180
+
181
+ // Crear respuesta según criterio: incluir metadata solo si se envió en el payload
182
+ const responseData = await handleEntityMetadataInResponse('user', createdUser as { id: string }, metadataWasProvided, authResult.user!.id);
183
+
184
+ const response = createApiResponse(responseData, { created: true }, 201);
185
+ return addCorsHeaders(response);
186
+ } catch (error) {
187
+ if (error instanceof z.ZodError) {
188
+ const response = createApiError('Validation error', 400, error.issues, 'VALIDATION_ERROR');
189
+ return addCorsHeaders(response);
190
+ }
191
+
192
+ console.error('Error creating user:', error);
193
+ const response = createApiError('Internal server error', 500);
194
+ return addCorsHeaders(response);
195
+ }
196
+ });
197
+
@@ -0,0 +1,117 @@
1
+ 'use client'
2
+
3
+ import { notFound, useRouter } from 'next/navigation'
4
+ import { useParams } from 'next/navigation'
5
+ import { useEffect, useState } from 'react'
6
+ import { clientEntityRegistry, ensureClientInitialized, type ClientEntityConfig } from '@nextsparkjs/registries/entity-registry.client'
7
+ import { EntityFormWrapper } from '@nextsparkjs/core/components/entities/wrappers/EntityFormWrapper'
8
+ import { BuilderEditorView } from '@nextsparkjs/core/components/dashboard/block-editor/builder-editor-view'
9
+ import { getTemplateOrDefaultClient } from '@nextsparkjs/registries/template-registry.client'
10
+ import { getEntityData } from '@nextsparkjs/core/lib/api/entities'
11
+
12
+ function EntityEditPage() {
13
+ const params = useParams()
14
+ const router = useRouter()
15
+ const [entityConfig, setEntityConfig] = useState<ClientEntityConfig | null>(null)
16
+ const [initialData, setInitialData] = useState<Record<string, unknown> | null>(null)
17
+ const [loading, setLoading] = useState(true)
18
+
19
+ // Extract entity slug and id from params
20
+ const entitySlug = params.entity as string
21
+ const entityId = params.id as string
22
+
23
+ useEffect(() => {
24
+ async function loadEntityData() {
25
+ if (!entitySlug || !entityId) {
26
+ setLoading(false)
27
+ return
28
+ }
29
+
30
+ try {
31
+ // Load entity config (client-safe)
32
+ await ensureClientInitialized()
33
+ const config = clientEntityRegistry.getBySlug(entitySlug)
34
+
35
+ if (!config) {
36
+ setEntityConfig(null)
37
+ setLoading(false)
38
+ return
39
+ }
40
+
41
+ setEntityConfig(config)
42
+
43
+ // For builder-enabled entities, BuilderEditorView handles its own data fetching
44
+ // For regular entities, fetch data here
45
+ if (!config.builder?.enabled) {
46
+ const data = await getEntityData(entitySlug, entityId, true)
47
+ setInitialData(data as Record<string, unknown>)
48
+ }
49
+ } catch (error) {
50
+ console.error('Error loading entity:', error)
51
+ setEntityConfig(null)
52
+ setInitialData(null)
53
+ } finally {
54
+ setLoading(false)
55
+ }
56
+ }
57
+
58
+ loadEntityData()
59
+ }, [entitySlug, entityId])
60
+
61
+ if (loading) {
62
+ return (
63
+ <div className="flex items-center justify-center min-h-[400px]">
64
+ <div className="animate-spin rounded-full h-8 w-8 border-b-2 border-primary" />
65
+ </div>
66
+ )
67
+ }
68
+
69
+ if (!entityConfig) {
70
+ notFound()
71
+ }
72
+
73
+ // For non-builder entities, we need the initial data
74
+ if (!entityConfig.builder?.enabled && !initialData) {
75
+ notFound()
76
+ }
77
+
78
+ if (!entityConfig.features?.enabled) {
79
+ notFound()
80
+ }
81
+
82
+ // Check if entity should be accessible via dashboard route
83
+ // Entities with showInMenu: false are managed elsewhere (e.g., settings)
84
+ if (!entityConfig.features?.showInMenu) {
85
+ notFound()
86
+ }
87
+
88
+ // Use BuilderEditorView for builder-enabled entities
89
+ if (entityConfig.builder?.enabled) {
90
+ return (
91
+ <BuilderEditorView
92
+ entitySlug={entitySlug}
93
+ entityConfig={entityConfig}
94
+ id={entityId}
95
+ mode="edit"
96
+ />
97
+ )
98
+ }
99
+
100
+ // Use EntityFormWrapper for regular entities
101
+ return (
102
+ <EntityFormWrapper
103
+ entityType={entitySlug}
104
+ id={entityId}
105
+ mode="edit"
106
+ onSuccess={() => {
107
+ // For edit, redirect to the entity detail view
108
+ router.push(`/dashboard/${entitySlug}/${entityId}`)
109
+ }}
110
+ onError={(error) => {
111
+ console.error(`Error updating ${entityConfig.displayName}:`, error)
112
+ }}
113
+ />
114
+ )
115
+ }
116
+
117
+ export default getTemplateOrDefaultClient('app/dashboard/(main)/[entity]/[id]/edit/page.tsx', EntityEditPage)
@@ -0,0 +1,103 @@
1
+ import { notFound, redirect } from 'next/navigation'
2
+ import { getEntity, getEntityRegistry, getChildEntities, setEntityRegistry } from '@nextsparkjs/core/lib/entities/queries'
3
+ import { EntityDetailWrapper } from '@nextsparkjs/core/components/entities/wrappers/EntityDetailWrapper'
4
+ import type { Metadata } from 'next'
5
+ import { TemplateService } from '@nextsparkjs/core/lib/services/template.service'
6
+ import type { EntityConfig, ChildEntityDefinition } from '@nextsparkjs/core/lib/entities/types'
7
+ // Import registry directly - webpack resolves @nextsparkjs/registries alias at compile time
8
+ import { ENTITY_REGISTRY, ENTITY_METADATA } from '@nextsparkjs/registries/entity-registry'
9
+
10
+ // Initialize registry at module load time (before any component renders)
11
+ setEntityRegistry(ENTITY_REGISTRY, ENTITY_METADATA)
12
+
13
+ // Type guard to check if entity is a full EntityConfig
14
+ function isEntityConfig(entity: EntityConfig | ChildEntityDefinition): entity is EntityConfig {
15
+ return 'slug' in entity
16
+ }
17
+
18
+ interface PageProps {
19
+ params: Promise<{
20
+ entity: string
21
+ id: string
22
+ }>
23
+ }
24
+
25
+ async function EntityDetailPage({ params }: PageProps) {
26
+ const resolvedParams = await params
27
+ const entitySlug = resolvedParams.entity
28
+
29
+ // Check if there's a specific template override for this entity
30
+ // e.g., app/dashboard/(main)/boards/[id]/page.tsx for boards
31
+ const specificTemplatePath = `app/dashboard/(main)/${entitySlug}/[id]/page.tsx`
32
+ if (TemplateService.hasOverride(specificTemplatePath)) {
33
+ const OverrideComponent = TemplateService.getComponent(specificTemplatePath)
34
+ if (OverrideComponent) {
35
+ console.log(`🎨 Entity-specific template override applied for ${specificTemplatePath}`)
36
+ return <OverrideComponent params={params} />
37
+ }
38
+ }
39
+
40
+ // Verificar que la entidad existe en el registro
41
+ if (!(entitySlug in getEntityRegistry())) {
42
+ notFound()
43
+ }
44
+
45
+ const entityConfig = getEntity(entitySlug as string)
46
+ if (!entityConfig || !isEntityConfig(entityConfig)) {
47
+ notFound()
48
+ }
49
+
50
+ // Verificar que la entidad está habilitada
51
+ if (!entityConfig.enabled) {
52
+ notFound()
53
+ }
54
+
55
+ // Check if entity should be accessible via dashboard route
56
+ // Entities with showInMenu: false are managed elsewhere (e.g., settings)
57
+ if (!entityConfig.ui?.dashboard?.showInMenu) {
58
+ notFound()
59
+ }
60
+
61
+ // Builder-enabled entities redirect to edit view
62
+ // Detail view doesn't make sense for entities using the page builder
63
+ if (entityConfig.builder?.enabled) {
64
+ redirect(`/dashboard/${entitySlug}/${resolvedParams.id}/edit`)
65
+ }
66
+
67
+ // Get child entities for this parent entity
68
+ const childEntities = getChildEntities(entitySlug as string)
69
+ const childEntityNames = childEntities.map(child => child.name)
70
+
71
+ return (
72
+ <EntityDetailWrapper
73
+ entityType={entityConfig.slug}
74
+ id={resolvedParams.id}
75
+ childEntityNames={childEntityNames}
76
+ />
77
+ )
78
+ }
79
+
80
+ export async function generateMetadata({ params }: PageProps): Promise<Metadata> {
81
+ const resolvedParams = await params
82
+ const entitySlug = resolvedParams.entity
83
+
84
+ if (!(entitySlug in getEntityRegistry())) {
85
+ return {
86
+ title: 'Not Found - Dashboard'
87
+ }
88
+ }
89
+
90
+ const entityConfig = getEntity(entitySlug as string)
91
+ if (!entityConfig || !isEntityConfig(entityConfig)) {
92
+ return {
93
+ title: 'Not Found - Dashboard'
94
+ }
95
+ }
96
+
97
+ return {
98
+ title: `${entityConfig.names.plural} #${resolvedParams.id} - Dashboard`,
99
+ description: `View details for ${entityConfig.names.singular}`
100
+ }
101
+ }
102
+
103
+ export default EntityDetailPage
@@ -0,0 +1,95 @@
1
+ 'use client'
2
+
3
+ import { notFound, useRouter } from 'next/navigation'
4
+ import { useParams } from 'next/navigation'
5
+ import { useEffect, useState } from 'react'
6
+ import { clientEntityRegistry, ensureClientInitialized, type ClientEntityConfig } from '@nextsparkjs/registries/entity-registry.client'
7
+ import { EntityFormWrapper } from '@nextsparkjs/core/components/entities/wrappers/EntityFormWrapper'
8
+ import { BuilderEditorView } from '@nextsparkjs/core/components/dashboard/block-editor/builder-editor-view'
9
+ import { getTemplateOrDefaultClient } from '@nextsparkjs/registries/template-registry.client'
10
+
11
+ function EntityCreatePage() {
12
+ const params = useParams()
13
+ const router = useRouter()
14
+ const [entityConfig, setEntityConfig] = useState<ClientEntityConfig | null>(null)
15
+ const [loading, setLoading] = useState(true)
16
+
17
+ // Extract entity slug from params
18
+ const entitySlug = params.entity as string
19
+
20
+ useEffect(() => {
21
+ async function loadEntityConfig() {
22
+ if (!entitySlug) {
23
+ setLoading(false)
24
+ return
25
+ }
26
+
27
+ try {
28
+ await ensureClientInitialized()
29
+ const config = clientEntityRegistry.getBySlug(entitySlug)
30
+ setEntityConfig(config || null)
31
+ } catch (error) {
32
+ console.error('Error loading entity config:', error)
33
+ setEntityConfig(null)
34
+ } finally {
35
+ setLoading(false)
36
+ }
37
+ }
38
+
39
+ loadEntityConfig()
40
+ }, [entitySlug])
41
+
42
+ if (loading) {
43
+ return (
44
+ <div className="flex items-center justify-center min-h-[400px]">
45
+ <div className="animate-spin rounded-full h-8 w-8 border-b-2 border-primary" />
46
+ </div>
47
+ )
48
+ }
49
+
50
+ if (!entityConfig) {
51
+ notFound()
52
+ }
53
+
54
+ if (!entityConfig.features?.enabled) {
55
+ notFound()
56
+ }
57
+
58
+ // Check if entity should be accessible via dashboard route
59
+ // Entities with showInMenu: false are managed elsewhere (e.g., settings)
60
+ if (!entityConfig.features?.showInMenu) {
61
+ notFound()
62
+ }
63
+
64
+ // Use BuilderEditorView for builder-enabled entities
65
+ if (entityConfig.builder?.enabled) {
66
+ return (
67
+ <BuilderEditorView
68
+ entitySlug={entitySlug}
69
+ entityConfig={entityConfig}
70
+ mode="create"
71
+ />
72
+ )
73
+ }
74
+
75
+ // Use EntityFormWrapper for regular entities
76
+ return (
77
+ <EntityFormWrapper
78
+ entityType={entitySlug}
79
+ mode="create"
80
+ onSuccess={(createdId) => {
81
+ // For create, redirect to the entity detail view if we have the ID, otherwise to list
82
+ if (createdId) {
83
+ router.push(`/dashboard/${entitySlug}/${createdId}`)
84
+ } else {
85
+ router.push(`/dashboard/${entitySlug}`)
86
+ }
87
+ }}
88
+ onError={(error) => {
89
+ console.error(`Error creating ${entityConfig.displayName}:`, error)
90
+ }}
91
+ />
92
+ )
93
+ }
94
+
95
+ export default getTemplateOrDefaultClient('app/dashboard/(main)/[entity]/create/page.tsx', EntityCreatePage)