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,131 +1,131 @@
1
- 'use client'
2
-
3
- import React, { useCallback, useEffect, useRef } from 'react'
4
- import FormInputElement from '@/components/form/form-input-element'
5
- import type { SlugFieldClientConfig } from 'nextjs-cms/core/fields'
6
- import { useFormContext, useController, useWatch } from 'react-hook-form'
7
- import { InputGroup, InputGroupInput, InputGroupAddon } from '@/components/ui/input-group'
8
- import { Link2 } from 'lucide-react'
9
- import { useI18n } from 'nextjs-cms/translations/client'
10
-
11
- /**
12
- * Convert a string to a URL-friendly slug (for real-time input).
13
- * - Converts to lowercase
14
- * - Replaces spaces with hyphens
15
- * - Removes special characters (keeps letters from any language, numbers, and hyphens)
16
- * - Collapses multiple consecutive hyphens into one
17
- * - Keeps trailing hyphen (user might still be typing)
18
- */
19
- function toSlugLive(value: string): string {
20
- return value
21
- .toLowerCase()
22
- .replace(/\s+/g, '-') // Replace spaces with hyphens
23
- .replace(/[^\p{L}\p{N}-]/gu, '') // Remove special characters (keep Unicode letters/numbers)
24
- .replace(/-+/g, '-') // Collapse multiple hyphens
25
- .replace(/^-/, '') // Remove leading hyphen only
26
- }
27
-
28
- /**
29
- * Normalize a slug (for blur/final value).
30
- * Same as toSlugLive but also trims and removes trailing hyphens.
31
- */
32
- function toSlugFinal(value: string): string {
33
- return toSlugLive(value.trim()).replace(/-$/, '') // Remove trailing hyphen
34
- }
35
-
36
- export default function SlugFormInput({
37
- input,
38
- direction,
39
- disabled = false,
40
- }: {
41
- input: SlugFieldClientConfig
42
- direction?: 'row' | 'col'
43
- disabled?: boolean
44
- }) {
45
- const t = useI18n()
46
- const { control } = useFormContext()
47
- const {
48
- field,
49
- fieldState: { error },
50
- } = useController({
51
- name: input.name,
52
- control,
53
- defaultValue: input.value ?? '',
54
- disabled: disabled,
55
- })
56
-
57
- // Watch the source field value
58
- const sourceFieldValue = useWatch({
59
- control,
60
- name: input.forFieldName,
61
- defaultValue: '',
62
- })
63
-
64
- // Track previous source value to only update when it actually changes
65
- const previousSourceValueRef = useRef<string>(sourceFieldValue ?? '')
66
-
67
- // Auto-generate slug from source field when it changes
68
- useEffect(() => {
69
- const currentSourceValue = sourceFieldValue ?? ''
70
-
71
- // Only update if the source field value actually changed
72
- if (currentSourceValue !== previousSourceValueRef.current) {
73
- previousSourceValueRef.current = currentSourceValue
74
-
75
- if (currentSourceValue) {
76
- const newSlug = toSlugFinal(currentSourceValue)
77
- field.onChange(newSlug)
78
- } else {
79
- field.onChange('')
80
- }
81
- }
82
- }, [sourceFieldValue, field])
83
-
84
- // Handle manual changes to the slug field
85
- const handleChange = useCallback(
86
- (e: React.ChangeEvent<HTMLInputElement>) => {
87
- const newValue = e.target.value
88
- // Apply slug normalization in real-time
89
- field.onChange(toSlugLive(newValue))
90
- },
91
- [field],
92
- )
93
-
94
- // Handle blur - finalize the slug value (remove trailing hyphens)
95
- const handleBlur = useCallback(() => {
96
- field.onBlur()
97
- if (field.value) {
98
- const finalSlug = toSlugFinal(field.value)
99
- if (finalSlug !== field.value) {
100
- field.onChange(finalSlug)
101
- }
102
- }
103
- }, [field])
104
-
105
- return (
106
- <FormInputElement
107
- validationError={error}
108
- value={input.value}
109
- readonly={input.readonly}
110
- label={input.label}
111
- required={input.required}
112
- >
113
- <InputGroup className='bg-input'>
114
- <InputGroupAddon align='inline-start' title={t('autoGeneratedFromLinkedField') as string}>
115
- <Link2 className='h-4 w-4' />
116
- </InputGroupAddon>
117
- <InputGroupInput
118
- placeholder={input.placeholder ? input.placeholder : input.label}
119
- type='text'
120
- readOnly={disabled}
121
- disabled={field.disabled}
122
- name={field.name}
123
- onChange={handleChange}
124
- onBlur={handleBlur}
125
- value={field.value ?? ''}
126
- ref={field.ref}
127
- />
128
- </InputGroup>
129
- </FormInputElement>
130
- )
131
- }
1
+ 'use client'
2
+
3
+ import React, { useCallback, useEffect, useRef } from 'react'
4
+ import FormInputElement from '@/components/form/form-input-element'
5
+ import type { SlugFieldClientConfig } from 'nextjs-cms/core/fields'
6
+ import { useFormContext, useController, useWatch } from 'react-hook-form'
7
+ import { InputGroup, InputGroupInput, InputGroupAddon } from '@/components/ui/input-group'
8
+ import { Link2 } from 'lucide-react'
9
+ import { useI18n } from 'nextjs-cms/translations/client'
10
+
11
+ /**
12
+ * Convert a string to a URL-friendly slug (for real-time input).
13
+ * - Converts to lowercase
14
+ * - Replaces spaces with hyphens
15
+ * - Removes special characters (keeps letters from any language, numbers, and hyphens)
16
+ * - Collapses multiple consecutive hyphens into one
17
+ * - Keeps trailing hyphen (user might still be typing)
18
+ */
19
+ function toSlugLive(value: string): string {
20
+ return value
21
+ .toLowerCase()
22
+ .replace(/\s+/g, '-') // Replace spaces with hyphens
23
+ .replace(/[^\p{L}\p{N}-]/gu, '') // Remove special characters (keep Unicode letters/numbers)
24
+ .replace(/-+/g, '-') // Collapse multiple hyphens
25
+ .replace(/^-/, '') // Remove leading hyphen only
26
+ }
27
+
28
+ /**
29
+ * Normalize a slug (for blur/final value).
30
+ * Same as toSlugLive but also trims and removes trailing hyphens.
31
+ */
32
+ function toSlugFinal(value: string): string {
33
+ return toSlugLive(value.trim()).replace(/-$/, '') // Remove trailing hyphen
34
+ }
35
+
36
+ export default function SlugFormInput({
37
+ input,
38
+ direction,
39
+ disabled = false,
40
+ }: {
41
+ input: SlugFieldClientConfig
42
+ direction?: 'row' | 'col'
43
+ disabled?: boolean
44
+ }) {
45
+ const t = useI18n()
46
+ const { control } = useFormContext()
47
+ const {
48
+ field,
49
+ fieldState: { error },
50
+ } = useController({
51
+ name: input.name,
52
+ control,
53
+ defaultValue: input.value ?? '',
54
+ disabled: disabled,
55
+ })
56
+
57
+ // Watch the source field value
58
+ const sourceFieldValue = useWatch({
59
+ control,
60
+ name: input.forFieldName,
61
+ defaultValue: '',
62
+ })
63
+
64
+ // Track previous source value to only update when it actually changes
65
+ const previousSourceValueRef = useRef<string>(sourceFieldValue ?? '')
66
+
67
+ // Auto-generate slug from source field when it changes
68
+ useEffect(() => {
69
+ const currentSourceValue = sourceFieldValue ?? ''
70
+
71
+ // Only update if the source field value actually changed
72
+ if (currentSourceValue !== previousSourceValueRef.current) {
73
+ previousSourceValueRef.current = currentSourceValue
74
+
75
+ if (currentSourceValue) {
76
+ const newSlug = toSlugFinal(currentSourceValue)
77
+ field.onChange(newSlug)
78
+ } else {
79
+ field.onChange('')
80
+ }
81
+ }
82
+ }, [sourceFieldValue, field])
83
+
84
+ // Handle manual changes to the slug field
85
+ const handleChange = useCallback(
86
+ (e: React.ChangeEvent<HTMLInputElement>) => {
87
+ const newValue = e.target.value
88
+ // Apply slug normalization in real-time
89
+ field.onChange(toSlugLive(newValue))
90
+ },
91
+ [field],
92
+ )
93
+
94
+ // Handle blur - finalize the slug value (remove trailing hyphens)
95
+ const handleBlur = useCallback(() => {
96
+ field.onBlur()
97
+ if (field.value) {
98
+ const finalSlug = toSlugFinal(field.value)
99
+ if (finalSlug !== field.value) {
100
+ field.onChange(finalSlug)
101
+ }
102
+ }
103
+ }, [field])
104
+
105
+ return (
106
+ <FormInputElement
107
+ validationError={error}
108
+ value={input.value}
109
+ readonly={input.readonly}
110
+ label={input.label}
111
+ required={input.required}
112
+ >
113
+ <InputGroup className='bg-input'>
114
+ <InputGroupAddon align='inline-start' title={t('autoGeneratedFromLinkedField') as string}>
115
+ <Link2 className='h-4 w-4' />
116
+ </InputGroupAddon>
117
+ <InputGroupInput
118
+ placeholder={input.placeholder ? input.placeholder : input.label}
119
+ type='text'
120
+ readOnly={disabled}
121
+ disabled={field.disabled}
122
+ name={field.name}
123
+ onChange={handleChange}
124
+ onBlur={handleBlur}
125
+ value={field.value ?? ''}
126
+ ref={field.ref}
127
+ />
128
+ </InputGroup>
129
+ </FormInputElement>
130
+ )
131
+ }