create-nextjs-cms 0.9.30 → 0.9.31

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 (143) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +71 -71
  3. package/dist/helpers/utils.js +16 -16
  4. package/dist/lib/section-creators.js +166 -166
  5. package/package.json +2 -2
  6. package/templates/default/.eslintrc.json +5 -5
  7. package/templates/default/.prettierignore +7 -7
  8. package/templates/default/.prettierrc.json +27 -27
  9. package/templates/default/CHANGELOG.md +140 -140
  10. package/templates/default/_gitignore +57 -57
  11. package/templates/default/app/(auth)/auth/login/LoginPage.tsx +192 -192
  12. package/templates/default/app/(auth)/auth/login/page.tsx +11 -11
  13. package/templates/default/app/(auth)/auth-language-provider.tsx +34 -34
  14. package/templates/default/app/(auth)/layout.tsx +81 -81
  15. package/templates/default/app/(rootLayout)/(plugins)/[...slug]/plugin-server-registry.ts +10 -6
  16. package/templates/default/app/(rootLayout)/admins/page.tsx +10 -10
  17. package/templates/default/app/(rootLayout)/browse/[section]/[page]/page.tsx +22 -22
  18. package/templates/default/app/(rootLayout)/categorized/[section]/page.tsx +15 -15
  19. package/templates/default/app/(rootLayout)/dashboard/page.tsx +70 -70
  20. package/templates/default/app/(rootLayout)/edit/[section]/[itemId]/page.tsx +20 -20
  21. package/templates/default/app/(rootLayout)/layout.tsx +81 -81
  22. package/templates/default/app/(rootLayout)/loading.tsx +10 -10
  23. package/templates/default/app/(rootLayout)/log/page.tsx +7 -7
  24. package/templates/default/app/(rootLayout)/new/[section]/page.tsx +15 -15
  25. package/templates/default/app/(rootLayout)/section/[section]/page.tsx +19 -19
  26. package/templates/default/app/(rootLayout)/settings/page.tsx +13 -13
  27. package/templates/default/app/api/auth/csrf/route.ts +25 -25
  28. package/templates/default/app/api/auth/refresh/route.ts +10 -10
  29. package/templates/default/app/api/auth/route.ts +49 -49
  30. package/templates/default/app/api/auth/session/route.ts +20 -20
  31. package/templates/default/app/api/document/route.ts +165 -165
  32. package/templates/default/app/api/editor/photo/route.ts +49 -49
  33. package/templates/default/app/api/photo/route.ts +27 -27
  34. package/templates/default/app/api/submit/section/item/[slug]/route.ts +95 -95
  35. package/templates/default/app/api/submit/section/item/route.ts +56 -56
  36. package/templates/default/app/api/submit/section/simple/route.ts +86 -86
  37. package/templates/default/app/api/video/route.ts +174 -174
  38. package/templates/default/app/globals.css +236 -236
  39. package/templates/default/cms.config.ts +56 -56
  40. package/templates/default/components/admin/admin-card.tsx +165 -165
  41. package/templates/default/components/admin/admin-edit-page.tsx +124 -124
  42. package/templates/default/components/admin/admin-privilege-card.tsx +184 -184
  43. package/templates/default/components/admin/new-admin-form.tsx +172 -172
  44. package/templates/default/components/container-box.tsx +24 -24
  45. package/templates/default/components/dnd-kit/draggable.tsx +21 -21
  46. package/templates/default/components/dnd-kit/droppable.tsx +20 -20
  47. package/templates/default/components/dnd-kit/sortable-item.tsx +18 -18
  48. package/templates/default/components/feedback/error-component.tsx +16 -16
  49. package/templates/default/components/feedback/info-card.tsx +93 -93
  50. package/templates/default/components/feedback/loading-spinners.tsx +67 -67
  51. package/templates/default/components/feedback/modal.tsx +166 -166
  52. package/templates/default/components/feedback/progress-bar.tsx +48 -48
  53. package/templates/default/components/feedback/tooltip-component.tsx +27 -27
  54. package/templates/default/components/form/form-input-element.tsx +70 -70
  55. package/templates/default/components/form/helpers/_section-hot-reload.js +1 -1
  56. package/templates/default/components/form/helpers/util.ts +17 -17
  57. package/templates/default/components/form/inputs/checkbox-form-input.tsx +46 -46
  58. package/templates/default/components/form/inputs/color-form-input.tsx +44 -44
  59. package/templates/default/components/form/inputs/date-form-input.tsx +93 -93
  60. package/templates/default/components/form/inputs/map-form-input.tsx +141 -141
  61. package/templates/default/components/form/inputs/multiple-select-form-input.tsx +85 -85
  62. package/templates/default/components/form/inputs/number-form-input.tsx +43 -43
  63. package/templates/default/components/form/inputs/password-form-input.tsx +47 -47
  64. package/templates/default/components/form/inputs/photo-form-input.tsx +279 -279
  65. package/templates/default/components/form/inputs/rich-text-form-input.tsx +148 -148
  66. package/templates/default/components/form/inputs/select-form-input.tsx +159 -159
  67. package/templates/default/components/form/inputs/slug-form-input.tsx +131 -131
  68. package/templates/default/components/form/inputs/tags-form-input.tsx +255 -255
  69. package/templates/default/components/form/inputs/text-form-input.tsx +61 -61
  70. package/templates/default/components/form/inputs/textarea-form-input.tsx +61 -61
  71. package/templates/default/components/layout/default-nav-items.tsx +3 -3
  72. package/templates/default/components/layout/layout.tsx +84 -84
  73. package/templates/default/components/layout/navbar.tsx +258 -258
  74. package/templates/default/components/layout/sidebar-dropdown-item.tsx +83 -83
  75. package/templates/default/components/layout/sidebar-item.tsx +24 -24
  76. package/templates/default/components/layout/sidebar.tsx +229 -229
  77. package/templates/default/components/layout/theme-provider.tsx +8 -8
  78. package/templates/default/components/layout/theme-toggle.tsx +39 -39
  79. package/templates/default/components/locale/locale-switcher.tsx +98 -98
  80. package/templates/default/components/media/dropzone.tsx +154 -154
  81. package/templates/default/components/media/protected-document.tsx +44 -44
  82. package/templates/default/components/media/protected-image.tsx +143 -143
  83. package/templates/default/components/media/protected-video.tsx +76 -76
  84. package/templates/default/components/multi-select.tsx +1150 -1150
  85. package/templates/default/components/pages/admins-page.tsx +43 -43
  86. package/templates/default/components/pages/browse-page.tsx +106 -106
  87. package/templates/default/components/pages/categorized-section-page.tsx +31 -31
  88. package/templates/default/components/pages/dashboard-page-alt.tsx +45 -45
  89. package/templates/default/components/pages/item-edit-page.tsx +267 -267
  90. package/templates/default/components/pages/log-page.tsx +107 -107
  91. package/templates/default/components/pages/new-page.tsx +183 -183
  92. package/templates/default/components/pages/section-page.tsx +203 -203
  93. package/templates/default/components/pages/settings-page.tsx +232 -232
  94. package/templates/default/components/pagination/pagination-buttons.tsx +147 -147
  95. package/templates/default/components/pagination/pagination.tsx +36 -36
  96. package/templates/default/components/sections/category-delete-confirm-page.tsx +130 -130
  97. package/templates/default/components/sections/category-section-select-input.tsx +139 -139
  98. package/templates/default/components/sections/conditional-fields.tsx +49 -49
  99. package/templates/default/components/sections/section-icon.tsx +8 -8
  100. package/templates/default/components/sections/section-item-card.tsx +143 -143
  101. package/templates/default/components/sections/section-item-status-badge.tsx +17 -17
  102. package/templates/default/components/sections/select-input-buttons.tsx +125 -125
  103. package/templates/default/components/select-box.tsx +98 -98
  104. package/templates/default/components/ui/accordion.tsx +53 -53
  105. package/templates/default/components/ui/alert-dialog.tsx +113 -113
  106. package/templates/default/components/ui/alert.tsx +47 -47
  107. package/templates/default/components/ui/badge.tsx +38 -38
  108. package/templates/default/components/ui/card.tsx +43 -43
  109. package/templates/default/components/ui/command.tsx +137 -137
  110. package/templates/default/components/ui/custom-alert-dialog.tsx +113 -113
  111. package/templates/default/components/ui/custom-dialog.tsx +123 -123
  112. package/templates/default/components/ui/dialog.tsx +123 -123
  113. package/templates/default/components/ui/direction.tsx +22 -22
  114. package/templates/default/components/ui/dropdown-menu.tsx +182 -182
  115. package/templates/default/components/ui/input-group.tsx +54 -54
  116. package/templates/default/components/ui/input.tsx +22 -22
  117. package/templates/default/components/ui/label.tsx +19 -19
  118. package/templates/default/components/ui/popover.tsx +42 -42
  119. package/templates/default/components/ui/progress.tsx +31 -31
  120. package/templates/default/components/ui/scroll-area.tsx +42 -42
  121. package/templates/default/components/ui/select.tsx +165 -165
  122. package/templates/default/components/ui/separator.tsx +28 -28
  123. package/templates/default/components/ui/sheet.tsx +103 -103
  124. package/templates/default/components/ui/spinner.tsx +16 -16
  125. package/templates/default/components/ui/switch.tsx +29 -29
  126. package/templates/default/components/ui/table.tsx +83 -83
  127. package/templates/default/components/ui/tabs.tsx +55 -55
  128. package/templates/default/components/ui/toast.tsx +113 -113
  129. package/templates/default/components/ui/toaster.tsx +35 -35
  130. package/templates/default/components/ui/tooltip.tsx +30 -30
  131. package/templates/default/components/ui/use-toast.ts +187 -187
  132. package/templates/default/drizzle.config.ts +4 -4
  133. package/templates/default/dynamic-schemas/schema.ts +225 -75
  134. package/templates/default/env/env.ts +46 -46
  135. package/templates/default/envConfig.ts +4 -4
  136. package/templates/default/lib/postinstall.js +14 -14
  137. package/templates/default/lib/utils.ts +6 -6
  138. package/templates/default/next-env.d.ts +6 -6
  139. package/templates/default/next.config.ts +24 -24
  140. package/templates/default/package.json +1 -1
  141. package/templates/default/postcss.config.mjs +6 -6
  142. package/templates/default/proxy.ts +32 -32
  143. package/templates/default/tsconfig.json +48 -48
@@ -1,3 +1,4 @@
1
+ import dynamic from 'next/dynamic'
1
2
  import type { ComponentType } from 'react'
2
3
 
3
4
  export const pluginNamesMap: Record<string, string> = {
@@ -12,11 +13,14 @@ export const getPluginServerComponent = async (
12
13
  pluginName: string,
13
14
  componentName: string | undefined,
14
15
  ): Promise<ComponentType | null> => {
15
- const modulePath = pluginNamesMap[pluginName] ?? pluginNamesMap.blank
16
+ const modulePath = pluginNamesMap[pluginName]
16
17
  if (!modulePath) return null
17
-
18
- const mod = (await import(modulePath)) as Record<string, unknown>
19
- const component = (componentName ? mod[componentName] : mod.default) as ComponentType | undefined
20
-
21
- return component ?? null
18
+ const Component = dynamic(
19
+ () =>
20
+ import(modulePath ?? 'nextjs-cms/plugins/blank-component').then(
21
+ (mod) => (componentName ? mod[componentName] : mod.default) as ComponentType,
22
+ ),
23
+ { ssr: true },
24
+ )
25
+ return Component
22
26
  }
@@ -1,10 +1,10 @@
1
- import AdminsPage from '@/components/pages/admins-page'
2
- import { api, HydrateClient } from '@/app/_trpc/server'
3
- export default async function Page() {
4
- await api.admins.list.prefetch()
5
- return (
6
- <HydrateClient>
7
- <AdminsPage />
8
- </HydrateClient>
9
- )
10
- }
1
+ import AdminsPage from '@/components/pages/admins-page'
2
+ import { api, HydrateClient } from '@/app/_trpc/server'
3
+ export default async function Page() {
4
+ await api.admins.list.prefetch()
5
+ return (
6
+ <HydrateClient>
7
+ <AdminsPage />
8
+ </HydrateClient>
9
+ )
10
+ }
@@ -1,22 +1,22 @@
1
- import BrowsePage from '@/components/pages/browse-page'
2
- import { api, HydrateClient } from '@/app/_trpc/server'
3
- type Params = Promise<{ section: string; page: string }>
4
- type SearchParams = Promise<{ q: string | undefined }>
5
- export default async function Page(props: { params: Params; searchParams: SearchParams }) {
6
- const params = await props.params
7
- const searchParams = await props.searchParams
8
- const { q } = searchParams
9
- const { section, page } = params
10
-
11
- await api.hasItemsSections.listItems.prefetch({
12
- sectionName: section,
13
- page: parseInt(page) ?? 1,
14
- q: q ?? '',
15
- })
16
-
17
- return (
18
- <HydrateClient>
19
- <BrowsePage page={params.page} section={params.section} />
20
- </HydrateClient>
21
- )
22
- }
1
+ import BrowsePage from '@/components/pages/browse-page'
2
+ import { api, HydrateClient } from '@/app/_trpc/server'
3
+ type Params = Promise<{ section: string; page: string }>
4
+ type SearchParams = Promise<{ q: string | undefined }>
5
+ export default async function Page(props: { params: Params; searchParams: SearchParams }) {
6
+ const params = await props.params
7
+ const searchParams = await props.searchParams
8
+ const { q } = searchParams
9
+ const { section, page } = params
10
+
11
+ await api.hasItemsSections.listItems.prefetch({
12
+ sectionName: section,
13
+ page: parseInt(page) ?? 1,
14
+ q: q ?? '',
15
+ })
16
+
17
+ return (
18
+ <HydrateClient>
19
+ <BrowsePage page={params.page} section={params.section} />
20
+ </HydrateClient>
21
+ )
22
+ }
@@ -1,15 +1,15 @@
1
- import CategorizedSectionPage from '@/components/pages/categorized-section-page'
2
- import { api, HydrateClient } from '@/app/_trpc/server'
3
- type Params = Promise<{ section: string }>
4
-
5
- export default async function Page(props: { params: Params }) {
6
- const params = await props.params
7
- await api.categorySections.get.prefetch({
8
- sectionName: params.section,
9
- })
10
- return (
11
- <HydrateClient>
12
- <CategorizedSectionPage section={params.section} />
13
- </HydrateClient>
14
- )
15
- }
1
+ import CategorizedSectionPage from '@/components/pages/categorized-section-page'
2
+ import { api, HydrateClient } from '@/app/_trpc/server'
3
+ type Params = Promise<{ section: string }>
4
+
5
+ export default async function Page(props: { params: Params }) {
6
+ const params = await props.params
7
+ await api.categorySections.get.prefetch({
8
+ sectionName: params.section,
9
+ })
10
+ return (
11
+ <HydrateClient>
12
+ <CategorizedSectionPage section={params.section} />
13
+ </HydrateClient>
14
+ )
15
+ }
@@ -1,70 +1,70 @@
1
- import { getDashboardOverride, runRoutePrefetches } from 'nextjs-cms/plugins/server'
2
- import { api, HydrateClient } from '@/app/_trpc/server'
3
- import { getPluginServerComponent } from '../(plugins)/[...slug]/plugin-server-registry'
4
-
5
- export const dynamic = 'force-dynamic'
6
-
7
- function DefaultDashboard() {
8
- return (
9
- <div className='w-full'>
10
- <div className='bg-linear-to-r from-amber-200 via-orange-200 to-rose-200 p-8 text-foreground dark:from-amber-900 dark:via-orange-900 dark:to-rose-900'>
11
- <h1 className='text-3xl font-extrabold tracking-tight'>Welcome to Mission Control</h1>
12
- <p className='mt-2 text-base text-muted-foreground'>
13
- Your CMS is ready. Chart a course, publish boldly, and keep the lights green.
14
- </p>
15
- </div>
16
-
17
- <div className='space-y-6 p-6'>
18
- <section className='rounded-lg border bg-card p-6 text-card-foreground shadow-sm'>
19
- <h2 className='text-xl font-semibold'>Today&apos;s focus</h2>
20
- <p className='mt-2 text-sm text-muted-foreground'>
21
- Start with one small win: verify a section, ship a quick update, or polish a headline.
22
- </p>
23
- </section>
24
-
25
- <section className='grid gap-4 md:grid-cols-3'>
26
- <div className='rounded-lg border bg-card p-4 text-card-foreground shadow-sm'>
27
- <div className='text-sm font-semibold'>Ship with confidence</div>
28
- <div className='mt-2 text-sm text-muted-foreground'>
29
- Keep changes tight and reversible. Small releases, fast feedback.
30
- </div>
31
- </div>
32
- <div className='rounded-lg border bg-card p-4 text-card-foreground shadow-sm'>
33
- <div className='text-sm font-semibold'>Stay organized</div>
34
- <div className='mt-2 text-sm text-muted-foreground'>
35
- Group your content by intent, not just by type.
36
- </div>
37
- </div>
38
- <div className='rounded-lg border bg-card p-4 text-card-foreground shadow-sm'>
39
- <div className='text-sm font-semibold'>Delight users</div>
40
- <div className='mt-2 text-sm text-muted-foreground'>
41
- A clean headline and a single strong image can do the heavy lifting.
42
- </div>
43
- </div>
44
- </section>
45
- </div>
46
- </div>
47
- )
48
- }
49
-
50
- export default async function DashboardPage() {
51
- // Check for dashboard override configuration
52
- const override = await getDashboardOverride()
53
- if (!override) {
54
- return <DefaultDashboard />
55
- }
56
-
57
- // Render the override plugin component (no privilege check for dashboard override)
58
- const PluginComponent = await getPluginServerComponent(override.pluginName, override.component)
59
- if (!PluginComponent) {
60
- return <DefaultDashboard />
61
- }
62
-
63
- await runRoutePrefetches(api, override)
64
-
65
- return (
66
- <HydrateClient>
67
- <PluginComponent />
68
- </HydrateClient>
69
- )
70
- }
1
+ import { getDashboardOverride, runRoutePrefetches } from 'nextjs-cms/plugins/server'
2
+ import { api, HydrateClient } from '@/app/_trpc/server'
3
+ import { getPluginServerComponent } from '../(plugins)/[...slug]/plugin-server-registry'
4
+
5
+ export const dynamic = 'force-dynamic'
6
+
7
+ function DefaultDashboard() {
8
+ return (
9
+ <div className='w-full'>
10
+ <div className='bg-linear-to-r from-amber-200 via-orange-200 to-rose-200 p-8 text-foreground dark:from-amber-900 dark:via-orange-900 dark:to-rose-900'>
11
+ <h1 className='text-3xl font-extrabold tracking-tight'>Welcome to Mission Control</h1>
12
+ <p className='mt-2 text-base text-muted-foreground'>
13
+ Your CMS is ready. Chart a course, publish boldly, and keep the lights green.
14
+ </p>
15
+ </div>
16
+
17
+ <div className='space-y-6 p-6'>
18
+ <section className='rounded-lg border bg-card p-6 text-card-foreground shadow-sm'>
19
+ <h2 className='text-xl font-semibold'>Today&apos;s focus</h2>
20
+ <p className='mt-2 text-sm text-muted-foreground'>
21
+ Start with one small win: verify a section, ship a quick update, or polish a headline.
22
+ </p>
23
+ </section>
24
+
25
+ <section className='grid gap-4 md:grid-cols-3'>
26
+ <div className='rounded-lg border bg-card p-4 text-card-foreground shadow-sm'>
27
+ <div className='text-sm font-semibold'>Ship with confidence</div>
28
+ <div className='mt-2 text-sm text-muted-foreground'>
29
+ Keep changes tight and reversible. Small releases, fast feedback.
30
+ </div>
31
+ </div>
32
+ <div className='rounded-lg border bg-card p-4 text-card-foreground shadow-sm'>
33
+ <div className='text-sm font-semibold'>Stay organized</div>
34
+ <div className='mt-2 text-sm text-muted-foreground'>
35
+ Group your content by intent, not just by type.
36
+ </div>
37
+ </div>
38
+ <div className='rounded-lg border bg-card p-4 text-card-foreground shadow-sm'>
39
+ <div className='text-sm font-semibold'>Delight users</div>
40
+ <div className='mt-2 text-sm text-muted-foreground'>
41
+ A clean headline and a single strong image can do the heavy lifting.
42
+ </div>
43
+ </div>
44
+ </section>
45
+ </div>
46
+ </div>
47
+ )
48
+ }
49
+
50
+ export default async function DashboardPage() {
51
+ // Check for dashboard override configuration
52
+ const override = await getDashboardOverride()
53
+ if (!override) {
54
+ return <DefaultDashboard />
55
+ }
56
+
57
+ // Render the override plugin component (no privilege check for dashboard override)
58
+ const PluginComponent = await getPluginServerComponent(override.pluginName, override.component)
59
+ if (!PluginComponent) {
60
+ return <DefaultDashboard />
61
+ }
62
+
63
+ await runRoutePrefetches(api, override)
64
+
65
+ return (
66
+ <HydrateClient>
67
+ <PluginComponent />
68
+ </HydrateClient>
69
+ )
70
+ }
@@ -1,20 +1,20 @@
1
- import ItemEditPage from '@/components/pages/item-edit-page'
2
- import { api, HydrateClient } from '@/app/_trpc/server'
3
- type Params = Promise<{ section: string; itemId: string }>
4
- type SearchParams = Promise<{ locale?: string }>
5
-
6
- export default async function Page(props: { params: Params; searchParams: SearchParams }) {
7
- const params = await props.params
8
- const searchParams = await props.searchParams
9
- await api.hasItemsSections.editItem.prefetch({
10
- sectionName: params.section,
11
- sectionItemId: params.itemId,
12
- locale: searchParams.locale,
13
- })
14
-
15
- return (
16
- <HydrateClient>
17
- <ItemEditPage sectionType='has_items' section={params.section} itemId={params.itemId} />
18
- </HydrateClient>
19
- )
20
- }
1
+ import ItemEditPage from '@/components/pages/item-edit-page'
2
+ import { api, HydrateClient } from '@/app/_trpc/server'
3
+ type Params = Promise<{ section: string; itemId: string }>
4
+ type SearchParams = Promise<{ locale?: string }>
5
+
6
+ export default async function Page(props: { params: Params; searchParams: SearchParams }) {
7
+ const params = await props.params
8
+ const searchParams = await props.searchParams
9
+ await api.hasItemsSections.editItem.prefetch({
10
+ sectionName: params.section,
11
+ sectionItemId: params.itemId,
12
+ locale: searchParams.locale,
13
+ })
14
+
15
+ return (
16
+ <HydrateClient>
17
+ <ItemEditPage sectionType='has_items' section={params.section} itemId={params.itemId} />
18
+ </HydrateClient>
19
+ )
20
+ }
@@ -1,81 +1,81 @@
1
- import { Inter, Cairo } from 'next/font/google'
2
- import '../globals.css'
3
- import type { Metadata } from 'next'
4
- import { cn } from '@/lib/utils'
5
- import { ThemeProvider } from '@/components/layout/theme-provider'
6
- import Providers from '@/app/providers'
7
- import auth from 'nextjs-cms/auth'
8
- import { getCMSConfig } from 'nextjs-cms/core'
9
- import { I18nProviderClient } from 'nextjs-cms/translations/client'
10
- import { resolveLanguage, RTL_LANGUAGES } from 'nextjs-cms/translations'
11
- import { getClientDictionaries } from 'nextjs-cms/translations/server'
12
- import { api, HydrateClient } from '@/app/_trpc/server'
13
- import Layout from '@/components/layout/layout'
14
- import { redirect } from 'next/navigation'
15
- import { DirectionProvider } from "@/components/ui/direction"
16
-
17
- const inter = Inter({
18
- subsets: ['latin'],
19
- variable: '--font-sans',
20
- })
21
-
22
- const cairo = Cairo({
23
- subsets: ['arabic', 'latin'],
24
- variable: '--font-cairo',
25
- })
26
-
27
- export async function generateMetadata(): Promise<Metadata> {
28
- const cmsConfig = await getCMSConfig()
29
- return {
30
- title: cmsConfig.ui.title,
31
- description: 'nextjs-cms',
32
- }
33
- }
34
-
35
- export default async function CMSLayout({ children }: { children: React.ReactNode }) {
36
- const [session, cmsConfig, _] = await Promise.all([
37
- auth(),
38
- getCMSConfig(),
39
- api.navigation.getSidebar.prefetch(),
40
- ])
41
-
42
- if (!session) {
43
- redirect('/auth/login')
44
- }
45
-
46
- const { supportedLanguages, fallbackLanguage } = cmsConfig.i18n
47
- const language = resolveLanguage(session.user.language, supportedLanguages, fallbackLanguage)
48
- const isRTL = RTL_LANGUAGES.has(language)
49
- const dictionaries = getClientDictionaries(supportedLanguages)
50
-
51
- return (
52
- <html lang={language} dir={isRTL ? 'rtl' : 'ltr'} suppressHydrationWarning>
53
- <body
54
- className={cn(
55
- 'bg-background min-h-screen font-sans antialiased',
56
- isRTL ? cairo.variable : inter.variable,
57
- )}
58
- >
59
- <DirectionProvider dir={isRTL ? 'rtl' : 'ltr'} direction={isRTL ? 'rtl' : 'ltr'}>
60
- <I18nProviderClient locale={language} dictionaries={dictionaries}>
61
- <ThemeProvider
62
- attribute='class'
63
- defaultTheme={cmsConfig.ui.defaultTheme}
64
- enableSystem
65
- disableTransitionOnChange
66
- >
67
- <Providers session={session ?? undefined}>
68
- <HydrateClient>
69
- <Layout logoUrlPath={cmsConfig.ui.logo} logoText={cmsConfig.ui.logoText} isRTL={isRTL}>
70
- {children}
71
- </Layout>
72
- </HydrateClient>
73
- </Providers>
74
- </ThemeProvider>
75
- </I18nProviderClient>
76
- </DirectionProvider>
77
- </body>
78
- </html>
79
-
80
- )
81
- }
1
+ import { Inter, Cairo } from 'next/font/google'
2
+ import '../globals.css'
3
+ import type { Metadata } from 'next'
4
+ import { cn } from '@/lib/utils'
5
+ import { ThemeProvider } from '@/components/layout/theme-provider'
6
+ import Providers from '@/app/providers'
7
+ import auth from 'nextjs-cms/auth'
8
+ import { getCMSConfig } from 'nextjs-cms/core'
9
+ import { I18nProviderClient } from 'nextjs-cms/translations/client'
10
+ import { resolveLanguage, RTL_LANGUAGES } from 'nextjs-cms/translations'
11
+ import { getClientDictionaries } from 'nextjs-cms/translations/server'
12
+ import { api, HydrateClient } from '@/app/_trpc/server'
13
+ import Layout from '@/components/layout/layout'
14
+ import { redirect } from 'next/navigation'
15
+ import { DirectionProvider } from "@/components/ui/direction"
16
+
17
+ const inter = Inter({
18
+ subsets: ['latin'],
19
+ variable: '--font-sans',
20
+ })
21
+
22
+ const cairo = Cairo({
23
+ subsets: ['arabic', 'latin'],
24
+ variable: '--font-cairo',
25
+ })
26
+
27
+ export async function generateMetadata(): Promise<Metadata> {
28
+ const cmsConfig = await getCMSConfig()
29
+ return {
30
+ title: cmsConfig.ui.title,
31
+ description: 'nextjs-cms',
32
+ }
33
+ }
34
+
35
+ export default async function CMSLayout({ children }: { children: React.ReactNode }) {
36
+ const [session, cmsConfig, _] = await Promise.all([
37
+ auth(),
38
+ getCMSConfig(),
39
+ api.navigation.getSidebar.prefetch(),
40
+ ])
41
+
42
+ if (!session) {
43
+ redirect('/auth/login')
44
+ }
45
+
46
+ const { supportedLanguages, fallbackLanguage } = cmsConfig.i18n
47
+ const language = resolveLanguage(session.user.language, supportedLanguages, fallbackLanguage)
48
+ const isRTL = RTL_LANGUAGES.has(language)
49
+ const dictionaries = getClientDictionaries(supportedLanguages)
50
+
51
+ return (
52
+ <html lang={language} dir={isRTL ? 'rtl' : 'ltr'} suppressHydrationWarning>
53
+ <body
54
+ className={cn(
55
+ 'bg-background min-h-screen font-sans antialiased',
56
+ isRTL ? cairo.variable : inter.variable,
57
+ )}
58
+ >
59
+ <DirectionProvider dir={isRTL ? 'rtl' : 'ltr'} direction={isRTL ? 'rtl' : 'ltr'}>
60
+ <I18nProviderClient locale={language} dictionaries={dictionaries}>
61
+ <ThemeProvider
62
+ attribute='class'
63
+ defaultTheme={cmsConfig.ui.defaultTheme}
64
+ enableSystem
65
+ disableTransitionOnChange
66
+ >
67
+ <Providers session={session ?? undefined}>
68
+ <HydrateClient>
69
+ <Layout logoUrlPath={cmsConfig.ui.logo} logoText={cmsConfig.ui.logoText} isRTL={isRTL}>
70
+ {children}
71
+ </Layout>
72
+ </HydrateClient>
73
+ </Providers>
74
+ </ThemeProvider>
75
+ </I18nProviderClient>
76
+ </DirectionProvider>
77
+ </body>
78
+ </html>
79
+
80
+ )
81
+ }
@@ -1,10 +1,10 @@
1
- import LoadingSpinners from '@/components/feedback/loading-spinners'
2
-
3
- export default function Loading() {
4
- // You can add any UI inside Loading, including a Skeleton.
5
- return (
6
- <div className='p-20'>
7
- <LoadingSpinners />
8
- </div>
9
- )
10
- }
1
+ import LoadingSpinners from '@/components/feedback/loading-spinners'
2
+
3
+ export default function Loading() {
4
+ // You can add any UI inside Loading, including a Skeleton.
5
+ return (
6
+ <div className='p-20'>
7
+ <LoadingSpinners />
8
+ </div>
9
+ )
10
+ }
@@ -1,7 +1,7 @@
1
- import LogPage from '@/components/pages/log-page'
2
-
3
- export const dynamic = 'force-dynamic'
4
-
5
- export default async function Page() {
6
- return <LogPage />
7
- }
1
+ import LogPage from '@/components/pages/log-page'
2
+
3
+ export const dynamic = 'force-dynamic'
4
+
5
+ export default async function Page() {
6
+ return <LogPage />
7
+ }
@@ -1,15 +1,15 @@
1
- import NewPage from '@/components/pages/new-page'
2
- import { api, HydrateClient } from '@/app/_trpc/server'
3
- type Params = Promise<{ section: string }>
4
-
5
- export default async function Page(props: { params: Params }) {
6
- const params = await props.params
7
- await api.hasItemsSections.newItem.prefetch({
8
- sectionName: params.section,
9
- })
10
- return (
11
- <HydrateClient>
12
- <NewPage sectionType='has_items' section={params.section} />
13
- </HydrateClient>
14
- )
15
- }
1
+ import NewPage from '@/components/pages/new-page'
2
+ import { api, HydrateClient } from '@/app/_trpc/server'
3
+ type Params = Promise<{ section: string }>
4
+
5
+ export default async function Page(props: { params: Params }) {
6
+ const params = await props.params
7
+ await api.hasItemsSections.newItem.prefetch({
8
+ sectionName: params.section,
9
+ })
10
+ return (
11
+ <HydrateClient>
12
+ <NewPage sectionType='has_items' section={params.section} />
13
+ </HydrateClient>
14
+ )
15
+ }
@@ -1,19 +1,19 @@
1
- import SectionPage from '@/components/pages/section-page'
2
- import { api, HydrateClient } from '@/app/_trpc/server'
3
- type Params = Promise<{ section: string }>
4
- type SearchParams = Promise<{ locale?: string }>
5
-
6
- export default async function Page(props: { params: Params; searchParams: SearchParams }) {
7
- const params = await props.params
8
- const searchParams = await props.searchParams
9
- await api.simpleSections.create.prefetch({
10
- sectionName: params.section,
11
- locale: searchParams.locale,
12
- })
13
-
14
- return (
15
- <HydrateClient>
16
- <SectionPage section={params.section} />
17
- </HydrateClient>
18
- )
19
- }
1
+ import SectionPage from '@/components/pages/section-page'
2
+ import { api, HydrateClient } from '@/app/_trpc/server'
3
+ type Params = Promise<{ section: string }>
4
+ type SearchParams = Promise<{ locale?: string }>
5
+
6
+ export default async function Page(props: { params: Params; searchParams: SearchParams }) {
7
+ const params = await props.params
8
+ const searchParams = await props.searchParams
9
+ await api.simpleSections.create.prefetch({
10
+ sectionName: params.section,
11
+ locale: searchParams.locale,
12
+ })
13
+
14
+ return (
15
+ <HydrateClient>
16
+ <SectionPage section={params.section} />
17
+ </HydrateClient>
18
+ )
19
+ }
@@ -1,13 +1,13 @@
1
- import SettingsPage from '@/components/pages/settings-page'
2
- import { api, HydrateClient } from '@/app/_trpc/server'
3
-
4
- export const dynamic = 'force-dynamic'
5
-
6
- export default async function Page() {
7
- await api.accountSettings.get.prefetch()
8
- return (
9
- <HydrateClient>
10
- <SettingsPage />
11
- </HydrateClient>
12
- )
13
- }
1
+ import SettingsPage from '@/components/pages/settings-page'
2
+ import { api, HydrateClient } from '@/app/_trpc/server'
3
+
4
+ export const dynamic = 'force-dynamic'
5
+
6
+ export default async function Page() {
7
+ await api.accountSettings.get.prefetch()
8
+ return (
9
+ <HydrateClient>
10
+ <SettingsPage />
11
+ </HydrateClient>
12
+ )
13
+ }