create-nextjs-cms 0.8.7 → 0.8.9

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 (189) 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 +3 -3
  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-locale-provider.tsx +34 -34
  14. package/templates/default/app/(auth)/layout.tsx +81 -81
  15. package/templates/default/app/(rootLayout)/(plugins)/[...slug]/page.tsx +40 -40
  16. package/templates/default/app/(rootLayout)/(plugins)/[...slug]/plugin-server-registry.ts +16 -16
  17. package/templates/default/app/(rootLayout)/admins/page.tsx +10 -10
  18. package/templates/default/app/(rootLayout)/browse/[section]/[page]/page.tsx +22 -22
  19. package/templates/default/app/(rootLayout)/categorized/[section]/page.tsx +15 -15
  20. package/templates/default/app/(rootLayout)/dashboard/page.tsx +63 -63
  21. package/templates/default/app/(rootLayout)/dashboard-new/page.tsx +7 -7
  22. package/templates/default/app/(rootLayout)/edit/[section]/[itemId]/page.tsx +17 -17
  23. package/templates/default/app/(rootLayout)/layout.tsx +81 -81
  24. package/templates/default/app/(rootLayout)/loading.tsx +10 -10
  25. package/templates/default/app/(rootLayout)/log/page.tsx +7 -7
  26. package/templates/default/app/(rootLayout)/new/[section]/page.tsx +15 -15
  27. package/templates/default/app/(rootLayout)/section/[section]/page.tsx +16 -16
  28. package/templates/default/app/(rootLayout)/settings/page.tsx +13 -13
  29. package/templates/default/app/_trpc/client.ts +3 -3
  30. package/templates/default/app/api/auth/csrf/route.ts +25 -25
  31. package/templates/default/app/api/auth/refresh/route.ts +10 -10
  32. package/templates/default/app/api/auth/route.ts +49 -49
  33. package/templates/default/app/api/auth/session/route.ts +20 -20
  34. package/templates/default/app/api/document/route.ts +165 -165
  35. package/templates/default/app/api/editor/photo/route.ts +49 -49
  36. package/templates/default/app/api/photo/route.ts +27 -27
  37. package/templates/default/app/api/submit/section/item/[slug]/route.ts +66 -66
  38. package/templates/default/app/api/submit/section/item/route.ts +56 -56
  39. package/templates/default/app/api/submit/section/simple/route.ts +57 -57
  40. package/templates/default/app/api/trpc/[trpc]/route.ts +33 -33
  41. package/templates/default/app/api/video/route.ts +174 -174
  42. package/templates/default/app/globals.css +219 -219
  43. package/templates/default/app/providers.tsx +152 -152
  44. package/templates/default/cms.config.ts +57 -60
  45. package/templates/default/components/AdminCard.tsx +166 -166
  46. package/templates/default/components/AdminEditPage.tsx +124 -124
  47. package/templates/default/components/AdminPrivilegeCard.tsx +185 -185
  48. package/templates/default/components/AdminsPage.tsx +43 -43
  49. package/templates/default/components/AnalyticsPage.tsx +128 -128
  50. package/templates/default/components/BarChartBox.tsx +42 -42
  51. package/templates/default/components/BrowsePage.tsx +106 -106
  52. package/templates/default/components/CategorizedSectionPage.tsx +31 -31
  53. package/templates/default/components/CategoryDeleteConfirmPage.tsx +130 -130
  54. package/templates/default/components/CategorySectionSelectInput.tsx +140 -140
  55. package/templates/default/components/ConditionalFields.tsx +49 -49
  56. package/templates/default/components/ContainerBox.tsx +24 -24
  57. package/templates/default/components/DashboardNewPage.tsx +253 -253
  58. package/templates/default/components/DashboardPage.tsx +188 -188
  59. package/templates/default/components/DashboardPageAlt.tsx +45 -45
  60. package/templates/default/components/DefaultNavItems.tsx +3 -3
  61. package/templates/default/components/Dropzone.tsx +154 -154
  62. package/templates/default/components/EmailCard.tsx +138 -138
  63. package/templates/default/components/EmailPasswordForm.tsx +85 -85
  64. package/templates/default/components/EmailQuotaForm.tsx +73 -73
  65. package/templates/default/components/EmailsPage.tsx +49 -49
  66. package/templates/default/components/ErrorComponent.tsx +16 -16
  67. package/templates/default/components/GalleryPhoto.tsx +93 -93
  68. package/templates/default/components/InfoCard.tsx +93 -93
  69. package/templates/default/components/ItemEditPage.tsx +214 -214
  70. package/templates/default/components/Layout.tsx +84 -84
  71. package/templates/default/components/LoadingSpinners.tsx +67 -67
  72. package/templates/default/components/LogPage.tsx +107 -107
  73. package/templates/default/components/Modal.tsx +166 -166
  74. package/templates/default/components/Navbar.tsx +258 -258
  75. package/templates/default/components/NewAdminForm.tsx +173 -173
  76. package/templates/default/components/NewEmailForm.tsx +132 -132
  77. package/templates/default/components/NewPage.tsx +205 -205
  78. package/templates/default/components/NewVariantComponent.tsx +229 -229
  79. package/templates/default/components/PhotoGallery.tsx +35 -35
  80. package/templates/default/components/PieChartBox.tsx +101 -101
  81. package/templates/default/components/ProgressBar.tsx +48 -48
  82. package/templates/default/components/ProtectedDocument.tsx +44 -44
  83. package/templates/default/components/ProtectedImage.tsx +143 -143
  84. package/templates/default/components/ProtectedVideo.tsx +76 -76
  85. package/templates/default/components/SectionIcon.tsx +8 -8
  86. package/templates/default/components/SectionItemCard.tsx +144 -144
  87. package/templates/default/components/SectionItemStatusBadge.tsx +17 -17
  88. package/templates/default/components/SectionPage.tsx +125 -125
  89. package/templates/default/components/SelectBox.tsx +98 -98
  90. package/templates/default/components/SelectInputButtons.tsx +125 -125
  91. package/templates/default/components/SettingsPage.tsx +232 -232
  92. package/templates/default/components/Sidebar.tsx +204 -204
  93. package/templates/default/components/SidebarDropdownItem.tsx +83 -83
  94. package/templates/default/components/SidebarItem.tsx +24 -24
  95. package/templates/default/components/ThemeProvider.tsx +8 -8
  96. package/templates/default/components/TooltipComponent.tsx +27 -27
  97. package/templates/default/components/VariantCard.tsx +124 -124
  98. package/templates/default/components/VariantEditPage.tsx +230 -230
  99. package/templates/default/components/analytics/BounceRate.tsx +70 -70
  100. package/templates/default/components/analytics/LivePageViews.tsx +55 -55
  101. package/templates/default/components/analytics/LiveUsersCount.tsx +33 -33
  102. package/templates/default/components/analytics/MonthlyPageViews.tsx +42 -42
  103. package/templates/default/components/analytics/TopCountries.tsx +52 -52
  104. package/templates/default/components/analytics/TopDevices.tsx +46 -46
  105. package/templates/default/components/analytics/TopMediums.tsx +58 -58
  106. package/templates/default/components/analytics/TopSources.tsx +45 -45
  107. package/templates/default/components/analytics/TotalPageViews.tsx +41 -41
  108. package/templates/default/components/analytics/TotalSessions.tsx +41 -41
  109. package/templates/default/components/analytics/TotalUniqueUsers.tsx +41 -41
  110. package/templates/default/components/custom/RightHomeRoomVariantCard.tsx +138 -138
  111. package/templates/default/components/dndKit/Draggable.tsx +21 -21
  112. package/templates/default/components/dndKit/Droppable.tsx +20 -20
  113. package/templates/default/components/dndKit/SortableItem.tsx +18 -18
  114. package/templates/default/components/form/DateRangeFormInput.tsx +57 -57
  115. package/templates/default/components/form/Form.tsx +317 -317
  116. package/templates/default/components/form/FormInputElement.tsx +70 -70
  117. package/templates/default/components/form/FormInputs.tsx +118 -118
  118. package/templates/default/components/form/helpers/_section-hot-reload.js +1 -1
  119. package/templates/default/components/form/helpers/util.ts +17 -17
  120. package/templates/default/components/form/inputs/CheckboxFormInput.tsx +46 -46
  121. package/templates/default/components/form/inputs/ColorFormInput.tsx +44 -44
  122. package/templates/default/components/form/inputs/DateFormInput.tsx +156 -156
  123. package/templates/default/components/form/inputs/DocumentFormInput.tsx +222 -222
  124. package/templates/default/components/form/inputs/MapFormInput.tsx +140 -140
  125. package/templates/default/components/form/inputs/MultipleSelectFormInput.tsx +85 -85
  126. package/templates/default/components/form/inputs/NumberFormInput.tsx +42 -42
  127. package/templates/default/components/form/inputs/PasswordFormInput.tsx +47 -47
  128. package/templates/default/components/form/inputs/PhotoFormInput.tsx +219 -219
  129. package/templates/default/components/form/inputs/RichTextFormInput.tsx +135 -135
  130. package/templates/default/components/form/inputs/SelectFormInput.tsx +175 -175
  131. package/templates/default/components/form/inputs/SlugFormInput.tsx +131 -131
  132. package/templates/default/components/form/inputs/TagsFormInput.tsx +260 -260
  133. package/templates/default/components/form/inputs/TextFormInput.tsx +48 -48
  134. package/templates/default/components/form/inputs/TextareaFormInput.tsx +47 -47
  135. package/templates/default/components/form/inputs/VideoFormInput.tsx +118 -118
  136. package/templates/default/components/locale-dropdown.tsx +74 -74
  137. package/templates/default/components/locale-picker.tsx +85 -85
  138. package/templates/default/components/login-locale-dropdown.tsx +46 -46
  139. package/templates/default/components/multi-select.tsx +1146 -1146
  140. package/templates/default/components/pagination/Pagination.tsx +36 -36
  141. package/templates/default/components/pagination/PaginationButtons.tsx +147 -147
  142. package/templates/default/components/theme-toggle.tsx +39 -39
  143. package/templates/default/components/ui/accordion.tsx +53 -53
  144. package/templates/default/components/ui/alert-dialog.tsx +157 -157
  145. package/templates/default/components/ui/alert.tsx +46 -46
  146. package/templates/default/components/ui/badge.tsx +38 -38
  147. package/templates/default/components/ui/button.tsx +62 -62
  148. package/templates/default/components/ui/calendar.tsx +166 -166
  149. package/templates/default/components/ui/card.tsx +43 -43
  150. package/templates/default/components/ui/checkbox.tsx +29 -29
  151. package/templates/default/components/ui/command.tsx +137 -137
  152. package/templates/default/components/ui/custom-alert-dialog.tsx +113 -113
  153. package/templates/default/components/ui/custom-dialog.tsx +123 -123
  154. package/templates/default/components/ui/dialog.tsx +123 -123
  155. package/templates/default/components/ui/direction.tsx +22 -22
  156. package/templates/default/components/ui/dropdown-menu.tsx +182 -182
  157. package/templates/default/components/ui/input-group.tsx +54 -54
  158. package/templates/default/components/ui/input.tsx +22 -22
  159. package/templates/default/components/ui/label.tsx +19 -19
  160. package/templates/default/components/ui/popover.tsx +42 -42
  161. package/templates/default/components/ui/progress.tsx +31 -31
  162. package/templates/default/components/ui/scroll-area.tsx +42 -42
  163. package/templates/default/components/ui/select.tsx +165 -165
  164. package/templates/default/components/ui/separator.tsx +28 -28
  165. package/templates/default/components/ui/sheet.tsx +103 -103
  166. package/templates/default/components/ui/spinner.tsx +16 -16
  167. package/templates/default/components/ui/switch.tsx +29 -29
  168. package/templates/default/components/ui/table.tsx +83 -83
  169. package/templates/default/components/ui/tabs.tsx +55 -55
  170. package/templates/default/components/ui/toast.tsx +113 -113
  171. package/templates/default/components/ui/toaster.tsx +35 -35
  172. package/templates/default/components/ui/tooltip.tsx +30 -30
  173. package/templates/default/components/ui/use-toast.ts +188 -188
  174. package/templates/default/components.json +21 -21
  175. package/templates/default/context/ModalProvider.tsx +53 -53
  176. package/templates/default/drizzle.config.ts +4 -4
  177. package/templates/default/dynamic-schemas/schema.ts +84 -19
  178. package/templates/default/env/env.js +130 -130
  179. package/templates/default/envConfig.ts +4 -4
  180. package/templates/default/hooks/useModal.ts +8 -8
  181. package/templates/default/lib/apiHelpers.ts +92 -92
  182. package/templates/default/lib/postinstall.js +14 -14
  183. package/templates/default/lib/utils.ts +6 -6
  184. package/templates/default/next-env.d.ts +6 -6
  185. package/templates/default/next.config.ts +23 -23
  186. package/templates/default/package.json +1 -1
  187. package/templates/default/postcss.config.mjs +6 -6
  188. package/templates/default/proxy.ts +32 -32
  189. package/templates/default/tsconfig.json +48 -48
@@ -1,317 +1,317 @@
1
- export const revalidate = 1
2
-
3
- import ContainerBox from '@/components/ContainerBox'
4
- import { useI18n } from 'nextjs-cms/translations/client'
5
- import FormInputs from '@/components/form/FormInputs'
6
- import Dropzone, { DropzoneHandles } from '@/components/Dropzone'
7
- import NewVariantComponent, { VariantHandles } from '@/components/NewVariantComponent'
8
- import classNames from 'classnames'
9
- import ProgressBar from '@/components/ProgressBar'
10
- import React, { RefObject, useEffect } from 'react'
11
- import type { RouterOutputs } from 'nextjs-cms/api'
12
- import * as z from 'zod'
13
- import { zodResolver } from '@hookform/resolvers/zod'
14
- import { useForm, FormProvider } from 'react-hook-form'
15
- import {
16
- CheckboxFieldClientConfig,
17
- ColorFieldClientConfig,
18
- DateFieldClientConfig,
19
- DocumentFieldClientConfig,
20
- MapFieldClientConfig,
21
- NumberFieldClientConfig,
22
- PasswordFieldClientConfig,
23
- PhotoFieldClientConfig,
24
- RichTextFieldClientConfig,
25
- SelectFieldClientConfig,
26
- SelectMultipleFieldClientConfig,
27
- SlugFieldClientConfig,
28
- TextAreaFieldClientConfig,
29
- TextFieldClientConfig,
30
- VideoFieldClientConfig,
31
- } from 'nextjs-cms/core/fields'
32
-
33
- import {
34
- numberFieldSchema,
35
- textFieldSchema,
36
- selectFieldSchema,
37
- selectMultipleFieldSchema,
38
- dateFieldSchema,
39
- checkboxFieldSchema,
40
- textareaFieldSchema,
41
- richTextFieldSchema,
42
- photoFieldSchema,
43
- documentFieldSchema,
44
- videoFieldSchema,
45
- colorFieldSchema,
46
- mapFieldSchema,
47
- passwordFieldSchema,
48
- slugFieldSchema,
49
- } from 'nextjs-cms/validators'
50
-
51
- import { ConditionalField, FieldType } from 'nextjs-cms/core/types'
52
- import { configLastUpdated } from '@/components/form/helpers/util'
53
- import PhotoGallery from '../PhotoGallery'
54
- import { useSession } from 'nextjs-cms/auth/react'
55
-
56
- export default function Form({
57
- formType,
58
- data,
59
- dropzoneRef,
60
- variantRef,
61
- handleSubmit,
62
- isSubmitting,
63
- response,
64
- progress,
65
- progressVariant,
66
- buttonType = 'big',
67
- }: {
68
- formType?: 'new' | 'edit'
69
- data:
70
- | RouterOutputs['hasItemsSections']['newItem']
71
- | RouterOutputs['hasItemsSections']['editItem']
72
- | RouterOutputs['simpleSections']['create']
73
- | {
74
- section: {
75
- name: string
76
- gallery?: boolean
77
- }
78
- inputGroups:
79
- | {
80
- groupId: number | undefined
81
- groupTitle: string
82
- groupOrder: number
83
- inputs: {
84
- type: FieldType
85
- name: string
86
- label: string
87
- required: boolean
88
- conditionalFields: ConditionalField[]
89
- placeholder?: string
90
- readonly: boolean
91
- value: any
92
- }[]
93
- }[]
94
- | undefined
95
- }
96
- dropzoneRef?: RefObject<DropzoneHandles | null>
97
- variantRef?: RefObject<VariantHandles[]>
98
- handleSubmit: any
99
- isSubmitting: boolean
100
- response?: any
101
- progress?: number
102
- progressVariant?: 'determinate' | 'query'
103
- buttonType?: 'big' | 'small'
104
- }) {
105
- const t = useI18n()
106
- const session = useSession()
107
- const locale = session?.data?.user?.locale
108
-
109
- let schema = z.object({})
110
- /**
111
- * Construct the schema for the form
112
- */
113
- data.inputGroups?.forEach((inputGroup) => {
114
- inputGroup.inputs.forEach((input) => {
115
- if (input.readonly) return
116
- switch (input.type) {
117
- case 'select_multiple':
118
- schema = schema.extend({
119
- [input.name]: selectMultipleFieldSchema(input as SelectMultipleFieldClientConfig, locale),
120
- })
121
- break
122
- case 'select':
123
- schema = schema.extend({
124
- [input.name]: selectFieldSchema(input as SelectFieldClientConfig, locale),
125
- })
126
- break
127
-
128
- case 'date':
129
- schema = schema.extend({
130
- [input.name]: dateFieldSchema(input as DateFieldClientConfig, locale),
131
- })
132
- break
133
-
134
- case 'checkbox':
135
- schema = schema.extend({
136
- [input.name]: checkboxFieldSchema(input as CheckboxFieldClientConfig, locale),
137
- })
138
- break
139
-
140
- case 'text':
141
- schema = schema.extend({
142
- [input.name]: textFieldSchema(input as TextFieldClientConfig, locale),
143
- })
144
- break
145
-
146
- case 'textarea':
147
- schema = schema.extend({
148
- [input.name]: textareaFieldSchema(input as TextAreaFieldClientConfig, locale),
149
- })
150
- break
151
-
152
- case 'rich_text':
153
- schema = schema.extend({
154
- [input.name]: richTextFieldSchema(input as RichTextFieldClientConfig, locale),
155
- })
156
- break
157
-
158
- case 'photo':
159
- if (formType === 'edit' && !!input.value) break
160
- schema = schema.extend({
161
- [input.name]: photoFieldSchema(input as PhotoFieldClientConfig, locale),
162
- })
163
- break
164
-
165
- case 'document':
166
- if (formType === 'edit' && !!input.value) break
167
- schema = schema.extend({
168
- [input.name]: documentFieldSchema(input as DocumentFieldClientConfig, locale),
169
- })
170
- break
171
-
172
- case 'video':
173
- if (formType === 'edit' && !!input.value) break
174
- schema = schema.extend({
175
- [input.name]: videoFieldSchema(input as VideoFieldClientConfig, locale),
176
- })
177
- break
178
-
179
- case 'number':
180
- schema = schema.extend({
181
- [input.name]: numberFieldSchema(input as NumberFieldClientConfig, locale),
182
- })
183
- break
184
- case 'color':
185
- schema = schema.extend({
186
- [input.name]: colorFieldSchema(input as ColorFieldClientConfig, locale),
187
- })
188
- break
189
-
190
- case 'map':
191
- schema = schema.extend({
192
- [input.name]: mapFieldSchema(input as MapFieldClientConfig, locale),
193
- })
194
- break
195
-
196
- case 'password':
197
- schema = schema.extend({
198
- [input.name]: passwordFieldSchema(input as PasswordFieldClientConfig, locale),
199
- })
200
- break
201
-
202
- case 'slug':
203
- schema = schema.extend({
204
- [input.name]: slugFieldSchema(input as SlugFieldClientConfig, locale),
205
- })
206
- break
207
- }
208
- })
209
- })
210
-
211
- const methods = useForm({
212
- resolver: zodResolver(schema),
213
- })
214
-
215
- return (
216
- <FormProvider {...methods}>
217
- <form
218
- method='post'
219
- onSubmit={(e) => {
220
- e.preventDefault()
221
- // e.stopPropagation()
222
- methods.handleSubmit((data) => {
223
- handleSubmit(new FormData(e.target as HTMLFormElement))
224
- })(e)
225
- }}
226
- encType='multipart/form-data'
227
- >
228
- <div className='w-full' data-section-schema-version={configLastUpdated}>
229
- {/*<ContainerBox title={formType ? t(formType === 'new' ? 'add_new' : 'edit') : undefined}>*/}
230
- <div className='p-4'>
231
- <div className=''>
232
- {data.inputGroups ? (
233
- <div className='flex flex-col gap-4'>
234
- {data.inputGroups.length > 0
235
- ? data.inputGroups.map((inputGroup, index: number) => {
236
- return (
237
- <ContainerBox title={inputGroup.groupTitle} key={index}>
238
- <FormInputs
239
- inputs={inputGroup.inputs}
240
- sectionName={data.section.name}
241
- />
242
- </ContainerBox>
243
- )
244
- })
245
- : null}
246
- {data.section.gallery ? (
247
- <>
248
- <div className='w-full'>
249
- <PhotoGallery sectionName={data.section.name} gallery={data.gallery} />
250
- </div>
251
- <div className='w-full'>
252
- <Dropzone ref={dropzoneRef} />
253
- </div>
254
- </>
255
- ) : null}
256
-
257
- {/*{data.section.variants && data.section.variants.length > 0 ? (
258
- <div className='w-full'>
259
- <div className='flex flex-col gap-4'>
260
- {data.section.variants.map((variant, index) => {
261
- // Only one variant is allowed for now
262
- // I have to find a way to make multiple variantRef in order to handle multiple variants
263
- if (index > 0) return
264
-
265
- return (
266
- <NewVariantComponent
267
- ref={(el) => (variantRef.current[index] = el)}
268
- key={index}
269
- section={section}
270
- variantInfo={variant.info}
271
- xsrfToken={xsrfToken}
272
- />
273
- )
274
- })}
275
- </div>
276
- </div>
277
- ) : null}*/}
278
-
279
- <div className='flex flex-col gap-3 pb-4'>
280
- <div className=''>
281
- <button
282
- className={classNames({
283
- 'w-full': buttonType === 'big',
284
- 'float-end': buttonType === 'small',
285
- 'rounded bg-linear-to-r px-4 py-2 font-bold text-white drop-shadow-sm':
286
- true,
287
- 'from-emerald-700 via-green-700 to-green-500 dark:from-blue-800 dark:via-sky-800 dark:to-slate-500':
288
- !isSubmitting,
289
- 'from-gray-600 via-gray-500 to-gray-400': isSubmitting,
290
- })}
291
- type='submit'
292
- disabled={isSubmitting}
293
- >
294
- {isSubmitting
295
- ? t('loading')
296
- : t(formType === 'new' ? 'create' : 'save')}
297
- </button>
298
- {progressVariant && progress ? (
299
- isSubmitting ? (
300
- <div className='mt-0.5'>
301
- <ProgressBar variant={progressVariant} value={progress} />
302
- </div>
303
- ) : null
304
- ) : null}
305
- </div>
306
- {response ? <div className='w-full'>{response}</div> : null}
307
- </div>
308
- </div>
309
- ) : null}
310
- </div>
311
- </div>
312
- {/*</ContainerBox>*/}
313
- </div>
314
- </form>
315
- </FormProvider>
316
- )
317
- }
1
+ export const revalidate = 1
2
+
3
+ import ContainerBox from '@/components/ContainerBox'
4
+ import { useI18n } from 'nextjs-cms/translations/client'
5
+ import FormInputs from '@/components/form/FormInputs'
6
+ import Dropzone, { DropzoneHandles } from '@/components/Dropzone'
7
+ import NewVariantComponent, { VariantHandles } from '@/components/NewVariantComponent'
8
+ import classNames from 'classnames'
9
+ import ProgressBar from '@/components/ProgressBar'
10
+ import React, { RefObject, useEffect } from 'react'
11
+ import type { RouterOutputs } from 'nextjs-cms/api'
12
+ import * as z from 'zod'
13
+ import { zodResolver } from '@hookform/resolvers/zod'
14
+ import { useForm, FormProvider } from 'react-hook-form'
15
+ import {
16
+ CheckboxFieldClientConfig,
17
+ ColorFieldClientConfig,
18
+ DateFieldClientConfig,
19
+ DocumentFieldClientConfig,
20
+ MapFieldClientConfig,
21
+ NumberFieldClientConfig,
22
+ PasswordFieldClientConfig,
23
+ PhotoFieldClientConfig,
24
+ RichTextFieldClientConfig,
25
+ SelectFieldClientConfig,
26
+ SelectMultipleFieldClientConfig,
27
+ SlugFieldClientConfig,
28
+ TextAreaFieldClientConfig,
29
+ TextFieldClientConfig,
30
+ VideoFieldClientConfig,
31
+ } from 'nextjs-cms/core/fields'
32
+
33
+ import {
34
+ numberFieldSchema,
35
+ textFieldSchema,
36
+ selectFieldSchema,
37
+ selectMultipleFieldSchema,
38
+ dateFieldSchema,
39
+ checkboxFieldSchema,
40
+ textareaFieldSchema,
41
+ richTextFieldSchema,
42
+ photoFieldSchema,
43
+ documentFieldSchema,
44
+ videoFieldSchema,
45
+ colorFieldSchema,
46
+ mapFieldSchema,
47
+ passwordFieldSchema,
48
+ slugFieldSchema,
49
+ } from 'nextjs-cms/validators'
50
+
51
+ import { ConditionalField, FieldType } from 'nextjs-cms/core/types'
52
+ import { configLastUpdated } from '@/components/form/helpers/util'
53
+ import PhotoGallery from '../PhotoGallery'
54
+ import { useSession } from 'nextjs-cms/auth/react'
55
+
56
+ export default function Form({
57
+ formType,
58
+ data,
59
+ dropzoneRef,
60
+ variantRef,
61
+ handleSubmit,
62
+ isSubmitting,
63
+ response,
64
+ progress,
65
+ progressVariant,
66
+ buttonType = 'big',
67
+ }: {
68
+ formType?: 'new' | 'edit'
69
+ data:
70
+ | RouterOutputs['hasItemsSections']['newItem']
71
+ | RouterOutputs['hasItemsSections']['editItem']
72
+ | RouterOutputs['simpleSections']['create']
73
+ | {
74
+ section: {
75
+ name: string
76
+ gallery?: boolean
77
+ }
78
+ inputGroups:
79
+ | {
80
+ groupId: number | undefined
81
+ groupTitle: string
82
+ groupOrder: number
83
+ inputs: {
84
+ type: FieldType
85
+ name: string
86
+ label: string
87
+ required: boolean
88
+ conditionalFields: ConditionalField[]
89
+ placeholder?: string
90
+ readonly: boolean
91
+ value: any
92
+ }[]
93
+ }[]
94
+ | undefined
95
+ }
96
+ dropzoneRef?: RefObject<DropzoneHandles | null>
97
+ variantRef?: RefObject<VariantHandles[]>
98
+ handleSubmit: any
99
+ isSubmitting: boolean
100
+ response?: any
101
+ progress?: number
102
+ progressVariant?: 'determinate' | 'query'
103
+ buttonType?: 'big' | 'small'
104
+ }) {
105
+ const t = useI18n()
106
+ const session = useSession()
107
+ const locale = session?.data?.user?.locale
108
+
109
+ let schema = z.object({})
110
+ /**
111
+ * Construct the schema for the form
112
+ */
113
+ data.inputGroups?.forEach((inputGroup) => {
114
+ inputGroup.inputs.forEach((input) => {
115
+ if (input.readonly) return
116
+ switch (input.type) {
117
+ case 'select_multiple':
118
+ schema = schema.extend({
119
+ [input.name]: selectMultipleFieldSchema(input as SelectMultipleFieldClientConfig, locale),
120
+ })
121
+ break
122
+ case 'select':
123
+ schema = schema.extend({
124
+ [input.name]: selectFieldSchema(input as SelectFieldClientConfig, locale),
125
+ })
126
+ break
127
+
128
+ case 'date':
129
+ schema = schema.extend({
130
+ [input.name]: dateFieldSchema(input as DateFieldClientConfig, locale),
131
+ })
132
+ break
133
+
134
+ case 'checkbox':
135
+ schema = schema.extend({
136
+ [input.name]: checkboxFieldSchema(input as CheckboxFieldClientConfig, locale),
137
+ })
138
+ break
139
+
140
+ case 'text':
141
+ schema = schema.extend({
142
+ [input.name]: textFieldSchema(input as TextFieldClientConfig, locale),
143
+ })
144
+ break
145
+
146
+ case 'textarea':
147
+ schema = schema.extend({
148
+ [input.name]: textareaFieldSchema(input as TextAreaFieldClientConfig, locale),
149
+ })
150
+ break
151
+
152
+ case 'rich_text':
153
+ schema = schema.extend({
154
+ [input.name]: richTextFieldSchema(input as RichTextFieldClientConfig, locale),
155
+ })
156
+ break
157
+
158
+ case 'photo':
159
+ if (formType === 'edit' && !!input.value) break
160
+ schema = schema.extend({
161
+ [input.name]: photoFieldSchema(input as PhotoFieldClientConfig, locale),
162
+ })
163
+ break
164
+
165
+ case 'document':
166
+ if (formType === 'edit' && !!input.value) break
167
+ schema = schema.extend({
168
+ [input.name]: documentFieldSchema(input as DocumentFieldClientConfig, locale),
169
+ })
170
+ break
171
+
172
+ case 'video':
173
+ if (formType === 'edit' && !!input.value) break
174
+ schema = schema.extend({
175
+ [input.name]: videoFieldSchema(input as VideoFieldClientConfig, locale),
176
+ })
177
+ break
178
+
179
+ case 'number':
180
+ schema = schema.extend({
181
+ [input.name]: numberFieldSchema(input as NumberFieldClientConfig, locale),
182
+ })
183
+ break
184
+ case 'color':
185
+ schema = schema.extend({
186
+ [input.name]: colorFieldSchema(input as ColorFieldClientConfig, locale),
187
+ })
188
+ break
189
+
190
+ case 'map':
191
+ schema = schema.extend({
192
+ [input.name]: mapFieldSchema(input as MapFieldClientConfig, locale),
193
+ })
194
+ break
195
+
196
+ case 'password':
197
+ schema = schema.extend({
198
+ [input.name]: passwordFieldSchema(input as PasswordFieldClientConfig, locale),
199
+ })
200
+ break
201
+
202
+ case 'slug':
203
+ schema = schema.extend({
204
+ [input.name]: slugFieldSchema(input as SlugFieldClientConfig, locale),
205
+ })
206
+ break
207
+ }
208
+ })
209
+ })
210
+
211
+ const methods = useForm({
212
+ resolver: zodResolver(schema),
213
+ })
214
+
215
+ return (
216
+ <FormProvider {...methods}>
217
+ <form
218
+ method='post'
219
+ onSubmit={(e) => {
220
+ e.preventDefault()
221
+ // e.stopPropagation()
222
+ methods.handleSubmit((data) => {
223
+ handleSubmit(new FormData(e.target as HTMLFormElement))
224
+ })(e)
225
+ }}
226
+ encType='multipart/form-data'
227
+ >
228
+ <div className='w-full' data-section-schema-version={configLastUpdated}>
229
+ {/*<ContainerBox title={formType ? t(formType === 'new' ? 'add_new' : 'edit') : undefined}>*/}
230
+ <div className='p-4'>
231
+ <div className=''>
232
+ {data.inputGroups ? (
233
+ <div className='flex flex-col gap-4'>
234
+ {data.inputGroups.length > 0
235
+ ? data.inputGroups.map((inputGroup, index: number) => {
236
+ return (
237
+ <ContainerBox title={inputGroup.groupTitle} key={index}>
238
+ <FormInputs
239
+ inputs={inputGroup.inputs}
240
+ sectionName={data.section.name}
241
+ />
242
+ </ContainerBox>
243
+ )
244
+ })
245
+ : null}
246
+ {data.section.gallery ? (
247
+ <>
248
+ <div className='w-full'>
249
+ <PhotoGallery sectionName={data.section.name} gallery={data.gallery} />
250
+ </div>
251
+ <div className='w-full'>
252
+ <Dropzone ref={dropzoneRef} />
253
+ </div>
254
+ </>
255
+ ) : null}
256
+
257
+ {/*{data.section.variants && data.section.variants.length > 0 ? (
258
+ <div className='w-full'>
259
+ <div className='flex flex-col gap-4'>
260
+ {data.section.variants.map((variant, index) => {
261
+ // Only one variant is allowed for now
262
+ // I have to find a way to make multiple variantRef in order to handle multiple variants
263
+ if (index > 0) return
264
+
265
+ return (
266
+ <NewVariantComponent
267
+ ref={(el) => (variantRef.current[index] = el)}
268
+ key={index}
269
+ section={section}
270
+ variantInfo={variant.info}
271
+ xsrfToken={xsrfToken}
272
+ />
273
+ )
274
+ })}
275
+ </div>
276
+ </div>
277
+ ) : null}*/}
278
+
279
+ <div className='flex flex-col gap-3 pb-4'>
280
+ <div className=''>
281
+ <button
282
+ className={classNames({
283
+ 'w-full': buttonType === 'big',
284
+ 'float-end': buttonType === 'small',
285
+ 'rounded bg-linear-to-r px-4 py-2 font-bold text-white drop-shadow-sm':
286
+ true,
287
+ 'from-emerald-700 via-green-700 to-green-500 dark:from-blue-800 dark:via-sky-800 dark:to-slate-500':
288
+ !isSubmitting,
289
+ 'from-gray-600 via-gray-500 to-gray-400': isSubmitting,
290
+ })}
291
+ type='submit'
292
+ disabled={isSubmitting}
293
+ >
294
+ {isSubmitting
295
+ ? t('loading')
296
+ : t(formType === 'new' ? 'create' : 'save')}
297
+ </button>
298
+ {progressVariant && progress ? (
299
+ isSubmitting ? (
300
+ <div className='mt-0.5'>
301
+ <ProgressBar variant={progressVariant} value={progress} />
302
+ </div>
303
+ ) : null
304
+ ) : null}
305
+ </div>
306
+ {response ? <div className='w-full'>{response}</div> : null}
307
+ </div>
308
+ </div>
309
+ ) : null}
310
+ </div>
311
+ </div>
312
+ {/*</ContainerBox>*/}
313
+ </div>
314
+ </form>
315
+ </FormProvider>
316
+ )
317
+ }