azamat-ui-kit-cli 0.2.2

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 (213) hide show
  1. package/README.md +8 -0
  2. package/dist/index.js +432 -0
  3. package/package.json +34 -0
  4. package/vendor/package.json +4 -0
  5. package/vendor/src/components/actions/action-bar.tsx +35 -0
  6. package/vendor/src/components/actions/action-menu.tsx +120 -0
  7. package/vendor/src/components/actions/button-group.tsx +47 -0
  8. package/vendor/src/components/actions/copy-button.tsx +91 -0
  9. package/vendor/src/components/actions/copy-field.tsx +31 -0
  10. package/vendor/src/components/actions/floating-action-button.tsx +33 -0
  11. package/vendor/src/components/actions/index.ts +7 -0
  12. package/vendor/src/components/actions/public.ts +5 -0
  13. package/vendor/src/components/actions/quick-action-grid.tsx +162 -0
  14. package/vendor/src/components/calendar/calendar.tsx +328 -0
  15. package/vendor/src/components/calendar/date-picker.tsx +78 -0
  16. package/vendor/src/components/calendar/date-range-picker.tsx +96 -0
  17. package/vendor/src/components/calendar/date-utils.ts +89 -0
  18. package/vendor/src/components/calendar/index.ts +4 -0
  19. package/vendor/src/components/charts/charts.tsx +275 -0
  20. package/vendor/src/components/charts/horizontal-bar-chart.tsx +46 -0
  21. package/vendor/src/components/charts/index.ts +4 -0
  22. package/vendor/src/components/charts/kpi.tsx +68 -0
  23. package/vendor/src/components/charts/progress-ring.tsx +45 -0
  24. package/vendor/src/components/charts/public.ts +1 -0
  25. package/vendor/src/components/command/command-palette.tsx +375 -0
  26. package/vendor/src/components/command/index.ts +1 -0
  27. package/vendor/src/components/data-table/data-table-actions-column.tsx +58 -0
  28. package/vendor/src/components/data-table/data-table-bulk-actions.tsx +84 -0
  29. package/vendor/src/components/data-table/data-table-column-visibility-menu.tsx +79 -0
  30. package/vendor/src/components/data-table/data-table-pagination.tsx +91 -0
  31. package/vendor/src/components/data-table/data-table-row-actions.tsx +48 -0
  32. package/vendor/src/components/data-table/data-table-select-column.tsx +59 -0
  33. package/vendor/src/components/data-table/data-table-sortable-header.tsx +45 -0
  34. package/vendor/src/components/data-table/data-table-toolbar.tsx +76 -0
  35. package/vendor/src/components/data-table/data-table-view-presets.tsx +128 -0
  36. package/vendor/src/components/data-table/data-table.tsx +507 -0
  37. package/vendor/src/components/data-table/index.ts +12 -0
  38. package/vendor/src/components/data-table/public.ts +10 -0
  39. package/vendor/src/components/data-table/table-export-menu.tsx +56 -0
  40. package/vendor/src/components/data-table/table-import-button.tsx +43 -0
  41. package/vendor/src/components/display/activity-feed.tsx +97 -0
  42. package/vendor/src/components/display/avatar.tsx +131 -0
  43. package/vendor/src/components/display/code-block.tsx +33 -0
  44. package/vendor/src/components/display/data-state.tsx +63 -0
  45. package/vendor/src/components/display/description-list.tsx +119 -0
  46. package/vendor/src/components/display/descriptions.tsx +83 -0
  47. package/vendor/src/components/display/entity-card.tsx +53 -0
  48. package/vendor/src/components/display/file-card.tsx +54 -0
  49. package/vendor/src/components/display/index.ts +30 -0
  50. package/vendor/src/components/display/kanban.tsx +104 -0
  51. package/vendor/src/components/display/keyboard-shortcut.tsx +31 -0
  52. package/vendor/src/components/display/list.tsx +100 -0
  53. package/vendor/src/components/display/metric-grid.tsx +86 -0
  54. package/vendor/src/components/display/progress.tsx +162 -0
  55. package/vendor/src/components/display/property-grid.tsx +54 -0
  56. package/vendor/src/components/display/result.tsx +90 -0
  57. package/vendor/src/components/display/smart-card.tsx +168 -0
  58. package/vendor/src/components/display/statistic.tsx +107 -0
  59. package/vendor/src/components/display/status-legend.tsx +108 -0
  60. package/vendor/src/components/display/tag-list.tsx +52 -0
  61. package/vendor/src/components/display/timeline.tsx +132 -0
  62. package/vendor/src/components/display/tree-view.tsx +116 -0
  63. package/vendor/src/components/feedback/alert.tsx +69 -0
  64. package/vendor/src/components/feedback/empty-state.tsx +56 -0
  65. package/vendor/src/components/feedback/index.ts +5 -0
  66. package/vendor/src/components/feedback/loading-state.tsx +39 -0
  67. package/vendor/src/components/feedback/page-state.tsx +69 -0
  68. package/vendor/src/components/feedback/status-badge.tsx +62 -0
  69. package/vendor/src/components/filters/filter-bar.tsx +89 -0
  70. package/vendor/src/components/filters/filter-chips.tsx +69 -0
  71. package/vendor/src/components/filters/index.ts +2 -0
  72. package/vendor/src/components/form/form-actions.tsx +53 -0
  73. package/vendor/src/components/form/form-async-select.tsx +26 -0
  74. package/vendor/src/components/form/form-date-input.tsx +19 -0
  75. package/vendor/src/components/form/form-date-picker.tsx +54 -0
  76. package/vendor/src/components/form/form-date-range-input.tsx +79 -0
  77. package/vendor/src/components/form/form-date-range-picker.tsx +57 -0
  78. package/vendor/src/components/form/form-field-shell.tsx +191 -0
  79. package/vendor/src/components/form/form-input.tsx +480 -0
  80. package/vendor/src/components/form/form-number-input.tsx +19 -0
  81. package/vendor/src/components/form/form-password-input.tsx +19 -0
  82. package/vendor/src/components/form/form-phone-input.tsx +22 -0
  83. package/vendor/src/components/form/form-search-input.tsx +19 -0
  84. package/vendor/src/components/form/form-section.tsx +29 -0
  85. package/vendor/src/components/form/form-select.tsx +194 -0
  86. package/vendor/src/components/form/form-switch.tsx +145 -0
  87. package/vendor/src/components/form/form-textarea.tsx +103 -0
  88. package/vendor/src/components/form/index.ts +17 -0
  89. package/vendor/src/components/form/public.ts +14 -0
  90. package/vendor/src/components/form/smart-form-shell.tsx +59 -0
  91. package/vendor/src/components/inputs/async-select.tsx +1143 -0
  92. package/vendor/src/components/inputs/clearable-input.tsx +78 -0
  93. package/vendor/src/components/inputs/color-input.tsx +47 -0
  94. package/vendor/src/components/inputs/combobox.tsx +89 -0
  95. package/vendor/src/components/inputs/date-input.tsx +32 -0
  96. package/vendor/src/components/inputs/date-range-input.tsx +67 -0
  97. package/vendor/src/components/inputs/index.ts +19 -0
  98. package/vendor/src/components/inputs/input-chrome.tsx +37 -0
  99. package/vendor/src/components/inputs/input-decorator.tsx +64 -0
  100. package/vendor/src/components/inputs/input-value.ts +42 -0
  101. package/vendor/src/components/inputs/masked-input.tsx +51 -0
  102. package/vendor/src/components/inputs/money-input.tsx +73 -0
  103. package/vendor/src/components/inputs/number-input.tsx +87 -0
  104. package/vendor/src/components/inputs/numeric-value.ts +39 -0
  105. package/vendor/src/components/inputs/otp-input.tsx +102 -0
  106. package/vendor/src/components/inputs/password-input.tsx +85 -0
  107. package/vendor/src/components/inputs/phone-input.tsx +46 -0
  108. package/vendor/src/components/inputs/quantity-input.tsx +116 -0
  109. package/vendor/src/components/inputs/quantity-stepper.tsx +49 -0
  110. package/vendor/src/components/inputs/rating.tsx +98 -0
  111. package/vendor/src/components/inputs/search-input.tsx +26 -0
  112. package/vendor/src/components/inputs/simple-select.tsx +72 -0
  113. package/vendor/src/components/inputs/slider.tsx +149 -0
  114. package/vendor/src/components/inputs/tag-input.tsx +104 -0
  115. package/vendor/src/components/layout/app-header.tsx +46 -0
  116. package/vendor/src/components/layout/app-shell.tsx +243 -0
  117. package/vendor/src/components/layout/app-sidebar.tsx +179 -0
  118. package/vendor/src/components/layout/breadcrumbs.tsx +72 -0
  119. package/vendor/src/components/layout/index.ts +11 -0
  120. package/vendor/src/components/layout/page-container.tsx +30 -0
  121. package/vendor/src/components/layout/page-header.tsx +60 -0
  122. package/vendor/src/components/layout/public.ts +10 -0
  123. package/vendor/src/components/layout/section.tsx +76 -0
  124. package/vendor/src/components/layout/sidebar-nav.tsx +147 -0
  125. package/vendor/src/components/layout/stat-card.tsx +88 -0
  126. package/vendor/src/components/layout/sticky-footer-bar.tsx +23 -0
  127. package/vendor/src/components/layout/workspace-shell.tsx +50 -0
  128. package/vendor/src/components/navigation/anchor-nav.tsx +44 -0
  129. package/vendor/src/components/navigation/index.ts +4 -0
  130. package/vendor/src/components/navigation/page-tabs.tsx +67 -0
  131. package/vendor/src/components/navigation/pagination.tsx +179 -0
  132. package/vendor/src/components/navigation/stepper-tabs.tsx +67 -0
  133. package/vendor/src/components/notifications/index.ts +1 -0
  134. package/vendor/src/components/notifications/toast.tsx +259 -0
  135. package/vendor/src/components/overlay/confirm-dialog.tsx +66 -0
  136. package/vendor/src/components/overlay/dialog-actions.tsx +68 -0
  137. package/vendor/src/components/overlay/index.ts +4 -0
  138. package/vendor/src/components/overlay/modal-shell.tsx +93 -0
  139. package/vendor/src/components/overlay/sheet-shell.tsx +212 -0
  140. package/vendor/src/components/patterns/action-system.tsx +116 -0
  141. package/vendor/src/components/patterns/crud-system.tsx +53 -0
  142. package/vendor/src/components/patterns/data-view.tsx +84 -0
  143. package/vendor/src/components/patterns/entity-details.tsx +66 -0
  144. package/vendor/src/components/patterns/filter-builder.tsx +113 -0
  145. package/vendor/src/components/patterns/form-builder-presets.ts +131 -0
  146. package/vendor/src/components/patterns/form-builder.tsx +334 -0
  147. package/vendor/src/components/patterns/index.ts +12 -0
  148. package/vendor/src/components/patterns/public.ts +4 -0
  149. package/vendor/src/components/patterns/resource-detail-page.tsx +160 -0
  150. package/vendor/src/components/patterns/resource-page.tsx +159 -0
  151. package/vendor/src/components/patterns/resource-system.tsx +61 -0
  152. package/vendor/src/components/patterns/settings-section.tsx +46 -0
  153. package/vendor/src/components/patterns/status-system.tsx +89 -0
  154. package/vendor/src/components/theme-provider.tsx +51 -0
  155. package/vendor/src/components/ui/badge.tsx +52 -0
  156. package/vendor/src/components/ui/button.tsx +61 -0
  157. package/vendor/src/components/ui/card.tsx +103 -0
  158. package/vendor/src/components/ui/checkbox.tsx +82 -0
  159. package/vendor/src/components/ui/collapse.tsx +126 -0
  160. package/vendor/src/components/ui/command.tsx +194 -0
  161. package/vendor/src/components/ui/dialog.tsx +160 -0
  162. package/vendor/src/components/ui/divider.tsx +46 -0
  163. package/vendor/src/components/ui/dropdown-menu.tsx +266 -0
  164. package/vendor/src/components/ui/input-group.tsx +158 -0
  165. package/vendor/src/components/ui/input.tsx +20 -0
  166. package/vendor/src/components/ui/popover.tsx +90 -0
  167. package/vendor/src/components/ui/segmented-control.tsx +78 -0
  168. package/vendor/src/components/ui/select.tsx +201 -0
  169. package/vendor/src/components/ui/skeleton.tsx +75 -0
  170. package/vendor/src/components/ui/spinner.tsx +50 -0
  171. package/vendor/src/components/ui/switch.tsx +71 -0
  172. package/vendor/src/components/ui/table.tsx +114 -0
  173. package/vendor/src/components/ui/tabs.tsx +55 -0
  174. package/vendor/src/components/ui/textarea.tsx +18 -0
  175. package/vendor/src/components/ui/tooltip.tsx +38 -0
  176. package/vendor/src/components/upload/file-upload.tsx +483 -0
  177. package/vendor/src/components/upload/image-upload.tsx +118 -0
  178. package/vendor/src/components/upload/index.ts +2 -0
  179. package/vendor/src/components/wizard/index.ts +2 -0
  180. package/vendor/src/components/wizard/stepper.tsx +53 -0
  181. package/vendor/src/components/wizard/wizard.tsx +60 -0
  182. package/vendor/src/families/card-family.ts +28 -0
  183. package/vendor/src/families/catalog.ts +96 -0
  184. package/vendor/src/families/data-table-family.ts +31 -0
  185. package/vendor/src/families/docs-adoption.ts +103 -0
  186. package/vendor/src/families/docs-groups.ts +209 -0
  187. package/vendor/src/families/docs-queries.ts +84 -0
  188. package/vendor/src/families/docs-routing.ts +89 -0
  189. package/vendor/src/families/form-family.ts +45 -0
  190. package/vendor/src/families/index.ts +17 -0
  191. package/vendor/src/families/input-family.ts +61 -0
  192. package/vendor/src/families/member-metadata.ts +466 -0
  193. package/vendor/src/families/member-queries.ts +28 -0
  194. package/vendor/src/families/member-snippet-queries.ts +54 -0
  195. package/vendor/src/families/member-snippets.ts +673 -0
  196. package/vendor/src/families/migration-map.ts +79 -0
  197. package/vendor/src/families/queries.ts +63 -0
  198. package/vendor/src/families/select-family.ts +33 -0
  199. package/vendor/src/families/views.ts +81 -0
  200. package/vendor/src/hooks/index.ts +6 -0
  201. package/vendor/src/hooks/use-before-unload-when-dirty.ts +21 -0
  202. package/vendor/src/hooks/use-data-table-view-state.ts +122 -0
  203. package/vendor/src/hooks/use-debounce.ts +52 -0
  204. package/vendor/src/hooks/use-disclosure.ts +38 -0
  205. package/vendor/src/hooks/use-is-mobile.ts +28 -0
  206. package/vendor/src/hooks/use-session-storage-state.ts +85 -0
  207. package/vendor/src/index.ts +38 -0
  208. package/vendor/src/lib/utils.ts +6 -0
  209. package/vendor/templates/components/button.tsx +0 -0
  210. package/vendor/templates/components/data-table.tsx +0 -0
  211. package/vendor/templates/components/input.tsx +0 -0
  212. package/vendor/templates/lib/utils.ts +0 -0
  213. package/vendor/templates/styles/globals.css +0 -0
@@ -0,0 +1,79 @@
1
+ import { Controller, type Control, type FieldPath, type FieldValues } from "react-hook-form"
2
+
3
+ import { FormFieldShell, type FormFieldShellProps } from "@/components/form/form-field-shell"
4
+ import {
5
+ DateRangeInput,
6
+ type DateRangeInputProps,
7
+ type DateRangeValue,
8
+ } from "@/components/inputs/date-range-input"
9
+
10
+ export type FormDateRangeInputProps<
11
+ TFieldValues extends FieldValues = FieldValues,
12
+ TFromName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>,
13
+ TToName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>,
14
+ > = Omit<DateRangeInputProps, "value" | "onValueChange"> &
15
+ Pick<FormFieldShellProps, "label" | "description" | "required" | "className"> & {
16
+ control: Control<TFieldValues>
17
+ fromName: TFromName
18
+ toName: TToName
19
+ emptyValue?: unknown
20
+ onValueChange?: (value: DateRangeValue) => void
21
+ }
22
+
23
+ function FormDateRangeInput<
24
+ TFieldValues extends FieldValues = FieldValues,
25
+ TFromName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>,
26
+ TToName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>,
27
+ >({
28
+ control,
29
+ fromName,
30
+ toName,
31
+ label,
32
+ description,
33
+ required,
34
+ className,
35
+ emptyValue = "",
36
+ onValueChange,
37
+ ...props
38
+ }: FormDateRangeInputProps<TFieldValues, TFromName, TToName>) {
39
+ return (
40
+ <Controller
41
+ control={control}
42
+ name={fromName}
43
+ render={({ field: fromField, fieldState: fromState }) => (
44
+ <Controller
45
+ control={control}
46
+ name={toName}
47
+ render={({ field: toField, fieldState: toState }) => {
48
+ const error = fromState.error?.message ?? toState.error?.message
49
+
50
+ return (
51
+ <FormFieldShell
52
+ label={label}
53
+ description={description}
54
+ required={required}
55
+ error={error}
56
+ className={className}
57
+ >
58
+ <DateRangeInput
59
+ value={{
60
+ from: fromField.value ?? "",
61
+ to: toField.value ?? "",
62
+ }}
63
+ onValueChange={(value) => {
64
+ fromField.onChange(value.from || emptyValue)
65
+ toField.onChange(value.to || emptyValue)
66
+ onValueChange?.(value)
67
+ }}
68
+ {...props}
69
+ />
70
+ </FormFieldShell>
71
+ )
72
+ }}
73
+ />
74
+ )}
75
+ />
76
+ )
77
+ }
78
+
79
+ export { FormDateRangeInput }
@@ -0,0 +1,57 @@
1
+ import { Controller, type Control, type FieldPath, type FieldValues } from "react-hook-form"
2
+
3
+ import { DateRangePicker, type DateRangePickerProps } from "@/components/calendar/date-range-picker"
4
+ import { FormFieldShell, type FormFieldShellProps } from "@/components/form/form-field-shell"
5
+
6
+ export type FormDateRangePickerProps<TFieldValues extends FieldValues = FieldValues> = Omit<
7
+ DateRangePickerProps,
8
+ "value" | "onValueChange"
9
+ > &
10
+ Pick<FormFieldShellProps, "label" | "description" | "required" | "className"> & {
11
+ control: Control<TFieldValues>
12
+ fromName: FieldPath<TFieldValues>
13
+ toName: FieldPath<TFieldValues>
14
+ emptyValue?: unknown
15
+ onValueChange?: (value: { from?: string; to?: string }) => void
16
+ }
17
+
18
+ function FormDateRangePicker<TFieldValues extends FieldValues = FieldValues>({
19
+ control,
20
+ fromName,
21
+ toName,
22
+ label,
23
+ description,
24
+ required,
25
+ className,
26
+ emptyValue = "",
27
+ onValueChange,
28
+ ...props
29
+ }: FormDateRangePickerProps<TFieldValues>) {
30
+ return (
31
+ <Controller
32
+ control={control}
33
+ name={fromName}
34
+ render={({ field: fromField, fieldState: fromState }) => (
35
+ <Controller
36
+ control={control}
37
+ name={toName}
38
+ render={({ field: toField, fieldState: toState }) => (
39
+ <FormFieldShell label={label} description={description} required={required} error={fromState.error?.message ?? toState.error?.message} className={className}>
40
+ <DateRangePicker
41
+ value={{ from: fromField.value ?? "", to: toField.value ?? "" }}
42
+ onValueChange={(value) => {
43
+ fromField.onChange(value.from || emptyValue)
44
+ toField.onChange(value.to || emptyValue)
45
+ onValueChange?.({ from: value.from || undefined, to: value.to || undefined })
46
+ }}
47
+ {...props}
48
+ />
49
+ </FormFieldShell>
50
+ )}
51
+ />
52
+ )}
53
+ />
54
+ )
55
+ }
56
+
57
+ export { FormDateRangePicker }
@@ -0,0 +1,191 @@
1
+ import * as React from "react"
2
+ import { AlertCircleIcon } from "lucide-react"
3
+
4
+ import { cn } from "@/lib/utils"
5
+
6
+ export type FormFieldLayout = "vertical" | "horizontal" | "inline"
7
+ export type FormFieldDescriptionPosition = "top" | "bottom"
8
+
9
+ export type FormFieldShellProps = React.ComponentProps<"div"> & {
10
+ label?: React.ReactNode
11
+ description?: React.ReactNode
12
+ error?: React.ReactNode
13
+ required?: boolean
14
+ htmlFor?: string
15
+ layout?: FormFieldLayout
16
+ descriptionPosition?: FormFieldDescriptionPosition
17
+ labelAction?: React.ReactNode
18
+ requiredIndicator?: React.ReactNode
19
+ errorIcon?: React.ReactNode
20
+ showErrorIcon?: boolean
21
+ disabled?: boolean
22
+ readOnly?: boolean
23
+ labelClassName?: string
24
+ labelRowClassName?: string
25
+ descriptionClassName?: string
26
+ errorClassName?: string
27
+ contentClassName?: string
28
+ }
29
+
30
+ export type FormFieldShellControlProps = Pick<
31
+ FormFieldShellProps,
32
+ | "label"
33
+ | "description"
34
+ | "required"
35
+ | "className"
36
+ | "layout"
37
+ | "descriptionPosition"
38
+ | "labelAction"
39
+ | "requiredIndicator"
40
+ | "errorIcon"
41
+ | "showErrorIcon"
42
+ | "disabled"
43
+ | "readOnly"
44
+ | "labelClassName"
45
+ | "labelRowClassName"
46
+ | "descriptionClassName"
47
+ | "errorClassName"
48
+ | "contentClassName"
49
+ >
50
+
51
+ const layoutClassName: Record<FormFieldLayout, string> = {
52
+ vertical: "grid gap-2",
53
+ horizontal: "grid gap-2 sm:grid-cols-[minmax(0,12rem)_1fr] sm:items-start sm:gap-5",
54
+ inline: "flex flex-col gap-2 sm:flex-row sm:items-center sm:gap-4",
55
+ }
56
+
57
+ function FormFieldShell({
58
+ className,
59
+ label,
60
+ description,
61
+ error,
62
+ required = false,
63
+ htmlFor,
64
+ layout = "vertical",
65
+ descriptionPosition = "top",
66
+ labelAction,
67
+ requiredIndicator = <span className="ml-1 text-destructive">*</span>,
68
+ errorIcon,
69
+ showErrorIcon = true,
70
+ disabled = false,
71
+ readOnly = false,
72
+ labelClassName,
73
+ labelRowClassName,
74
+ descriptionClassName,
75
+ errorClassName,
76
+ contentClassName,
77
+ children,
78
+ ...props
79
+ }: FormFieldShellProps) {
80
+ const hasHeader = Boolean(label || labelAction)
81
+ const hasDescriptionTop = description && descriptionPosition === "top"
82
+ const hasDescriptionBottom = description && descriptionPosition === "bottom"
83
+ const resolvedErrorIcon = errorIcon ?? <AlertCircleIcon className="size-3.5" />
84
+
85
+ const header = hasHeader ? (
86
+ <div
87
+ data-slot="form-field-label-row"
88
+ className={cn(
89
+ "flex min-w-0 items-center justify-between gap-3",
90
+ layout === "inline" && "sm:w-auto",
91
+ labelRowClassName
92
+ )}
93
+ >
94
+ {label ? (
95
+ <label
96
+ data-slot="form-field-label"
97
+ htmlFor={htmlFor}
98
+ className={cn(
99
+ "min-w-0 text-sm font-semibold leading-none text-foreground",
100
+ disabled && "cursor-not-allowed opacity-60",
101
+ readOnly && "opacity-80",
102
+ labelClassName
103
+ )}
104
+ >
105
+ {label}
106
+ {required && requiredIndicator}
107
+ </label>
108
+ ) : (
109
+ <span />
110
+ )}
111
+
112
+ {labelAction && (
113
+ <div data-slot="form-field-label-action" className="shrink-0 text-sm">
114
+ {labelAction}
115
+ </div>
116
+ )}
117
+ </div>
118
+ ) : null
119
+
120
+ const descriptionNode = description ? (
121
+ <p
122
+ data-slot="form-field-description"
123
+ className={cn("text-sm leading-6 text-muted-foreground", disabled && "opacity-60", descriptionClassName)}
124
+ >
125
+ {description}
126
+ </p>
127
+ ) : null
128
+
129
+ const content = (
130
+ <div data-slot="form-field-content" className={cn("min-w-0 space-y-2", contentClassName)}>
131
+ {children}
132
+ </div>
133
+ )
134
+
135
+ const errorNode = error ? (
136
+ <p
137
+ data-slot="form-field-error"
138
+ className={cn("flex items-start gap-2 text-sm font-medium leading-6 text-destructive", errorClassName)}
139
+ >
140
+ {showErrorIcon && <span className="mt-1 shrink-0">{resolvedErrorIcon}</span>}
141
+ <span className="min-w-0">{error}</span>
142
+ </p>
143
+ ) : null
144
+
145
+ return (
146
+ <div
147
+ data-slot="form-field-shell"
148
+ data-layout={layout}
149
+ data-invalid={Boolean(error) || undefined}
150
+ data-disabled={disabled || undefined}
151
+ data-readonly={readOnly || undefined}
152
+ aria-disabled={disabled || undefined}
153
+ className={cn(layoutClassName[layout], className)}
154
+ {...props}
155
+ >
156
+ {layout === "horizontal" ? (
157
+ <>
158
+ <div className="grid gap-2 sm:pt-2.5">
159
+ {header}
160
+ {hasDescriptionTop && descriptionNode}
161
+ </div>
162
+ <div className="grid min-w-0 gap-2">
163
+ {content}
164
+ {hasDescriptionBottom && descriptionNode}
165
+ {errorNode}
166
+ </div>
167
+ </>
168
+ ) : layout === "inline" ? (
169
+ <>
170
+ {header}
171
+ <div className="grid min-w-0 flex-1 gap-2">
172
+ {hasDescriptionTop && descriptionNode}
173
+ {content}
174
+ {hasDescriptionBottom && descriptionNode}
175
+ {errorNode}
176
+ </div>
177
+ </>
178
+ ) : (
179
+ <>
180
+ {header}
181
+ {hasDescriptionTop && descriptionNode}
182
+ {content}
183
+ {hasDescriptionBottom && descriptionNode}
184
+ {errorNode}
185
+ </>
186
+ )}
187
+ </div>
188
+ )
189
+ }
190
+
191
+ export { FormFieldShell }