create-nextjs-cms 0.9.6 → 0.9.7

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 (184) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +71 -71
  3. package/dist/helpers/check-directory.d.ts +1 -0
  4. package/dist/helpers/check-directory.d.ts.map +1 -1
  5. package/dist/helpers/check-directory.js +98 -23
  6. package/dist/helpers/utils.js +16 -16
  7. package/dist/index.js +13 -8
  8. package/dist/lib/create-project.js +1 -1
  9. package/dist/lib/section-creators.js +166 -166
  10. package/package.json +1 -1
  11. package/templates/default/.eslintrc.json +5 -5
  12. package/templates/default/.prettierignore +7 -7
  13. package/templates/default/.prettierrc.json +27 -27
  14. package/templates/default/CHANGELOG.md +140 -140
  15. package/templates/default/_gitignore +57 -57
  16. package/templates/default/app/(auth)/auth/login/LoginPage.tsx +192 -192
  17. package/templates/default/app/(auth)/auth/login/page.tsx +11 -11
  18. package/templates/default/app/(auth)/auth-language-provider.tsx +34 -34
  19. package/templates/default/app/(auth)/layout.tsx +81 -81
  20. package/templates/default/app/(rootLayout)/(plugins)/[...slug]/page.tsx +40 -40
  21. package/templates/default/app/(rootLayout)/(plugins)/[...slug]/plugin-server-registry.ts +16 -16
  22. package/templates/default/app/(rootLayout)/admins/page.tsx +10 -10
  23. package/templates/default/app/(rootLayout)/browse/[section]/[page]/page.tsx +22 -22
  24. package/templates/default/app/(rootLayout)/categorized/[section]/page.tsx +15 -15
  25. package/templates/default/app/(rootLayout)/dashboard/page.tsx +63 -63
  26. package/templates/default/app/(rootLayout)/edit/[section]/[itemId]/page.tsx +20 -20
  27. package/templates/default/app/(rootLayout)/layout.tsx +81 -81
  28. package/templates/default/app/(rootLayout)/loading.tsx +10 -10
  29. package/templates/default/app/(rootLayout)/log/page.tsx +7 -7
  30. package/templates/default/app/(rootLayout)/new/[section]/page.tsx +15 -15
  31. package/templates/default/app/(rootLayout)/section/[section]/page.tsx +19 -19
  32. package/templates/default/app/(rootLayout)/settings/page.tsx +13 -13
  33. package/templates/default/app/_trpc/client.ts +3 -3
  34. package/templates/default/app/api/auth/csrf/route.ts +25 -25
  35. package/templates/default/app/api/auth/refresh/route.ts +10 -10
  36. package/templates/default/app/api/auth/route.ts +49 -49
  37. package/templates/default/app/api/auth/session/route.ts +20 -20
  38. package/templates/default/app/api/document/route.ts +165 -165
  39. package/templates/default/app/api/editor/photo/route.ts +49 -49
  40. package/templates/default/app/api/photo/route.ts +27 -27
  41. package/templates/default/app/api/submit/section/item/[slug]/route.ts +95 -95
  42. package/templates/default/app/api/submit/section/item/route.ts +56 -56
  43. package/templates/default/app/api/submit/section/simple/route.ts +86 -86
  44. package/templates/default/app/api/trpc/[trpc]/route.ts +33 -33
  45. package/templates/default/app/api/video/route.ts +174 -174
  46. package/templates/default/app/globals.css +228 -228
  47. package/templates/default/app/providers.tsx +152 -152
  48. package/templates/default/cms.config.ts +57 -58
  49. package/templates/default/components/AdminCard.tsx +166 -166
  50. package/templates/default/components/AdminEditPage.tsx +124 -124
  51. package/templates/default/components/AdminPrivilegeCard.tsx +185 -185
  52. package/templates/default/components/AdminsPage.tsx +43 -43
  53. package/templates/default/components/AnalyticsPage.tsx +144 -144
  54. package/templates/default/components/BarChartBox.tsx +42 -42
  55. package/templates/default/components/BrowsePage.tsx +106 -106
  56. package/templates/default/components/CategorizedSectionPage.tsx +31 -31
  57. package/templates/default/components/CategoryDeleteConfirmPage.tsx +130 -130
  58. package/templates/default/components/CategorySectionSelectInput.tsx +140 -140
  59. package/templates/default/components/ConditionalFields.tsx +49 -49
  60. package/templates/default/components/ContainerBox.tsx +24 -24
  61. package/templates/default/components/DashboardPageAlt.tsx +45 -45
  62. package/templates/default/components/DefaultNavItems.tsx +3 -3
  63. package/templates/default/components/Dropzone.tsx +154 -154
  64. package/templates/default/components/ErrorComponent.tsx +16 -16
  65. package/templates/default/components/GalleryPhoto.tsx +93 -93
  66. package/templates/default/components/InfoCard.tsx +93 -93
  67. package/templates/default/components/ItemEditPage.tsx +294 -294
  68. package/templates/default/components/Layout.tsx +84 -84
  69. package/templates/default/components/LoadingSpinners.tsx +67 -67
  70. package/templates/default/components/LocaleSwitcher.tsx +89 -89
  71. package/templates/default/components/LogPage.tsx +107 -107
  72. package/templates/default/components/Modal.tsx +166 -166
  73. package/templates/default/components/Navbar.tsx +258 -258
  74. package/templates/default/components/NewAdminForm.tsx +173 -173
  75. package/templates/default/components/NewPage.tsx +206 -206
  76. package/templates/default/components/NewVariantComponent.tsx +229 -229
  77. package/templates/default/components/PhotoGallery.tsx +35 -35
  78. package/templates/default/components/PieChartBox.tsx +101 -101
  79. package/templates/default/components/ProgressBar.tsx +48 -48
  80. package/templates/default/components/ProtectedDocument.tsx +44 -44
  81. package/templates/default/components/ProtectedImage.tsx +143 -143
  82. package/templates/default/components/ProtectedVideo.tsx +76 -76
  83. package/templates/default/components/SectionIcon.tsx +8 -8
  84. package/templates/default/components/SectionItemCard.tsx +144 -144
  85. package/templates/default/components/SectionItemStatusBadge.tsx +17 -17
  86. package/templates/default/components/SectionPage.tsx +205 -205
  87. package/templates/default/components/SelectBox.tsx +98 -98
  88. package/templates/default/components/SelectInputButtons.tsx +125 -125
  89. package/templates/default/components/SettingsPage.tsx +232 -232
  90. package/templates/default/components/Sidebar.tsx +204 -204
  91. package/templates/default/components/SidebarDropdownItem.tsx +83 -83
  92. package/templates/default/components/SidebarItem.tsx +24 -24
  93. package/templates/default/components/ThemeProvider.tsx +8 -8
  94. package/templates/default/components/TooltipComponent.tsx +27 -27
  95. package/templates/default/components/VariantCard.tsx +124 -124
  96. package/templates/default/components/VariantEditPage.tsx +230 -230
  97. package/templates/default/components/analytics/BounceRate.tsx +70 -70
  98. package/templates/default/components/analytics/LivePageViews.tsx +55 -55
  99. package/templates/default/components/analytics/LiveUsersCount.tsx +33 -33
  100. package/templates/default/components/analytics/MonthlyPageViews.tsx +42 -42
  101. package/templates/default/components/analytics/TopCountries.tsx +52 -52
  102. package/templates/default/components/analytics/TopDevices.tsx +46 -46
  103. package/templates/default/components/analytics/TopMediums.tsx +58 -58
  104. package/templates/default/components/analytics/TopSources.tsx +45 -45
  105. package/templates/default/components/analytics/TotalPageViews.tsx +41 -41
  106. package/templates/default/components/analytics/TotalSessions.tsx +41 -41
  107. package/templates/default/components/analytics/TotalUniqueUsers.tsx +41 -41
  108. package/templates/default/components/custom/RightHomeRoomVariantCard.tsx +138 -138
  109. package/templates/default/components/dndKit/Draggable.tsx +21 -21
  110. package/templates/default/components/dndKit/Droppable.tsx +20 -20
  111. package/templates/default/components/dndKit/SortableItem.tsx +18 -18
  112. package/templates/default/components/form/Form.tsx +370 -370
  113. package/templates/default/components/form/FormInputElement.tsx +70 -70
  114. package/templates/default/components/form/FormInputs.tsx +136 -136
  115. package/templates/default/components/form/helpers/_section-hot-reload.js +1 -1
  116. package/templates/default/components/form/helpers/util.ts +17 -17
  117. package/templates/default/components/form/inputs/CheckboxFormInput.tsx +46 -46
  118. package/templates/default/components/form/inputs/ColorFormInput.tsx +44 -44
  119. package/templates/default/components/form/inputs/DateFormInput.tsx +167 -110
  120. package/templates/default/components/form/inputs/DateRangeFormInput.tsx +23 -1
  121. package/templates/default/components/form/inputs/DocumentFormInput.tsx +222 -222
  122. package/templates/default/components/form/inputs/MapFormInput.tsx +140 -140
  123. package/templates/default/components/form/inputs/MultipleSelectFormInput.tsx +85 -85
  124. package/templates/default/components/form/inputs/NumberFormInput.tsx +43 -43
  125. package/templates/default/components/form/inputs/PasswordFormInput.tsx +47 -47
  126. package/templates/default/components/form/inputs/PhotoFormInput.tsx +275 -275
  127. package/templates/default/components/form/inputs/RichTextFormInput.tsx +138 -138
  128. package/templates/default/components/form/inputs/SelectFormInput.tsx +175 -175
  129. package/templates/default/components/form/inputs/SlugFormInput.tsx +131 -131
  130. package/templates/default/components/form/inputs/TagsFormInput.tsx +265 -265
  131. package/templates/default/components/form/inputs/TextFormInput.tsx +51 -51
  132. package/templates/default/components/form/inputs/TextareaFormInput.tsx +50 -50
  133. package/templates/default/components/form/inputs/VideoFormInput.tsx +118 -118
  134. package/templates/default/components/multi-select.tsx +1146 -1146
  135. package/templates/default/components/pagination/Pagination.tsx +36 -36
  136. package/templates/default/components/pagination/PaginationButtons.tsx +147 -147
  137. package/templates/default/components/theme-toggle.tsx +39 -39
  138. package/templates/default/components/ui/accordion.tsx +53 -53
  139. package/templates/default/components/ui/alert-dialog.tsx +157 -157
  140. package/templates/default/components/ui/alert.tsx +47 -47
  141. package/templates/default/components/ui/badge.tsx +38 -38
  142. package/templates/default/components/ui/button.tsx +62 -62
  143. package/templates/default/components/ui/calendar.tsx +166 -166
  144. package/templates/default/components/ui/card.tsx +43 -43
  145. package/templates/default/components/ui/checkbox.tsx +29 -29
  146. package/templates/default/components/ui/command.tsx +137 -137
  147. package/templates/default/components/ui/custom-alert-dialog.tsx +113 -113
  148. package/templates/default/components/ui/custom-dialog.tsx +123 -123
  149. package/templates/default/components/ui/dialog.tsx +123 -123
  150. package/templates/default/components/ui/direction.tsx +22 -22
  151. package/templates/default/components/ui/dropdown-menu.tsx +182 -182
  152. package/templates/default/components/ui/input-group.tsx +54 -54
  153. package/templates/default/components/ui/input.tsx +22 -22
  154. package/templates/default/components/ui/label.tsx +19 -19
  155. package/templates/default/components/ui/popover.tsx +42 -42
  156. package/templates/default/components/ui/progress.tsx +31 -31
  157. package/templates/default/components/ui/scroll-area.tsx +42 -42
  158. package/templates/default/components/ui/select.tsx +165 -165
  159. package/templates/default/components/ui/separator.tsx +28 -28
  160. package/templates/default/components/ui/sheet.tsx +103 -103
  161. package/templates/default/components/ui/spinner.tsx +16 -16
  162. package/templates/default/components/ui/switch.tsx +29 -29
  163. package/templates/default/components/ui/table.tsx +83 -83
  164. package/templates/default/components/ui/tabs.tsx +55 -55
  165. package/templates/default/components/ui/toast.tsx +113 -113
  166. package/templates/default/components/ui/toaster.tsx +35 -35
  167. package/templates/default/components/ui/tooltip.tsx +30 -30
  168. package/templates/default/components/ui/use-toast.ts +188 -188
  169. package/templates/default/components.json +21 -21
  170. package/templates/default/context/ModalProvider.tsx +53 -53
  171. package/templates/default/drizzle.config.ts +4 -4
  172. package/templates/default/dynamic-schemas/schema.ts +475 -0
  173. package/templates/default/env/env.js +130 -130
  174. package/templates/default/envConfig.ts +4 -4
  175. package/templates/default/hooks/useModal.ts +8 -8
  176. package/templates/default/lib/apiHelpers.ts +92 -92
  177. package/templates/default/lib/postinstall.js +14 -14
  178. package/templates/default/lib/utils.ts +6 -6
  179. package/templates/default/next-env.d.ts +6 -6
  180. package/templates/default/next.config.ts +23 -23
  181. package/templates/default/package.json +1 -1
  182. package/templates/default/postcss.config.mjs +6 -6
  183. package/templates/default/proxy.ts +32 -32
  184. package/templates/default/tsconfig.json +48 -48
@@ -1,84 +1,84 @@
1
- 'use client'
2
-
3
- import classNames from 'classnames'
4
- import React, { useState } from 'react'
5
- import Navbar from '@/components/Navbar'
6
- import Sidebar from '@/components/Sidebar'
7
- import Modal from '@/components/Modal'
8
- import { Toaster } from '@/components/ui/toaster'
9
- import { useSession } from 'nextjs-cms/auth/react'
10
- import LoadingSpinners from '@/components/LoadingSpinners'
11
-
12
- function Layout({
13
- children,
14
- logoUrlPath,
15
- logoText,
16
- isRTL,
17
- }: {
18
- children: React.ReactNode
19
- logoUrlPath: string
20
- logoText: string
21
- isRTL: boolean
22
- }) {
23
- const [showMobileSidebar, setShowMobileSidebar] = useState(false)
24
- const session = useSession({
25
- required: true,
26
- })
27
-
28
- return (
29
- <div
30
- className={classNames({
31
- 'bg-background text-foreground flex min-h-screen flex-col': true,
32
- 'overflow-hidden': true,
33
- })}
34
- >
35
- <div>
36
- {session.status === 'loading' ? (
37
- <div className='align-center flex w-full flex-row items-center justify-center p-2'>
38
- <LoadingSpinners />
39
- </div>
40
- ) : (
41
- <>
42
- <Sidebar
43
- closeSideBar={() => {
44
- setShowMobileSidebar(false)
45
- }}
46
- mobileSidebar={showMobileSidebar}
47
- logoUrlPath={logoUrlPath}
48
- logoText={logoText}
49
- isRTL={isRTL}
50
- />
51
- {showMobileSidebar && (
52
- // Display a black transparent div to close the sidebar when clicked outside
53
- <div
54
- className='fixed top-0 left-0 z-15 h-full w-full bg-black/80 md:hidden'
55
- onClick={() => setShowMobileSidebar(false)}
56
- ></div>
57
- )}
58
- <div
59
- className={classNames({
60
- 'transition-all duration-100 ease-in-out': true,
61
- 'float-right': !isRTL,
62
- 'float-left': isRTL,
63
- 'w-full md:w-[calc(100%-275px)]': true,
64
- })}
65
- >
66
- <Navbar onMenuButtonClick={() => setShowMobileSidebar((prev) => !prev)} />
67
- <div
68
- className={classNames({
69
- 'h-full w-full': true,
70
- })}
71
- >
72
- {children}
73
- </div>
74
- </div>
75
- <Modal />
76
- <Toaster />
77
- </>
78
- )}
79
- </div>
80
- </div>
81
- )
82
- }
83
-
84
- export default Layout
1
+ 'use client'
2
+
3
+ import classNames from 'classnames'
4
+ import React, { useState } from 'react'
5
+ import Navbar from '@/components/Navbar'
6
+ import Sidebar from '@/components/Sidebar'
7
+ import Modal from '@/components/Modal'
8
+ import { Toaster } from '@/components/ui/toaster'
9
+ import { useSession } from 'nextjs-cms/auth/react'
10
+ import LoadingSpinners from '@/components/LoadingSpinners'
11
+
12
+ function Layout({
13
+ children,
14
+ logoUrlPath,
15
+ logoText,
16
+ isRTL,
17
+ }: {
18
+ children: React.ReactNode
19
+ logoUrlPath: string
20
+ logoText: string
21
+ isRTL: boolean
22
+ }) {
23
+ const [showMobileSidebar, setShowMobileSidebar] = useState(false)
24
+ const session = useSession({
25
+ required: true,
26
+ })
27
+
28
+ return (
29
+ <div
30
+ className={classNames({
31
+ 'bg-background text-foreground flex min-h-screen flex-col': true,
32
+ 'overflow-hidden': true,
33
+ })}
34
+ >
35
+ <div>
36
+ {session.status === 'loading' ? (
37
+ <div className='align-center flex w-full flex-row items-center justify-center p-2'>
38
+ <LoadingSpinners />
39
+ </div>
40
+ ) : (
41
+ <>
42
+ <Sidebar
43
+ closeSideBar={() => {
44
+ setShowMobileSidebar(false)
45
+ }}
46
+ mobileSidebar={showMobileSidebar}
47
+ logoUrlPath={logoUrlPath}
48
+ logoText={logoText}
49
+ isRTL={isRTL}
50
+ />
51
+ {showMobileSidebar && (
52
+ // Display a black transparent div to close the sidebar when clicked outside
53
+ <div
54
+ className='fixed top-0 left-0 z-15 h-full w-full bg-black/80 md:hidden'
55
+ onClick={() => setShowMobileSidebar(false)}
56
+ ></div>
57
+ )}
58
+ <div
59
+ className={classNames({
60
+ 'transition-all duration-100 ease-in-out': true,
61
+ 'float-right': !isRTL,
62
+ 'float-left': isRTL,
63
+ 'w-full md:w-[calc(100%-275px)]': true,
64
+ })}
65
+ >
66
+ <Navbar onMenuButtonClick={() => setShowMobileSidebar((prev) => !prev)} />
67
+ <div
68
+ className={classNames({
69
+ 'h-full w-full': true,
70
+ })}
71
+ >
72
+ {children}
73
+ </div>
74
+ </div>
75
+ <Modal />
76
+ <Toaster />
77
+ </>
78
+ )}
79
+ </div>
80
+ </div>
81
+ )
82
+ }
83
+
84
+ export default Layout
@@ -1,67 +1,67 @@
1
- const LoadingSpinners = ({ single = false }: { single?: boolean }) => {
2
- return (
3
- <div className='flex justify-center'>
4
- <div role='status'>
5
- <svg
6
- aria-hidden='true'
7
- className='mr-2 inline h-8 w-8 animate-spin fill-blue-600 text-gray-200 dark:text-gray-600'
8
- viewBox='0 0 100 101'
9
- fill='none'
10
- xmlns='http://www.w3.org/2000/svg'
11
- >
12
- <path
13
- d='M100 50.5908C100 78.2051 77.6142 100.591 50 100.591C22.3858 100.591 0 78.2051 0 50.5908C0 22.9766 22.3858 0.59082 50 0.59082C77.6142 0.59082 100 22.9766 100 50.5908ZM9.08144 50.5908C9.08144 73.1895 27.4013 91.5094 50 91.5094C72.5987 91.5094 90.9186 73.1895 90.9186 50.5908C90.9186 27.9921 72.5987 9.67226 50 9.67226C27.4013 9.67226 9.08144 27.9921 9.08144 50.5908Z'
14
- fill='currentColor'
15
- />
16
- <path
17
- d='M93.9676 39.0409C96.393 38.4038 97.8624 35.9116 97.0079 33.5539C95.2932 28.8227 92.871 24.3692 89.8167 20.348C85.8452 15.1192 80.8826 10.7238 75.2124 7.41289C69.5422 4.10194 63.2754 1.94025 56.7698 1.05124C51.7666 0.367541 46.6976 0.446843 41.7345 1.27873C39.2613 1.69328 37.813 4.19778 38.4501 6.62326C39.0873 9.04874 41.5694 10.4717 44.0505 10.1071C47.8511 9.54855 51.7191 9.52689 55.5402 10.0491C60.8642 10.7766 65.9928 12.5457 70.6331 15.2552C75.2735 17.9648 79.3347 21.5619 82.5849 25.841C84.9175 28.9121 86.7997 32.2913 88.1811 35.8758C89.083 38.2158 91.5421 39.6781 93.9676 39.0409Z'
18
- fill='currentFill'
19
- />
20
- </svg>
21
- </div>
22
- {!single && (
23
- <>
24
- <div role='status'>
25
- <svg
26
- aria-hidden='true'
27
- className='mr-2 inline h-8 w-8 animate-spin fill-red-600 text-gray-200 dark:fill-gray-300 dark:text-gray-600'
28
- viewBox='0 0 100 101'
29
- fill='none'
30
- xmlns='http://www.w3.org/2000/svg'
31
- >
32
- <path
33
- d='M100 50.5908C100 78.2051 77.6142 100.591 50 100.591C22.3858 100.591 0 78.2051 0 50.5908C0 22.9766 22.3858 0.59082 50 0.59082C77.6142 0.59082 100 22.9766 100 50.5908ZM9.08144 50.5908C9.08144 73.1895 27.4013 91.5094 50 91.5094C72.5987 91.5094 90.9186 73.1895 90.9186 50.5908C90.9186 27.9921 72.5987 9.67226 50 9.67226C27.4013 9.67226 9.08144 27.9921 9.08144 50.5908Z'
34
- fill='currentColor'
35
- />
36
- <path
37
- d='M93.9676 39.0409C96.393 38.4038 97.8624 35.9116 97.0079 33.5539C95.2932 28.8227 92.871 24.3692 89.8167 20.348C85.8452 15.1192 80.8826 10.7238 75.2124 7.41289C69.5422 4.10194 63.2754 1.94025 56.7698 1.05124C51.7666 0.367541 46.6976 0.446843 41.7345 1.27873C39.2613 1.69328 37.813 4.19778 38.4501 6.62326C39.0873 9.04874 41.5694 10.4717 44.0505 10.1071C47.8511 9.54855 51.7191 9.52689 55.5402 10.0491C60.8642 10.7766 65.9928 12.5457 70.6331 15.2552C75.2735 17.9648 79.3347 21.5619 82.5849 25.841C84.9175 28.9121 86.7997 32.2913 88.1811 35.8758C89.083 38.2158 91.5421 39.6781 93.9676 39.0409Z'
38
- fill='currentFill'
39
- />
40
- </svg>
41
- </div>
42
- <div role='status'>
43
- <svg
44
- aria-hidden='true'
45
- className='mr-2 inline h-8 w-8 animate-spin fill-green-500 text-gray-200 dark:text-gray-600'
46
- viewBox='0 0 100 101'
47
- fill='none'
48
- xmlns='http://www.w3.org/2000/svg'
49
- >
50
- <path
51
- d='M100 50.5908C100 78.2051 77.6142 100.591 50 100.591C22.3858 100.591 0 78.2051 0 50.5908C0 22.9766 22.3858 0.59082 50 0.59082C77.6142 0.59082 100 22.9766 100 50.5908ZM9.08144 50.5908C9.08144 73.1895 27.4013 91.5094 50 91.5094C72.5987 91.5094 90.9186 73.1895 90.9186 50.5908C90.9186 27.9921 72.5987 9.67226 50 9.67226C27.4013 9.67226 9.08144 27.9921 9.08144 50.5908Z'
52
- fill='currentColor'
53
- />
54
- <path
55
- d='M93.9676 39.0409C96.393 38.4038 97.8624 35.9116 97.0079 33.5539C95.2932 28.8227 92.871 24.3692 89.8167 20.348C85.8452 15.1192 80.8826 10.7238 75.2124 7.41289C69.5422 4.10194 63.2754 1.94025 56.7698 1.05124C51.7666 0.367541 46.6976 0.446843 41.7345 1.27873C39.2613 1.69328 37.813 4.19778 38.4501 6.62326C39.0873 9.04874 41.5694 10.4717 44.0505 10.1071C47.8511 9.54855 51.7191 9.52689 55.5402 10.0491C60.8642 10.7766 65.9928 12.5457 70.6331 15.2552C75.2735 17.9648 79.3347 21.5619 82.5849 25.841C84.9175 28.9121 86.7997 32.2913 88.1811 35.8758C89.083 38.2158 91.5421 39.6781 93.9676 39.0409Z'
56
- fill='currentFill'
57
- />
58
- </svg>
59
- </div>
60
- </>
61
- )}
62
- <span className='sr-only'>Loading</span>
63
- </div>
64
- )
65
- }
66
-
67
- export default LoadingSpinners
1
+ const LoadingSpinners = ({ single = false }: { single?: boolean }) => {
2
+ return (
3
+ <div className='flex justify-center'>
4
+ <div role='status'>
5
+ <svg
6
+ aria-hidden='true'
7
+ className='mr-2 inline h-8 w-8 animate-spin fill-blue-600 text-gray-200 dark:text-gray-600'
8
+ viewBox='0 0 100 101'
9
+ fill='none'
10
+ xmlns='http://www.w3.org/2000/svg'
11
+ >
12
+ <path
13
+ d='M100 50.5908C100 78.2051 77.6142 100.591 50 100.591C22.3858 100.591 0 78.2051 0 50.5908C0 22.9766 22.3858 0.59082 50 0.59082C77.6142 0.59082 100 22.9766 100 50.5908ZM9.08144 50.5908C9.08144 73.1895 27.4013 91.5094 50 91.5094C72.5987 91.5094 90.9186 73.1895 90.9186 50.5908C90.9186 27.9921 72.5987 9.67226 50 9.67226C27.4013 9.67226 9.08144 27.9921 9.08144 50.5908Z'
14
+ fill='currentColor'
15
+ />
16
+ <path
17
+ d='M93.9676 39.0409C96.393 38.4038 97.8624 35.9116 97.0079 33.5539C95.2932 28.8227 92.871 24.3692 89.8167 20.348C85.8452 15.1192 80.8826 10.7238 75.2124 7.41289C69.5422 4.10194 63.2754 1.94025 56.7698 1.05124C51.7666 0.367541 46.6976 0.446843 41.7345 1.27873C39.2613 1.69328 37.813 4.19778 38.4501 6.62326C39.0873 9.04874 41.5694 10.4717 44.0505 10.1071C47.8511 9.54855 51.7191 9.52689 55.5402 10.0491C60.8642 10.7766 65.9928 12.5457 70.6331 15.2552C75.2735 17.9648 79.3347 21.5619 82.5849 25.841C84.9175 28.9121 86.7997 32.2913 88.1811 35.8758C89.083 38.2158 91.5421 39.6781 93.9676 39.0409Z'
18
+ fill='currentFill'
19
+ />
20
+ </svg>
21
+ </div>
22
+ {!single && (
23
+ <>
24
+ <div role='status'>
25
+ <svg
26
+ aria-hidden='true'
27
+ className='mr-2 inline h-8 w-8 animate-spin fill-red-600 text-gray-200 dark:fill-gray-300 dark:text-gray-600'
28
+ viewBox='0 0 100 101'
29
+ fill='none'
30
+ xmlns='http://www.w3.org/2000/svg'
31
+ >
32
+ <path
33
+ d='M100 50.5908C100 78.2051 77.6142 100.591 50 100.591C22.3858 100.591 0 78.2051 0 50.5908C0 22.9766 22.3858 0.59082 50 0.59082C77.6142 0.59082 100 22.9766 100 50.5908ZM9.08144 50.5908C9.08144 73.1895 27.4013 91.5094 50 91.5094C72.5987 91.5094 90.9186 73.1895 90.9186 50.5908C90.9186 27.9921 72.5987 9.67226 50 9.67226C27.4013 9.67226 9.08144 27.9921 9.08144 50.5908Z'
34
+ fill='currentColor'
35
+ />
36
+ <path
37
+ d='M93.9676 39.0409C96.393 38.4038 97.8624 35.9116 97.0079 33.5539C95.2932 28.8227 92.871 24.3692 89.8167 20.348C85.8452 15.1192 80.8826 10.7238 75.2124 7.41289C69.5422 4.10194 63.2754 1.94025 56.7698 1.05124C51.7666 0.367541 46.6976 0.446843 41.7345 1.27873C39.2613 1.69328 37.813 4.19778 38.4501 6.62326C39.0873 9.04874 41.5694 10.4717 44.0505 10.1071C47.8511 9.54855 51.7191 9.52689 55.5402 10.0491C60.8642 10.7766 65.9928 12.5457 70.6331 15.2552C75.2735 17.9648 79.3347 21.5619 82.5849 25.841C84.9175 28.9121 86.7997 32.2913 88.1811 35.8758C89.083 38.2158 91.5421 39.6781 93.9676 39.0409Z'
38
+ fill='currentFill'
39
+ />
40
+ </svg>
41
+ </div>
42
+ <div role='status'>
43
+ <svg
44
+ aria-hidden='true'
45
+ className='mr-2 inline h-8 w-8 animate-spin fill-green-500 text-gray-200 dark:text-gray-600'
46
+ viewBox='0 0 100 101'
47
+ fill='none'
48
+ xmlns='http://www.w3.org/2000/svg'
49
+ >
50
+ <path
51
+ d='M100 50.5908C100 78.2051 77.6142 100.591 50 100.591C22.3858 100.591 0 78.2051 0 50.5908C0 22.9766 22.3858 0.59082 50 0.59082C77.6142 0.59082 100 22.9766 100 50.5908ZM9.08144 50.5908C9.08144 73.1895 27.4013 91.5094 50 91.5094C72.5987 91.5094 90.9186 73.1895 90.9186 50.5908C90.9186 27.9921 72.5987 9.67226 50 9.67226C27.4013 9.67226 9.08144 27.9921 9.08144 50.5908Z'
52
+ fill='currentColor'
53
+ />
54
+ <path
55
+ d='M93.9676 39.0409C96.393 38.4038 97.8624 35.9116 97.0079 33.5539C95.2932 28.8227 92.871 24.3692 89.8167 20.348C85.8452 15.1192 80.8826 10.7238 75.2124 7.41289C69.5422 4.10194 63.2754 1.94025 56.7698 1.05124C51.7666 0.367541 46.6976 0.446843 41.7345 1.27873C39.2613 1.69328 37.813 4.19778 38.4501 6.62326C39.0873 9.04874 41.5694 10.4717 44.0505 10.1071C47.8511 9.54855 51.7191 9.52689 55.5402 10.0491C60.8642 10.7766 65.9928 12.5457 70.6331 15.2552C75.2735 17.9648 79.3347 21.5619 82.5849 25.841C84.9175 28.9121 86.7997 32.2913 88.1811 35.8758C89.083 38.2158 91.5421 39.6781 93.9676 39.0409Z'
56
+ fill='currentFill'
57
+ />
58
+ </svg>
59
+ </div>
60
+ </>
61
+ )}
62
+ <span className='sr-only'>Loading</span>
63
+ </div>
64
+ )
65
+ }
66
+
67
+ export default LoadingSpinners
@@ -1,89 +1,89 @@
1
- 'use client'
2
-
3
- import Link from 'next/link'
4
- import { cn } from '@/lib/utils'
5
- import { Check } from 'lucide-react'
6
- import { useI18n } from 'nextjs-cms/translations/client'
7
- import { useSession } from 'nextjs-cms/auth/react'
8
- import ContainerBox from './ContainerBox'
9
-
10
- type LocaleConfig = {
11
- code: string
12
- label: string
13
- rtl?: boolean
14
- }
15
-
16
- type Props = {
17
- section: string
18
- itemId?: string
19
- currentLocale: LocaleConfig
20
- defaultLocale: LocaleConfig
21
- existingTranslations: string[]
22
- locales: LocaleConfig[]
23
- hasUnsavedChanges?: boolean
24
- /** Base path for locale links. Defaults to `/edit/${section}/${itemId}` for hasItems sections. */
25
- basePath?: string
26
- }
27
-
28
- export default function LocaleSwitcher({
29
- section,
30
- itemId,
31
- currentLocale,
32
- defaultLocale,
33
- existingTranslations,
34
- locales,
35
- hasUnsavedChanges,
36
- basePath,
37
- }: Props) {
38
- const t = useI18n()
39
- const resolvedBasePath = basePath ?? `/edit/${section}/${itemId}`
40
-
41
- const getHref = (localeCode: string) => {
42
- if (localeCode === defaultLocale.code) {
43
- return resolvedBasePath
44
- }
45
- return `${resolvedBasePath}?locale=${localeCode}`
46
- }
47
-
48
- const handleClick = (e: React.MouseEvent, localeCode: string) => {
49
- if (localeCode === currentLocale.code) {
50
- e.preventDefault()
51
- return
52
- }
53
- if (hasUnsavedChanges) {
54
- if (!window.confirm(t('unsavedChangesConfirm'))) {
55
- e.preventDefault()
56
- }
57
- }
58
- }
59
-
60
- return (
61
- <ContainerBox title={t('localesHeading')}>
62
- <div className="flex flex-wrap items-center gap-2">
63
- {locales.map((locale) => {
64
- const isActive = locale.code === currentLocale.code
65
- const hasTranslation = locale.code === defaultLocale.code || existingTranslations.includes(locale.code)
66
- const isDefault = locale.code === defaultLocale.code
67
-
68
- return (
69
- <Link
70
- key={locale.code}
71
- href={getHref(locale.code)}
72
- onClick={(e) => handleClick(e, locale.code)}
73
- className={cn(
74
- 'inline-flex border border-primary/20 items-center gap-1.5 rounded-md px-3 py-1.5 text-sm font-medium transition-colors',
75
- isActive
76
- ? 'bg-success text-success-foreground'
77
- : 'bg-muted text-muted-foreground hover:bg-accent hover:text-accent-foreground',
78
- )}
79
- >
80
- {locale.label}
81
- {isDefault && <span className="text-[10px] opacity-70">{t('baseLocaleBadge')}</span>}
82
- {hasTranslation && !isDefault && <Check className="h-3 w-3" />}
83
- </Link>
84
- )
85
- })}
86
- </div>
87
- </ContainerBox>
88
- )
89
- }
1
+ 'use client'
2
+
3
+ import Link from 'next/link'
4
+ import { cn } from '@/lib/utils'
5
+ import { Check } from 'lucide-react'
6
+ import { useI18n } from 'nextjs-cms/translations/client'
7
+ import { useSession } from 'nextjs-cms/auth/react'
8
+ import ContainerBox from './ContainerBox'
9
+
10
+ type LocaleConfig = {
11
+ code: string
12
+ label: string
13
+ rtl?: boolean
14
+ }
15
+
16
+ type Props = {
17
+ section: string
18
+ itemId?: string
19
+ currentLocale: LocaleConfig
20
+ defaultLocale: LocaleConfig
21
+ existingTranslations: string[]
22
+ locales: LocaleConfig[]
23
+ hasUnsavedChanges?: boolean
24
+ /** Base path for locale links. Defaults to `/edit/${section}/${itemId}` for hasItems sections. */
25
+ basePath?: string
26
+ }
27
+
28
+ export default function LocaleSwitcher({
29
+ section,
30
+ itemId,
31
+ currentLocale,
32
+ defaultLocale,
33
+ existingTranslations,
34
+ locales,
35
+ hasUnsavedChanges,
36
+ basePath,
37
+ }: Props) {
38
+ const t = useI18n()
39
+ const resolvedBasePath = basePath ?? `/edit/${section}/${itemId}`
40
+
41
+ const getHref = (localeCode: string) => {
42
+ if (localeCode === defaultLocale.code) {
43
+ return resolvedBasePath
44
+ }
45
+ return `${resolvedBasePath}?locale=${localeCode}`
46
+ }
47
+
48
+ const handleClick = (e: React.MouseEvent, localeCode: string) => {
49
+ if (localeCode === currentLocale.code) {
50
+ e.preventDefault()
51
+ return
52
+ }
53
+ if (hasUnsavedChanges) {
54
+ if (!window.confirm(t('unsavedChangesConfirm'))) {
55
+ e.preventDefault()
56
+ }
57
+ }
58
+ }
59
+
60
+ return (
61
+ <ContainerBox title={t('localesHeading')}>
62
+ <div className="flex flex-wrap items-center gap-2">
63
+ {locales.map((locale) => {
64
+ const isActive = locale.code === currentLocale.code
65
+ const hasTranslation = locale.code === defaultLocale.code || existingTranslations.includes(locale.code)
66
+ const isDefault = locale.code === defaultLocale.code
67
+
68
+ return (
69
+ <Link
70
+ key={locale.code}
71
+ href={getHref(locale.code)}
72
+ onClick={(e) => handleClick(e, locale.code)}
73
+ className={cn(
74
+ 'inline-flex border border-primary/20 items-center gap-1.5 rounded-md px-3 py-1.5 text-sm font-medium transition-colors',
75
+ isActive
76
+ ? 'bg-success text-success-foreground'
77
+ : 'bg-muted text-muted-foreground hover:bg-accent hover:text-accent-foreground',
78
+ )}
79
+ >
80
+ {locale.label}
81
+ {isDefault && <span className="text-[10px] opacity-70">{t('baseLocaleBadge')}</span>}
82
+ {hasTranslation && !isDefault && <Check className="h-3 w-3" />}
83
+ </Link>
84
+ )
85
+ })}
86
+ </div>
87
+ </ContainerBox>
88
+ )
89
+ }