@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.
- package/dist/styles/classes.json +1 -1
- package/dist/templates/app/(auth)/forgot-password/page.tsx +216 -0
- package/dist/templates/app/(auth)/layout.tsx +51 -0
- package/dist/templates/app/(auth)/login/page.tsx +21 -0
- package/dist/templates/app/(auth)/reset-password/page.tsx +212 -0
- package/dist/templates/app/(auth)/signup/page.tsx +21 -0
- package/dist/templates/app/(auth)/verify-email/page.tsx +190 -0
- package/dist/templates/app/(public)/[...slug]/page.tsx +378 -0
- package/dist/templates/app/(public)/docs/[section]/[page]/page.tsx +90 -0
- package/dist/templates/app/(public)/docs/layout.tsx +25 -0
- package/dist/templates/app/(public)/docs/page.tsx +81 -0
- package/dist/templates/app/(public)/layout.tsx +41 -0
- package/dist/templates/app/(public)/page.tsx +19 -0
- package/dist/templates/app/403/page.tsx +89 -0
- package/dist/templates/app/api/auth/[...all]/route.ts +78 -0
- package/dist/templates/app/api/cron/billing/lifecycle/route.ts +98 -0
- package/dist/templates/app/api/csp-report/route.ts +175 -0
- package/dist/templates/app/api/devtools/config/entities/route.ts +108 -0
- package/dist/templates/app/api/devtools/config/theme/route.ts +66 -0
- package/dist/templates/app/api/devtools/tests/[...path]/route.ts +130 -0
- package/dist/templates/app/api/devtools/tests/route.ts +134 -0
- package/dist/templates/app/api/health/route.ts +29 -0
- package/dist/templates/app/api/internal/user-metadata/route.ts +36 -0
- package/dist/templates/app/api/superadmin/subscriptions/route.ts +310 -0
- package/dist/templates/app/api/superadmin/teams/[teamId]/route.ts +286 -0
- package/dist/templates/app/api/superadmin/teams/route.ts +188 -0
- package/dist/templates/app/api/superadmin/users/[userId]/route.ts +540 -0
- package/dist/templates/app/api/superadmin/users/route.ts +323 -0
- package/dist/templates/app/api/user/delete-account/route.ts +55 -0
- package/dist/templates/app/api/user/plan-flags/route.ts +283 -0
- package/dist/templates/app/api/user/profile/route.ts +133 -0
- package/dist/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +210 -0
- package/dist/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +331 -0
- package/dist/templates/app/api/v1/[entity]/[id]/route.ts +35 -0
- package/dist/templates/app/api/v1/[entity]/docs.md +369 -0
- package/dist/templates/app/api/v1/[entity]/presets.ts +194 -0
- package/dist/templates/app/api/v1/[entity]/route.ts +31 -0
- package/dist/templates/app/api/v1/api-keys/[id]/route.ts +303 -0
- package/dist/templates/app/api/v1/api-keys/docs.md +101 -0
- package/dist/templates/app/api/v1/api-keys/presets.ts +31 -0
- package/dist/templates/app/api/v1/api-keys/route.ts +250 -0
- package/dist/templates/app/api/v1/auth/docs.md +184 -0
- package/dist/templates/app/api/v1/auth/presets.ts +44 -0
- package/dist/templates/app/api/v1/auth/signup-with-invite/route.ts +227 -0
- package/dist/templates/app/api/v1/billing/cancel/route.ts +206 -0
- package/dist/templates/app/api/v1/billing/change-plan/route.ts +97 -0
- package/dist/templates/app/api/v1/billing/check-action/route.ts +81 -0
- package/dist/templates/app/api/v1/billing/checkout/route.ts +124 -0
- package/dist/templates/app/api/v1/billing/docs.md +209 -0
- package/dist/templates/app/api/v1/billing/plans/route.ts +85 -0
- package/dist/templates/app/api/v1/billing/portal/route.ts +90 -0
- package/dist/templates/app/api/v1/billing/presets.ts +121 -0
- package/dist/templates/app/api/v1/billing/webhooks/stripe/route.ts +428 -0
- package/dist/templates/app/api/v1/blocks/[slug]/route.ts +29 -0
- package/dist/templates/app/api/v1/blocks/docs.md +173 -0
- package/dist/templates/app/api/v1/blocks/presets.ts +121 -0
- package/dist/templates/app/api/v1/blocks/route.ts +45 -0
- package/dist/templates/app/api/v1/blocks/validate/route.ts +45 -0
- package/dist/templates/app/api/v1/cron/docs.md +116 -0
- package/dist/templates/app/api/v1/cron/presets.ts +26 -0
- package/dist/templates/app/api/v1/cron/process/route.ts +108 -0
- package/dist/templates/app/api/v1/devtools/blocks/route.ts +82 -0
- package/dist/templates/app/api/v1/devtools/docs/route.ts +150 -0
- package/dist/templates/app/api/v1/devtools/docs.md +204 -0
- package/dist/templates/app/api/v1/devtools/features/route.ts +61 -0
- package/dist/templates/app/api/v1/devtools/flows/route.ts +61 -0
- package/dist/templates/app/api/v1/devtools/presets.ts +113 -0
- package/dist/templates/app/api/v1/devtools/scheduled-actions/route.ts +120 -0
- package/dist/templates/app/api/v1/devtools/testing/route.ts +82 -0
- package/dist/templates/app/api/v1/media/docs.md +117 -0
- package/dist/templates/app/api/v1/media/presets.ts +24 -0
- package/dist/templates/app/api/v1/media/upload/route.ts +150 -0
- package/dist/templates/app/api/v1/patterns/[id]/usages/route.ts +116 -0
- package/dist/templates/app/api/v1/plugin/[...path]/route.ts +373 -0
- package/dist/templates/app/api/v1/plugin/docs.md +79 -0
- package/dist/templates/app/api/v1/plugin/presets.ts +21 -0
- package/dist/templates/app/api/v1/plugin/route.ts +96 -0
- package/dist/templates/app/api/v1/post-categories/[id]/route.ts +255 -0
- package/dist/templates/app/api/v1/post-categories/docs.md +134 -0
- package/dist/templates/app/api/v1/post-categories/presets.ts +78 -0
- package/dist/templates/app/api/v1/post-categories/route.ts +119 -0
- package/dist/templates/app/api/v1/team-invitations/[token]/accept/route.ts +179 -0
- package/dist/templates/app/api/v1/team-invitations/[token]/decline/route.ts +120 -0
- package/dist/templates/app/api/v1/team-invitations/[token]/route.ts +89 -0
- package/dist/templates/app/api/v1/team-invitations/docs.md +88 -0
- package/dist/templates/app/api/v1/team-invitations/presets.ts +43 -0
- package/dist/templates/app/api/v1/team-invitations/route.ts +114 -0
- package/dist/templates/app/api/v1/teams/[teamId]/invitations/route.ts +171 -0
- package/dist/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +105 -0
- package/dist/templates/app/api/v1/teams/[teamId]/invoices/route.ts +125 -0
- package/dist/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +263 -0
- package/dist/templates/app/api/v1/teams/[teamId]/members/route.ts +358 -0
- package/dist/templates/app/api/v1/teams/[teamId]/route.ts +322 -0
- package/dist/templates/app/api/v1/teams/[teamId]/subscription/route.ts +50 -0
- package/dist/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +91 -0
- package/dist/templates/app/api/v1/teams/docs.md +320 -0
- package/dist/templates/app/api/v1/teams/presets.ts +178 -0
- package/dist/templates/app/api/v1/teams/route.ts +293 -0
- package/dist/templates/app/api/v1/teams/switch/route.ts +88 -0
- package/dist/templates/app/api/v1/theme/[...path]/route.ts +361 -0
- package/dist/templates/app/api/v1/theme/docs.md +74 -0
- package/dist/templates/app/api/v1/theme/presets.ts +21 -0
- package/dist/templates/app/api/v1/theme/route.ts +96 -0
- package/dist/templates/app/api/v1/users/[id]/meta/[key]/route.ts +363 -0
- package/dist/templates/app/api/v1/users/[id]/route.ts +302 -0
- package/dist/templates/app/api/v1/users/docs.md +93 -0
- package/dist/templates/app/api/v1/users/presets.ts +59 -0
- package/dist/templates/app/api/v1/users/route.ts +197 -0
- package/dist/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +117 -0
- package/dist/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +103 -0
- package/dist/templates/app/dashboard/(main)/[entity]/create/page.tsx +95 -0
- package/dist/templates/app/dashboard/(main)/[entity]/error.tsx +51 -0
- package/dist/templates/app/dashboard/(main)/[entity]/layout.tsx +113 -0
- package/dist/templates/app/dashboard/(main)/[entity]/loading.tsx +61 -0
- package/dist/templates/app/dashboard/(main)/[entity]/page.tsx +90 -0
- package/dist/templates/app/dashboard/(main)/layout.tsx +98 -0
- package/dist/templates/app/dashboard/(main)/loading.tsx +5 -0
- package/dist/templates/app/dashboard/(main)/page.tsx +201 -0
- package/dist/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +114 -0
- package/dist/templates/app/dashboard/(main)/patterns/[id]/page.tsx +20 -0
- package/dist/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +171 -0
- package/dist/templates/app/dashboard/(main)/patterns/create/page.tsx +86 -0
- package/dist/templates/app/dashboard/(main)/patterns/page.tsx +444 -0
- package/dist/templates/app/dashboard/features/analytics/page.tsx +35 -0
- package/dist/templates/app/dashboard/features/automation/page.tsx +35 -0
- package/dist/templates/app/dashboard/features/layout.tsx +13 -0
- package/dist/templates/app/dashboard/features/loading.tsx +5 -0
- package/dist/templates/app/dashboard/features/webhooks/page.tsx +35 -0
- package/dist/templates/app/dashboard/layout.tsx +86 -0
- package/dist/templates/app/dashboard/permission-denied/page.tsx +29 -0
- package/dist/templates/app/dashboard/settings/api-keys/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/api-keys/page.tsx +513 -0
- package/dist/templates/app/dashboard/settings/billing/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/billing/page.tsx +284 -0
- package/dist/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +222 -0
- package/dist/templates/app/dashboard/settings/invoices/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/invoices/page.tsx +82 -0
- package/dist/templates/app/dashboard/settings/layout.tsx +151 -0
- package/dist/templates/app/dashboard/settings/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/notifications/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/notifications/page.tsx +462 -0
- package/dist/templates/app/dashboard/settings/page.tsx +92 -0
- package/dist/templates/app/dashboard/settings/password/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/password/page.tsx +306 -0
- package/dist/templates/app/dashboard/settings/plans/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/plans/page.tsx +40 -0
- package/dist/templates/app/dashboard/settings/profile/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/profile/page.tsx +686 -0
- package/dist/templates/app/dashboard/settings/security/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/security/page.tsx +505 -0
- package/dist/templates/app/dashboard/settings/teams/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/teams/page.tsx +272 -0
- package/dist/templates/app/dashboard/settings/teams/permissions/page.tsx +92 -0
- package/dist/templates/app/devtools/blocks/[slug]/page.tsx +39 -0
- package/dist/templates/app/devtools/blocks/page.tsx +31 -0
- package/dist/templates/app/devtools/config/page.tsx +31 -0
- package/dist/templates/app/devtools/features/page.tsx +31 -0
- package/dist/templates/app/devtools/flows/page.tsx +31 -0
- package/dist/templates/app/devtools/layout.tsx +58 -0
- package/dist/templates/app/devtools/page.tsx +121 -0
- package/dist/templates/app/devtools/scheduled-actions/page.tsx +157 -0
- package/dist/templates/app/devtools/style/page.tsx +330 -0
- package/dist/templates/app/devtools/tags/page.tsx +31 -0
- package/dist/templates/app/devtools/tests/[[...path]]/page.tsx +47 -0
- package/dist/templates/app/favicon.ico +0 -0
- package/dist/templates/app/globals.css +12 -0
- package/dist/templates/app/layout.tsx +96 -0
- package/dist/templates/app/public/page.tsx +30 -0
- package/dist/templates/app/superadmin/docs/[section]/[page]/page.tsx +92 -0
- package/dist/templates/app/superadmin/docs/page.tsx +75 -0
- package/dist/templates/app/superadmin/layout.tsx +67 -0
- package/dist/templates/app/superadmin/page.tsx +149 -0
- package/dist/templates/app/superadmin/subscriptions/page.tsx +655 -0
- package/dist/templates/app/superadmin/team-roles/page.tsx +493 -0
- package/dist/templates/app/superadmin/teams/[teamId]/page.tsx +687 -0
- package/dist/templates/app/superadmin/teams/page.tsx +302 -0
- package/dist/templates/app/superadmin/users/[userId]/page.tsx +548 -0
- package/dist/templates/app/superadmin/users/page.tsx +528 -0
- package/package.json +15 -15
- package/scripts/build/docs-registry.mjs +0 -0
- package/scripts/create-theme.mjs +0 -0
- package/scripts/deploy/release-version.mjs +0 -0
- package/scripts/deploy/vercel-deploy.mjs +0 -0
- package/scripts/dev/watch-plugins.mjs +0 -0
- package/scripts/maintenance/update-core.mjs +0 -0
- package/scripts/setup/npm-postinstall.mjs +0 -0
- package/scripts/setup/setup-claude.mjs +0 -0
- package/scripts/validation/check-imports.sh +0 -0
- package/templates/app/(auth)/forgot-password/page.tsx +216 -0
- package/templates/app/(auth)/layout.tsx +51 -0
- package/templates/app/(auth)/login/page.tsx +21 -0
- package/templates/app/(auth)/reset-password/page.tsx +212 -0
- package/templates/app/(auth)/signup/page.tsx +21 -0
- package/templates/app/(auth)/verify-email/page.tsx +190 -0
- package/templates/app/(public)/[...slug]/page.tsx +378 -0
- package/templates/app/(public)/docs/[section]/[page]/page.tsx +90 -0
- package/templates/app/(public)/docs/layout.tsx +25 -0
- package/templates/app/(public)/docs/page.tsx +81 -0
- package/templates/app/(public)/layout.tsx +41 -0
- package/templates/app/(public)/page.tsx +19 -0
- package/templates/app/403/page.tsx +89 -0
- package/templates/app/api/auth/[...all]/route.ts +78 -0
- package/templates/app/api/cron/billing/lifecycle/route.ts +98 -0
- package/templates/app/api/csp-report/route.ts +175 -0
- package/templates/app/api/devtools/config/entities/route.ts +108 -0
- package/templates/app/api/devtools/config/theme/route.ts +66 -0
- package/templates/app/api/devtools/tests/[...path]/route.ts +130 -0
- package/templates/app/api/devtools/tests/route.ts +134 -0
- package/templates/app/api/health/route.ts +29 -0
- package/templates/app/api/internal/user-metadata/route.ts +36 -0
- package/templates/app/api/superadmin/subscriptions/route.ts +310 -0
- package/templates/app/api/superadmin/teams/[teamId]/route.ts +286 -0
- package/templates/app/api/superadmin/teams/route.ts +188 -0
- package/templates/app/api/superadmin/users/[userId]/route.ts +540 -0
- package/templates/app/api/superadmin/users/route.ts +323 -0
- package/templates/app/api/user/delete-account/route.ts +55 -0
- package/templates/app/api/user/plan-flags/route.ts +283 -0
- package/templates/app/api/user/profile/route.ts +133 -0
- package/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +210 -0
- package/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +331 -0
- package/templates/app/api/v1/[entity]/[id]/route.ts +35 -0
- package/templates/app/api/v1/[entity]/docs.md +369 -0
- package/templates/app/api/v1/[entity]/presets.ts +194 -0
- package/templates/app/api/v1/[entity]/route.ts +31 -0
- package/templates/app/api/v1/api-keys/[id]/route.ts +303 -0
- package/templates/app/api/v1/api-keys/docs.md +101 -0
- package/templates/app/api/v1/api-keys/presets.ts +31 -0
- package/templates/app/api/v1/api-keys/route.ts +250 -0
- package/templates/app/api/v1/auth/docs.md +184 -0
- package/templates/app/api/v1/auth/presets.ts +44 -0
- package/templates/app/api/v1/auth/signup-with-invite/route.ts +227 -0
- package/templates/app/api/v1/billing/cancel/route.ts +206 -0
- package/templates/app/api/v1/billing/change-plan/route.ts +97 -0
- package/templates/app/api/v1/billing/check-action/route.ts +81 -0
- package/templates/app/api/v1/billing/checkout/route.ts +124 -0
- package/templates/app/api/v1/billing/docs.md +209 -0
- package/templates/app/api/v1/billing/plans/route.ts +85 -0
- package/templates/app/api/v1/billing/portal/route.ts +90 -0
- package/templates/app/api/v1/billing/presets.ts +121 -0
- package/templates/app/api/v1/billing/webhooks/stripe/route.ts +428 -0
- package/templates/app/api/v1/blocks/[slug]/route.ts +29 -0
- package/templates/app/api/v1/blocks/docs.md +173 -0
- package/templates/app/api/v1/blocks/presets.ts +121 -0
- package/templates/app/api/v1/blocks/route.ts +45 -0
- package/templates/app/api/v1/blocks/validate/route.ts +45 -0
- package/templates/app/api/v1/cron/docs.md +116 -0
- package/templates/app/api/v1/cron/presets.ts +26 -0
- package/templates/app/api/v1/cron/process/route.ts +108 -0
- package/templates/app/api/v1/devtools/blocks/route.ts +82 -0
- package/templates/app/api/v1/devtools/docs/route.ts +150 -0
- package/templates/app/api/v1/devtools/docs.md +204 -0
- package/templates/app/api/v1/devtools/features/route.ts +61 -0
- package/templates/app/api/v1/devtools/flows/route.ts +61 -0
- package/templates/app/api/v1/devtools/presets.ts +113 -0
- package/templates/app/api/v1/devtools/scheduled-actions/route.ts +120 -0
- package/templates/app/api/v1/devtools/testing/route.ts +82 -0
- package/templates/app/api/v1/media/docs.md +117 -0
- package/templates/app/api/v1/media/presets.ts +24 -0
- package/templates/app/api/v1/media/upload/route.ts +150 -0
- package/templates/app/api/v1/patterns/[id]/usages/route.ts +116 -0
- package/templates/app/api/v1/plugin/[...path]/route.ts +373 -0
- package/templates/app/api/v1/plugin/docs.md +79 -0
- package/templates/app/api/v1/plugin/presets.ts +21 -0
- package/templates/app/api/v1/plugin/route.ts +96 -0
- package/templates/app/api/v1/post-categories/[id]/route.ts +255 -0
- package/templates/app/api/v1/post-categories/docs.md +134 -0
- package/templates/app/api/v1/post-categories/presets.ts +78 -0
- package/templates/app/api/v1/post-categories/route.ts +119 -0
- package/templates/app/api/v1/team-invitations/[token]/accept/route.ts +179 -0
- package/templates/app/api/v1/team-invitations/[token]/decline/route.ts +120 -0
- package/templates/app/api/v1/team-invitations/[token]/route.ts +89 -0
- package/templates/app/api/v1/team-invitations/docs.md +88 -0
- package/templates/app/api/v1/team-invitations/presets.ts +43 -0
- package/templates/app/api/v1/team-invitations/route.ts +114 -0
- package/templates/app/api/v1/teams/[teamId]/invitations/route.ts +171 -0
- package/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +105 -0
- package/templates/app/api/v1/teams/[teamId]/invoices/route.ts +125 -0
- package/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +263 -0
- package/templates/app/api/v1/teams/[teamId]/members/route.ts +358 -0
- package/templates/app/api/v1/teams/[teamId]/route.ts +322 -0
- package/templates/app/api/v1/teams/[teamId]/subscription/route.ts +50 -0
- package/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +91 -0
- package/templates/app/api/v1/teams/docs.md +320 -0
- package/templates/app/api/v1/teams/presets.ts +178 -0
- package/templates/app/api/v1/teams/route.ts +293 -0
- package/templates/app/api/v1/teams/switch/route.ts +88 -0
- package/templates/app/api/v1/theme/[...path]/route.ts +361 -0
- package/templates/app/api/v1/theme/docs.md +74 -0
- package/templates/app/api/v1/theme/presets.ts +21 -0
- package/templates/app/api/v1/theme/route.ts +96 -0
- package/templates/app/api/v1/users/[id]/meta/[key]/route.ts +363 -0
- package/templates/app/api/v1/users/[id]/route.ts +302 -0
- package/templates/app/api/v1/users/docs.md +93 -0
- package/templates/app/api/v1/users/presets.ts +59 -0
- package/templates/app/api/v1/users/route.ts +197 -0
- package/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +117 -0
- package/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +103 -0
- package/templates/app/dashboard/(main)/[entity]/create/page.tsx +95 -0
- package/templates/app/dashboard/(main)/[entity]/error.tsx +51 -0
- package/templates/app/dashboard/(main)/[entity]/layout.tsx +113 -0
- package/templates/app/dashboard/(main)/[entity]/loading.tsx +61 -0
- package/templates/app/dashboard/(main)/[entity]/page.tsx +90 -0
- package/templates/app/dashboard/(main)/layout.tsx +98 -0
- package/templates/app/dashboard/(main)/loading.tsx +5 -0
- package/templates/app/dashboard/(main)/page.tsx +201 -0
- package/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +114 -0
- package/templates/app/dashboard/(main)/patterns/[id]/page.tsx +20 -0
- package/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +171 -0
- package/templates/app/dashboard/(main)/patterns/create/page.tsx +86 -0
- package/templates/app/dashboard/(main)/patterns/page.tsx +444 -0
- package/templates/app/dashboard/features/analytics/page.tsx +35 -0
- package/templates/app/dashboard/features/automation/page.tsx +35 -0
- package/templates/app/dashboard/features/layout.tsx +13 -0
- package/templates/app/dashboard/features/loading.tsx +5 -0
- package/templates/app/dashboard/features/webhooks/page.tsx +35 -0
- package/templates/app/dashboard/layout.tsx +86 -0
- package/templates/app/dashboard/permission-denied/page.tsx +29 -0
- package/templates/app/dashboard/settings/api-keys/loading.tsx +5 -0
- package/templates/app/dashboard/settings/api-keys/page.tsx +513 -0
- package/templates/app/dashboard/settings/billing/loading.tsx +5 -0
- package/templates/app/dashboard/settings/billing/page.tsx +284 -0
- package/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +222 -0
- package/templates/app/dashboard/settings/invoices/loading.tsx +5 -0
- package/templates/app/dashboard/settings/invoices/page.tsx +82 -0
- package/templates/app/dashboard/settings/layout.tsx +151 -0
- package/templates/app/dashboard/settings/loading.tsx +5 -0
- package/templates/app/dashboard/settings/notifications/loading.tsx +5 -0
- package/templates/app/dashboard/settings/notifications/page.tsx +462 -0
- package/templates/app/dashboard/settings/page.tsx +92 -0
- package/templates/app/dashboard/settings/password/loading.tsx +5 -0
- package/templates/app/dashboard/settings/password/page.tsx +306 -0
- package/templates/app/dashboard/settings/plans/loading.tsx +5 -0
- package/templates/app/dashboard/settings/plans/page.tsx +40 -0
- package/templates/app/dashboard/settings/profile/loading.tsx +5 -0
- package/templates/app/dashboard/settings/profile/page.tsx +686 -0
- package/templates/app/dashboard/settings/security/loading.tsx +5 -0
- package/templates/app/dashboard/settings/security/page.tsx +505 -0
- package/templates/app/dashboard/settings/teams/loading.tsx +5 -0
- package/templates/app/dashboard/settings/teams/page.tsx +272 -0
- package/templates/app/dashboard/settings/teams/permissions/page.tsx +92 -0
- package/templates/app/devtools/blocks/[slug]/page.tsx +39 -0
- package/templates/app/devtools/blocks/page.tsx +31 -0
- package/templates/app/devtools/config/page.tsx +31 -0
- package/templates/app/devtools/features/page.tsx +31 -0
- package/templates/app/devtools/flows/page.tsx +31 -0
- package/templates/app/devtools/layout.tsx +58 -0
- package/templates/app/devtools/page.tsx +121 -0
- package/templates/app/devtools/scheduled-actions/page.tsx +157 -0
- package/templates/app/devtools/style/page.tsx +330 -0
- package/templates/app/devtools/tags/page.tsx +31 -0
- package/templates/app/devtools/tests/[[...path]]/page.tsx +47 -0
- package/templates/app/favicon.ico +0 -0
- package/templates/app/globals.css +12 -0
- package/templates/app/layout.tsx +96 -0
- package/templates/app/public/page.tsx +30 -0
- package/templates/app/superadmin/docs/[section]/[page]/page.tsx +92 -0
- package/templates/app/superadmin/docs/page.tsx +75 -0
- package/templates/app/superadmin/layout.tsx +67 -0
- package/templates/app/superadmin/page.tsx +149 -0
- package/templates/app/superadmin/subscriptions/page.tsx +655 -0
- package/templates/app/superadmin/team-roles/page.tsx +493 -0
- package/templates/app/superadmin/teams/[teamId]/page.tsx +687 -0
- package/templates/app/superadmin/teams/page.tsx +302 -0
- package/templates/app/superadmin/users/[userId]/page.tsx +548 -0
- 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)
|