@k3-universe/react-kit 0.0.41 → 0.0.42

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 (701) hide show
  1. package/.babelrc +12 -12
  2. package/.storybook/main.ts +22 -22
  3. package/.storybook/preview.ts +198 -198
  4. package/Dockerfile.storybook +43 -43
  5. package/README.md +9 -9
  6. package/biome.json +10 -10
  7. package/components.json +20 -20
  8. package/dist/index.js +32 -14
  9. package/dist/kit/builder/auth/hooks/core-hooks.d.ts.map +1 -1
  10. package/dist/kit/builder/form/components/fields/AutocompleteField.d.ts.map +1 -1
  11. package/dist/kit/builder/form/components/fields/DatePickerField.d.ts.map +1 -1
  12. package/dist/kit/builder/form/components/fields/DateRangePickerField.d.ts.map +1 -1
  13. package/dist/kit/builder/form/types.d.ts +2 -0
  14. package/dist/kit/builder/form/types.d.ts.map +1 -1
  15. package/dist/kit/components/autocomplete/Autocomplete.d.ts +2 -1
  16. package/dist/kit/components/autocomplete/Autocomplete.d.ts.map +1 -1
  17. package/dist/kit/layouts/admin/hooks/menu.d.ts.map +1 -1
  18. package/dist/kit/themes/clean-slate.css +4 -8
  19. package/dist/kit/themes/default.css +5 -9
  20. package/dist/kit/themes/minimal-modern.css +4 -8
  21. package/dist/kit/themes/spotify.css +4 -8
  22. package/package.json +1 -1
  23. package/src/index.ts +99 -99
  24. package/src/kit/builder/auth/components/Can.tsx +27 -27
  25. package/src/kit/builder/auth/components/RequireAuth.tsx +78 -78
  26. package/src/kit/builder/auth/components/ShowWhenAuthenticated.tsx +10 -10
  27. package/src/kit/builder/auth/components/ShowWhenError.tsx +10 -10
  28. package/src/kit/builder/auth/components/ShowWhenLoading.tsx +10 -10
  29. package/src/kit/builder/auth/components/ShowWhenUnauthenticated.tsx +10 -10
  30. package/src/kit/builder/auth/components/withPermission.tsx +23 -23
  31. package/src/kit/builder/auth/hooks/action-hooks.ts +34 -34
  32. package/src/kit/builder/auth/hooks/core-hooks.ts +65 -65
  33. package/src/kit/builder/auth/hooks/index.ts +4 -4
  34. package/src/kit/builder/auth/hooks/permission-hooks.ts +43 -43
  35. package/src/kit/builder/auth/hooks/token-hooks.ts +25 -25
  36. package/src/kit/builder/auth/index.ts +132 -132
  37. package/src/kit/builder/auth/providers/AuthProvider.tsx +131 -131
  38. package/src/kit/builder/auth/types/adapter-config.ts +44 -44
  39. package/src/kit/builder/auth/types/adapter.ts +132 -132
  40. package/src/kit/builder/auth/types/core.ts +27 -27
  41. package/src/kit/builder/auth/types/index.ts +9 -9
  42. package/src/kit/builder/auth/types/middleware.ts +20 -20
  43. package/src/kit/builder/auth/types/permissions.ts +23 -23
  44. package/src/kit/builder/auth/types/state.ts +16 -16
  45. package/src/kit/builder/auth/types/storage.ts +21 -21
  46. package/src/kit/builder/auth/types/token-manager.ts +9 -9
  47. package/src/kit/builder/auth/types/utils.ts +55 -55
  48. package/src/kit/builder/auth/utils/auth-adapter.ts +437 -437
  49. package/src/kit/builder/auth/utils/client-adapters/apollo-link.ts +30 -30
  50. package/src/kit/builder/auth/utils/client-adapters/axios.ts +61 -61
  51. package/src/kit/builder/auth/utils/client-adapters/fetch.ts +48 -48
  52. package/src/kit/builder/auth/utils/client-adapters/graphql.ts +60 -60
  53. package/src/kit/builder/auth/utils/client-adapters/index.ts +6 -6
  54. package/src/kit/builder/auth/utils/client-adapters/rest.ts +60 -60
  55. package/src/kit/builder/auth/utils/client-adapters/urql-exchange.ts +76 -76
  56. package/src/kit/builder/auth/utils/permission-checker.ts +150 -150
  57. package/src/kit/builder/auth/utils/storage/browser.ts +99 -99
  58. package/src/kit/builder/auth/utils/storage/cookie.ts +116 -116
  59. package/src/kit/builder/auth/utils/storage/encryption.ts +80 -80
  60. package/src/kit/builder/auth/utils/storage/env.ts +2 -2
  61. package/src/kit/builder/auth/utils/storage/factory.ts +37 -37
  62. package/src/kit/builder/auth/utils/storage/index.ts +6 -6
  63. package/src/kit/builder/auth/utils/storage/memory.ts +15 -15
  64. package/src/kit/builder/auth/utils/token-manager.ts +55 -55
  65. package/src/kit/builder/data-table/components/DataTable.tsx +547 -547
  66. package/src/kit/builder/data-table/components/DataTableColumnHeader.tsx +67 -67
  67. package/src/kit/builder/data-table/components/DataTablePagination.tsx +112 -112
  68. package/src/kit/builder/data-table/components/DataTableViewOptions.tsx +51 -51
  69. package/src/kit/builder/data-table/index.ts +6 -6
  70. package/src/kit/builder/data-table/types.ts +44 -44
  71. package/src/kit/builder/data-table/utils/dotAccessor.ts +6 -6
  72. package/src/kit/builder/dialog/index.ts +6 -6
  73. package/src/kit/builder/dialog/provider.tsx +170 -170
  74. package/src/kit/builder/form/components/FormBuilder.tsx +295 -295
  75. package/src/kit/builder/form/components/FormBuilderActions.tsx +78 -78
  76. package/src/kit/builder/form/components/FormBuilderContext.tsx +52 -52
  77. package/src/kit/builder/form/components/FormBuilderField.tsx +456 -456
  78. package/src/kit/builder/form/components/fields/ArrayField.tsx +311 -311
  79. package/src/kit/builder/form/components/fields/AutocompleteField.tsx +89 -88
  80. package/src/kit/builder/form/components/fields/CheckboxField.tsx +65 -65
  81. package/src/kit/builder/form/components/fields/DateField.tsx +26 -26
  82. package/src/kit/builder/form/components/fields/DatePickerField.tsx +31 -29
  83. package/src/kit/builder/form/components/fields/DateRangePickerField.tsx +55 -46
  84. package/src/kit/builder/form/components/fields/DateTimePickerField.tsx +38 -38
  85. package/src/kit/builder/form/components/fields/DateTimeRangePickerField.tsx +48 -48
  86. package/src/kit/builder/form/components/fields/FileField.tsx +31 -31
  87. package/src/kit/builder/form/components/fields/MonthPickerField.tsx +33 -33
  88. package/src/kit/builder/form/components/fields/MonthRangePickerField.tsx +41 -41
  89. package/src/kit/builder/form/components/fields/NumberField.tsx +20 -20
  90. package/src/kit/builder/form/components/fields/ObjectField.tsx +35 -35
  91. package/src/kit/builder/form/components/fields/RadioField.tsx +47 -47
  92. package/src/kit/builder/form/components/fields/SelectField.tsx +46 -46
  93. package/src/kit/builder/form/components/fields/SwitchField.tsx +65 -65
  94. package/src/kit/builder/form/components/fields/TextField.tsx +25 -25
  95. package/src/kit/builder/form/components/fields/TextareaField.tsx +20 -20
  96. package/src/kit/builder/form/components/fields/TimePickerField.tsx +35 -35
  97. package/src/kit/builder/form/components/fields/TimeRangePickerField.tsx +43 -43
  98. package/src/kit/builder/form/components/fields/index.ts +21 -21
  99. package/src/kit/builder/form/components/fields/types.ts +18 -18
  100. package/src/kit/builder/form/components/index.ts +3 -3
  101. package/src/kit/builder/form/components/sectionNodes.tsx +147 -147
  102. package/src/kit/builder/form/hooks/useFormBuilder.ts +451 -451
  103. package/src/kit/builder/form/index.ts +4 -4
  104. package/src/kit/builder/form/types.ts +245 -243
  105. package/src/kit/builder/form/utils/common-forms.ts +68 -68
  106. package/src/kit/builder/form/utils/dependencies.ts +56 -56
  107. package/src/kit/builder/form/utils/field-factories.ts +177 -177
  108. package/src/kit/builder/form/utils/index.ts +17 -17
  109. package/src/kit/builder/form/utils/section-factories.ts +38 -38
  110. package/src/kit/builder/form/utils/transformers.ts +100 -100
  111. package/src/kit/builder/form/utils/validations.ts +22 -22
  112. package/src/kit/builder/form/utils/validators.ts +20 -20
  113. package/src/kit/builder/page/Page.tsx +202 -202
  114. package/src/kit/builder/page/index.ts +1 -1
  115. package/src/kit/builder/section/SectionBuilder.tsx +435 -435
  116. package/src/kit/builder/section/SectionContainer.tsx +85 -85
  117. package/src/kit/builder/section/index.ts +2 -2
  118. package/src/kit/builder/section/types.ts +80 -80
  119. package/src/kit/builder/stack-dialog/context.ts +15 -15
  120. package/src/kit/builder/stack-dialog/hooks.ts +13 -13
  121. package/src/kit/builder/stack-dialog/index.ts +12 -12
  122. package/src/kit/builder/stack-dialog/provider.tsx +51 -51
  123. package/src/kit/builder/stack-dialog/renderer.tsx +36 -36
  124. package/src/kit/builder/stack-dialog/types.ts +34 -34
  125. package/src/kit/components/autocomplete/Autocomplete.tsx +671 -664
  126. package/src/kit/components/autocomplete/index.ts +3 -3
  127. package/src/kit/components/autocomplete/types.ts +27 -27
  128. package/src/kit/components/datepicker/DatePicker.tsx +173 -173
  129. package/src/kit/components/datepicker/DateRangePicker.tsx +526 -526
  130. package/src/kit/components/datepicker/index.ts +2 -2
  131. package/src/kit/components/datetimepicker/DateTimePicker.tsx +420 -420
  132. package/src/kit/components/datetimepicker/DateTimeRangePicker.tsx +676 -676
  133. package/src/kit/components/datetimepicker/index.ts +2 -2
  134. package/src/kit/components/fileuploader/FileUploader.tsx +626 -626
  135. package/src/kit/components/fileuploader/index.ts +3 -3
  136. package/src/kit/components/fileuploader/types.ts +76 -76
  137. package/src/kit/components/forminfo/FormInfoError.tsx +135 -135
  138. package/src/kit/components/forminfo/index.ts +1 -1
  139. package/src/kit/components/keyboard/Keyboard.tsx +916 -916
  140. package/src/kit/components/keyboard/index.ts +4 -4
  141. package/src/kit/components/login/Login.tsx +141 -141
  142. package/src/kit/components/login/index.ts +1 -1
  143. package/src/kit/components/monthpicker/MonthInput.tsx +131 -131
  144. package/src/kit/components/monthpicker/MonthPicker.tsx +224 -224
  145. package/src/kit/components/monthpicker/MonthRangeInput.tsx +153 -153
  146. package/src/kit/components/monthpicker/MonthRangePicker.tsx +409 -409
  147. package/src/kit/components/numpad/Numpad.tsx +377 -377
  148. package/src/kit/components/numpad/index.ts +4 -4
  149. package/src/kit/components/timepicker/TimePicker.tsx +319 -319
  150. package/src/kit/components/timepicker/TimeRangePicker.tsx +368 -368
  151. package/src/kit/components/timepicker/index.ts +2 -2
  152. package/src/kit/layouts/admin/components/AdminLayout.tsx +286 -286
  153. package/src/kit/layouts/admin/components/ThemeToggle.tsx +54 -54
  154. package/src/kit/layouts/admin/hooks/menu.ts +32 -32
  155. package/src/kit/layouts/admin/providers/AdminMenuProvider.tsx +78 -78
  156. package/src/kit/layouts/admin/types/index.ts +26 -26
  157. package/src/kit/providers/ThemeProvider.tsx +90 -90
  158. package/src/kit/themes/base.css +11 -11
  159. package/src/kit/themes/clean-slate.css +155 -155
  160. package/src/kit/themes/default.css +192 -192
  161. package/src/kit/themes/minimal-modern.css +155 -155
  162. package/src/kit/themes/spotify.css +164 -164
  163. package/src/shadcn/hooks/use-mobile.ts +21 -21
  164. package/src/shadcn/lib/utils.ts +6 -6
  165. package/src/shadcn/ui/accordion.tsx +66 -66
  166. package/src/shadcn/ui/alert-dialog.tsx +157 -157
  167. package/src/shadcn/ui/alert.tsx +66 -66
  168. package/src/shadcn/ui/aspect-ratio.tsx +11 -11
  169. package/src/shadcn/ui/avatar.tsx +53 -53
  170. package/src/shadcn/ui/badge.tsx +46 -46
  171. package/src/shadcn/ui/breadcrumb.tsx +108 -108
  172. package/src/shadcn/ui/button.tsx +59 -59
  173. package/src/shadcn/ui/calendar.tsx +217 -217
  174. package/src/shadcn/ui/card.tsx +92 -92
  175. package/src/shadcn/ui/carousel.tsx +240 -240
  176. package/src/shadcn/ui/chart.tsx +353 -353
  177. package/src/shadcn/ui/checkbox.tsx +32 -32
  178. package/src/shadcn/ui/collapsible.tsx +33 -33
  179. package/src/shadcn/ui/command.tsx +184 -184
  180. package/src/shadcn/ui/context-menu.tsx +252 -252
  181. package/src/shadcn/ui/dialog.tsx +143 -143
  182. package/src/shadcn/ui/drawer.tsx +135 -135
  183. package/src/shadcn/ui/dropdown-menu.tsx +257 -257
  184. package/src/shadcn/ui/form.tsx +168 -168
  185. package/src/shadcn/ui/hover-card.tsx +44 -44
  186. package/src/shadcn/ui/input-otp.tsx +77 -77
  187. package/src/shadcn/ui/input.tsx +21 -21
  188. package/src/shadcn/ui/label.tsx +24 -24
  189. package/src/shadcn/ui/menubar.tsx +276 -276
  190. package/src/shadcn/ui/navigation-menu.tsx +168 -168
  191. package/src/shadcn/ui/pagination.tsx +127 -127
  192. package/src/shadcn/ui/popover.tsx +48 -48
  193. package/src/shadcn/ui/progress.tsx +31 -31
  194. package/src/shadcn/ui/radio-group.tsx +45 -45
  195. package/src/shadcn/ui/resizable.tsx +56 -56
  196. package/src/shadcn/ui/scroll-area.tsx +58 -58
  197. package/src/shadcn/ui/select.tsx +185 -185
  198. package/src/shadcn/ui/separator.tsx +28 -28
  199. package/src/shadcn/ui/sheet.tsx +139 -139
  200. package/src/shadcn/ui/sidebar.tsx +726 -726
  201. package/src/shadcn/ui/skeleton.tsx +13 -13
  202. package/src/shadcn/ui/slider.tsx +63 -63
  203. package/src/shadcn/ui/sonner.tsx +25 -25
  204. package/src/shadcn/ui/switch.tsx +31 -31
  205. package/src/shadcn/ui/table.tsx +116 -116
  206. package/src/shadcn/ui/tabs.tsx +66 -66
  207. package/src/shadcn/ui/textarea.tsx +18 -18
  208. package/src/shadcn/ui/toggle-group.tsx +73 -73
  209. package/src/shadcn/ui/toggle.tsx +47 -47
  210. package/src/shadcn/ui/tooltip.tsx +61 -61
  211. package/src/stories/FileUploader.stories.tsx +185 -185
  212. package/src/stories/kit/builder/DataTable.Basic.stories.tsx +71 -71
  213. package/src/stories/kit/builder/DataTable.Filters.stories.tsx +98 -98
  214. package/src/stories/kit/builder/DataTable.Pagination.stories.tsx +73 -73
  215. package/src/stories/kit/builder/DataTable.SelectionAndActions.stories.tsx +77 -77
  216. package/src/stories/kit/builder/DataTable.Sorting.stories.tsx +67 -67
  217. package/src/stories/kit/builder/Dialog.stories.tsx +90 -90
  218. package/src/stories/kit/builder/Form.ArrayLayouts.stories.tsx +177 -177
  219. package/src/stories/kit/builder/Form.Autocomplete.stories.tsx +220 -220
  220. package/src/stories/kit/builder/Form.Basic.stories.tsx +79 -79
  221. package/src/stories/kit/builder/Form.Complex.stories.tsx +569 -569
  222. package/src/stories/kit/builder/Form.DateTime.stories.tsx +70 -70
  223. package/src/stories/kit/builder/Form.Dynamic.stories.tsx +926 -926
  224. package/src/stories/kit/builder/Form.Files.stories.tsx +136 -136
  225. package/src/stories/kit/builder/Form.MultipleFormBuilder.stories.tsx +377 -377
  226. package/src/stories/kit/builder/Form.Pickers.stories.tsx +75 -75
  227. package/src/stories/kit/builder/Form.Simple.stories.tsx +91 -91
  228. package/src/stories/kit/builder/Form.Time.stories.tsx +68 -68
  229. package/src/stories/kit/builder/Page.stories.tsx +115 -115
  230. package/src/stories/kit/builder/Section.stories.tsx +282 -282
  231. package/src/stories/kit/components/Autocomplete.stories.tsx +313 -313
  232. package/src/stories/kit/components/DatePicker.stories.tsx +195 -195
  233. package/src/stories/kit/components/DateRangePicker.stories.tsx +195 -195
  234. package/src/stories/kit/components/Keyboard.stories.tsx +263 -263
  235. package/src/stories/kit/components/Login.stories.tsx +167 -167
  236. package/src/stories/kit/components/MonthPicker.stories.tsx +100 -100
  237. package/src/stories/kit/components/MonthRangePicker.stories.tsx +117 -117
  238. package/src/stories/kit/components/Numpad.stories.tsx +195 -195
  239. package/src/stories/kit/components/TimePicker.stories.tsx +71 -71
  240. package/src/stories/kit/components/TimeRangePicker.stories.tsx +43 -43
  241. package/src/stories/kit/layouts/admin/AdminLayout.Basic.stories.tsx +118 -118
  242. package/src/stories/kit/layouts/admin/AdminLayout.Collapsible.stories.tsx +81 -81
  243. package/src/stories/kit/layouts/admin/AdminLayout.Complex.stories.tsx +251 -251
  244. package/src/stories/kit/layouts/admin/AdminLayout.CustomSidebarHeaderComponent.stories.tsx +80 -80
  245. package/src/stories/kit/layouts/admin/AdminLayout.CustomSidebarTitleAndIcon.stories.tsx +70 -70
  246. package/src/stories/kit/layouts/admin/AdminLayout.HeaderSlots.stories.tsx +87 -87
  247. package/src/stories/shadcn/ui/Accordion.stories.tsx +117 -117
  248. package/src/stories/shadcn/ui/Alert.stories.tsx +36 -36
  249. package/src/stories/shadcn/ui/AlertDialog.stories.tsx +73 -73
  250. package/src/stories/shadcn/ui/AspectRatio.stories.tsx +58 -58
  251. package/src/stories/shadcn/ui/Avatar.stories.tsx +26 -26
  252. package/src/stories/shadcn/ui/Badge.stories.tsx +28 -28
  253. package/src/stories/shadcn/ui/Breadcrumb.stories.tsx +47 -47
  254. package/src/stories/shadcn/ui/Button.stories.tsx +51 -51
  255. package/src/stories/shadcn/ui/Calendar.stories.tsx +33 -33
  256. package/src/stories/shadcn/ui/Card.stories.tsx +39 -39
  257. package/src/stories/shadcn/ui/Carousel.stories.tsx +37 -37
  258. package/src/stories/shadcn/ui/Chart.stories.tsx +72 -72
  259. package/src/stories/shadcn/ui/Checkbox.stories.tsx +64 -64
  260. package/src/stories/shadcn/ui/Collapsible.stories.tsx +40 -40
  261. package/src/stories/shadcn/ui/Command.stories.tsx +53 -53
  262. package/src/stories/shadcn/ui/ContextMenu.stories.tsx +67 -67
  263. package/src/stories/shadcn/ui/Dialog.stories.tsx +45 -45
  264. package/src/stories/shadcn/ui/Drawer.stories.tsx +49 -49
  265. package/src/stories/shadcn/ui/DropdownMenu.stories.tsx +75 -75
  266. package/src/stories/shadcn/ui/Form.stories.tsx +93 -93
  267. package/src/stories/shadcn/ui/HoverCard.stories.tsx +43 -43
  268. package/src/stories/shadcn/ui/Input.stories.tsx +41 -41
  269. package/src/stories/shadcn/ui/InputOtp.stories.tsx +39 -39
  270. package/src/stories/shadcn/ui/Label.stories.tsx +29 -29
  271. package/src/stories/shadcn/ui/Menubar.stories.tsx +87 -87
  272. package/src/stories/shadcn/ui/NavigationMenu.stories.tsx +78 -78
  273. package/src/stories/shadcn/ui/Pagination.stories.tsx +51 -51
  274. package/src/stories/shadcn/ui/Popover.stories.tsx +53 -53
  275. package/src/stories/shadcn/ui/Progress.stories.tsx +30 -30
  276. package/src/stories/shadcn/ui/RadioGroup.stories.tsx +31 -31
  277. package/src/stories/shadcn/ui/Resizable.stories.tsx +55 -55
  278. package/src/stories/shadcn/ui/ScrollArea.stories.tsx +84 -84
  279. package/src/stories/shadcn/ui/Select.stories.tsx +61 -61
  280. package/src/stories/shadcn/ui/Separator.stories.tsx +35 -35
  281. package/src/stories/shadcn/ui/Sheet.stories.tsx +48 -48
  282. package/src/stories/shadcn/ui/Sidebar.stories.tsx +131 -131
  283. package/src/stories/shadcn/ui/Skeleton.stories.tsx +23 -23
  284. package/src/stories/shadcn/ui/Slider.stories.tsx +21 -21
  285. package/src/stories/shadcn/ui/Sonner.stories.tsx +49 -49
  286. package/src/stories/shadcn/ui/Switch.stories.tsx +28 -28
  287. package/src/stories/shadcn/ui/Table.stories.tsx +374 -374
  288. package/src/stories/shadcn/ui/Tabs.stories.tsx +31 -31
  289. package/src/stories/shadcn/ui/Textarea.stories.tsx +42 -42
  290. package/src/stories/shadcn/ui/Toggle.stories.tsx +44 -44
  291. package/src/stories/shadcn/ui/ToggleGroup.stories.tsx +28 -28
  292. package/src/stories/shadcn/ui/Tooltip.stories.tsx +30 -30
  293. package/tsconfig.json +20 -20
  294. package/tsconfig.lib.json +52 -52
  295. package/tsconfig.spec.json +38 -38
  296. package/vite.config.ts +93 -93
  297. package/storybook-static/assets/Accordion.stories-KU4JBR8U.js +0 -52
  298. package/storybook-static/assets/AdminLayout-CPvVCwfY.js +0 -53
  299. package/storybook-static/assets/AdminLayout.Basic.stories-DkP2UVXe.js +0 -4
  300. package/storybook-static/assets/AdminLayout.Collapsible.stories-BuuVjtpW.js +0 -4
  301. package/storybook-static/assets/AdminLayout.Complex.stories-D-k4H0hJ.js +0 -29
  302. package/storybook-static/assets/AdminLayout.CustomSidebarHeaderComponent.stories-B_0IEDd4.js +0 -9
  303. package/storybook-static/assets/AdminLayout.CustomSidebarTitleAndIcon.stories-CvAeXUyA.js +0 -4
  304. package/storybook-static/assets/AdminLayout.HeaderSlots.stories-RBFHoSZK.js +0 -7
  305. package/storybook-static/assets/Alert.stories-DKxKtIc0.js +0 -27
  306. package/storybook-static/assets/AlertDialog.stories-BqTpZ_nG.js +0 -43
  307. package/storybook-static/assets/AspectRatio.stories-DPO9QQ5F.js +0 -22
  308. package/storybook-static/assets/Autocomplete-Cpg4CaJe.js +0 -56
  309. package/storybook-static/assets/Autocomplete.stories-CWj4G5fh.js +0 -56
  310. package/storybook-static/assets/Avatar.stories-DPhov_2g.js +0 -12
  311. package/storybook-static/assets/Badge.stories-DFKrRdXq.js +0 -12
  312. package/storybook-static/assets/Breadcrumb.stories-CTE6CZUC.js +0 -25
  313. package/storybook-static/assets/Button.stories-cbt2InL-.js +0 -26
  314. package/storybook-static/assets/Calendar.stories-DRhTw_43.js +0 -3
  315. package/storybook-static/assets/Card.stories-Isf6n_K3.js +0 -15
  316. package/storybook-static/assets/Carousel.stories-Cmg0I3fR.js +0 -15
  317. package/storybook-static/assets/Chart.stories-aQ-fNijT.js +0 -126
  318. package/storybook-static/assets/Checkbox.stories-B7YMXPDc.js +0 -12
  319. package/storybook-static/assets/Collapsible.stories-BUzl17ZZ.js +0 -18
  320. package/storybook-static/assets/Combination-BdQWAuko.js +0 -41
  321. package/storybook-static/assets/Command.stories-DzBlWQs0.js +0 -30
  322. package/storybook-static/assets/ContextMenu.stories-CJlBQyXc.js +0 -31
  323. package/storybook-static/assets/DataTable.Basic.stories-BWYKFDmK.js +0 -6
  324. package/storybook-static/assets/DataTable.Filters.stories-uZdtJk8t.js +0 -21
  325. package/storybook-static/assets/DataTable.Pagination.stories-C5N1khkp.js +0 -24
  326. package/storybook-static/assets/DataTable.SelectionAndActions.stories-FhCqZKvO.js +0 -26
  327. package/storybook-static/assets/DataTable.Sorting.stories-D-k7EtRj.js +0 -6
  328. package/storybook-static/assets/Dialog.stories-C62AF-Gx.js +0 -54
  329. package/storybook-static/assets/Dialog.stories-lrjRwOus.js +0 -18
  330. package/storybook-static/assets/Drawer.stories-CGjkdJeV.js +0 -24
  331. package/storybook-static/assets/DropdownMenu.stories-DkGClRAA.js +0 -35
  332. package/storybook-static/assets/Form.ArrayLayouts.stories-C5d_062d.js +0 -130
  333. package/storybook-static/assets/Form.Autocomplete.stories-CPZPkk4o.js +0 -142
  334. package/storybook-static/assets/Form.Basic.stories-Bhcu3-3n.js +0 -58
  335. package/storybook-static/assets/Form.Complex.stories-QnXh5a7Q.js +0 -361
  336. package/storybook-static/assets/Form.Dynamic.stories-DFW6wIuT.js +0 -502
  337. package/storybook-static/assets/Form.Simple.stories-BhJcyhbE.js +0 -53
  338. package/storybook-static/assets/Form.stories-PFNsMYxO.js +0 -3
  339. package/storybook-static/assets/FormBuilder-BQBBxo_k.js +0 -5
  340. package/storybook-static/assets/HoverCard.stories-CAlQEVn8.js +0 -21
  341. package/storybook-static/assets/Input.stories-CEhODt0V.js +0 -16
  342. package/storybook-static/assets/InputOtp.stories-DSvNP4dS.js +0 -42
  343. package/storybook-static/assets/Label.stories-B3pa8ZLY.js +0 -14
  344. package/storybook-static/assets/Login.stories-C7KQkmR_.js +0 -37
  345. package/storybook-static/assets/Menubar.stories-CHXhSHxc.js +0 -44
  346. package/storybook-static/assets/MonthPicker.stories-BnrOc4fm.js +0 -99
  347. package/storybook-static/assets/MonthRangePicker.stories-55Gk1t-7.js +0 -134
  348. package/storybook-static/assets/NavigationMenu.stories-CXoS080P.js +0 -30
  349. package/storybook-static/assets/Page.stories-GdSJgZ6-.js +0 -91
  350. package/storybook-static/assets/Pagination.stories-BEBwqH4N.js +0 -29
  351. package/storybook-static/assets/Popover.stories-BICy98Cw.js +0 -15
  352. package/storybook-static/assets/Progress.stories-DECHNOME.js +0 -8
  353. package/storybook-static/assets/RadioGroup.stories-DA7-uKfV.js +0 -16
  354. package/storybook-static/assets/Resizable.stories-B99kWkH7.js +0 -25
  355. package/storybook-static/assets/ScrollArea.stories-BqvUAXqU.js +0 -12
  356. package/storybook-static/assets/Section.stories-lFMlFBQn.js +0 -277
  357. package/storybook-static/assets/SectionBuilder-BQW705x0.js +0 -1
  358. package/storybook-static/assets/Select.stories-BsKyZ6Io.js +0 -17
  359. package/storybook-static/assets/Separator.stories-BTDOaOM2.js +0 -17
  360. package/storybook-static/assets/Sheet.stories-Cam1gR6G.js +0 -24
  361. package/storybook-static/assets/Sidebar.stories-DnOa6G7y.js +0 -106
  362. package/storybook-static/assets/Skeleton.stories-BQNIuIe5.js +0 -9
  363. package/storybook-static/assets/Slider.stories-Bslq7hjq.js +0 -6
  364. package/storybook-static/assets/Sonner.stories-D34pBBtI.js +0 -18
  365. package/storybook-static/assets/Switch.stories-Puyb1-Bx.js +0 -3
  366. package/storybook-static/assets/Table.stories-ClZxAhut.js +0 -35
  367. package/storybook-static/assets/Tabs.stories-CURNTETB.js +0 -10
  368. package/storybook-static/assets/Textarea.stories-Cf1ZBrWw.js +0 -17
  369. package/storybook-static/assets/Toggle.stories-CdMHY_bi.js +0 -3
  370. package/storybook-static/assets/ToggleGroup.stories-BM68m1dX.js +0 -13
  371. package/storybook-static/assets/Tooltip.stories-DiQv64dM.js +0 -10
  372. package/storybook-static/assets/accordion-DVgwQcnw.js +0 -1
  373. package/storybook-static/assets/alert-dialog-DCUEwpqm.js +0 -7
  374. package/storybook-static/assets/avatar-BzwOE-mi.js +0 -1
  375. package/storybook-static/assets/axe-HmUsR1st.js +0 -30
  376. package/storybook-static/assets/badge-BnQWua6u.js +0 -1
  377. package/storybook-static/assets/button-0oMkiryo.js +0 -1
  378. package/storybook-static/assets/card-BJpPOzP8.js +0 -1
  379. package/storybook-static/assets/chart-column-DZGb4ZZS.js +0 -6
  380. package/storybook-static/assets/check-B9hBGj6o.js +0 -6
  381. package/storybook-static/assets/checkbox-CyIeaWHX.js +0 -1
  382. package/storybook-static/assets/chevron-down-D_37S6il.js +0 -6
  383. package/storybook-static/assets/chevron-left-BBoN0vbI.js +0 -6
  384. package/storybook-static/assets/chevron-right-B5vIMLxK.js +0 -6
  385. package/storybook-static/assets/circle-C5Lzx6Nx.js +0 -6
  386. package/storybook-static/assets/clean-slate-D1HmMFJM.css +0 -1
  387. package/storybook-static/assets/command-Csa9p8_a.js +0 -6
  388. package/storybook-static/assets/createLucideIcon-BrHXro7t.js +0 -21
  389. package/storybook-static/assets/default-CN_Fo1GY.css +0 -1
  390. package/storybook-static/assets/dependencies-ctrV69dx.js +0 -1
  391. package/storybook-static/assets/dialog-CsnqITTn.js +0 -1
  392. package/storybook-static/assets/dropdown-menu-BWxxwPHL.js +0 -1
  393. package/storybook-static/assets/ellipsis-BRS038RR.js +0 -6
  394. package/storybook-static/assets/grip-vertical-BxXG8KNA.js +0 -6
  395. package/storybook-static/assets/iframe-C9bogcIc.css +0 -1
  396. package/storybook-static/assets/iframe-v7iAhKit.js +0 -1555
  397. package/storybook-static/assets/index-0-qMRXou.js +0 -1
  398. package/storybook-static/assets/index-AvwFFKJc.js +0 -1
  399. package/storybook-static/assets/index-BVDb4dFc.js +0 -1
  400. package/storybook-static/assets/index-B_qx7A5T.js +0 -1
  401. package/storybook-static/assets/index-BdQq_4o_.js +0 -1
  402. package/storybook-static/assets/index-BfiCOk42.js +0 -1
  403. package/storybook-static/assets/index-Bv9yk470.js +0 -1
  404. package/storybook-static/assets/index-Bw1A27Kp.js +0 -1
  405. package/storybook-static/assets/index-ByqivBWx.js +0 -1
  406. package/storybook-static/assets/index-C9Ta0ZTH.js +0 -1
  407. package/storybook-static/assets/index-CDY5kTx5.js +0 -1
  408. package/storybook-static/assets/index-CGnyVRgB.js +0 -1
  409. package/storybook-static/assets/index-CGrAONsN.js +0 -1
  410. package/storybook-static/assets/index-CWjrGFAQ.js +0 -1
  411. package/storybook-static/assets/index-CwBdPBFz.js +0 -9
  412. package/storybook-static/assets/index-D8RXF03I.js +0 -1
  413. package/storybook-static/assets/index-DLIxT4Z7.js +0 -1
  414. package/storybook-static/assets/index-DW48STyt.js +0 -1
  415. package/storybook-static/assets/index-DbaA6-o1.js +0 -1
  416. package/storybook-static/assets/index-Dph_5COR.js +0 -1
  417. package/storybook-static/assets/index-DrN5n71E.js +0 -1
  418. package/storybook-static/assets/index-Tp9IdbR8.js +0 -1
  419. package/storybook-static/assets/index-WyF3-wTE.js +0 -9
  420. package/storybook-static/assets/index-XSmPROEP.js +0 -1
  421. package/storybook-static/assets/index-Z6wF44KX.js +0 -5
  422. package/storybook-static/assets/index-_RPqOjlQ.js +0 -1
  423. package/storybook-static/assets/index-jrimW4QO.js +0 -1
  424. package/storybook-static/assets/index-nqc17SX4.js +0 -1
  425. package/storybook-static/assets/input-11YRd9gD.js +0 -1
  426. package/storybook-static/assets/jsx-runtime-D_zvdyIk.js +0 -9
  427. package/storybook-static/assets/label-Do8ODIVk.js +0 -1
  428. package/storybook-static/assets/lodash-DDwpuhPG.js +0 -73
  429. package/storybook-static/assets/matchers-7Z3WT2CE-T3xScrR7.js +0 -14
  430. package/storybook-static/assets/minimal-modern-BlYVzfQU.css +0 -1
  431. package/storybook-static/assets/popover-CcciSWAw.js +0 -1
  432. package/storybook-static/assets/preload-helper-Dp1pzeXC.js +0 -1
  433. package/storybook-static/assets/radio-group-DiJ0Y_KQ.js +0 -1
  434. package/storybook-static/assets/react-18-Cr9fq_Ip.js +0 -25
  435. package/storybook-static/assets/react-icons.esm-B_ULMmNU.js +0 -1
  436. package/storybook-static/assets/refresh-cw-BmRDhIV_.js +0 -6
  437. package/storybook-static/assets/schemas-CGNYCiJ6.js +0 -18
  438. package/storybook-static/assets/section-factories-DCCY9R35.js +0 -1
  439. package/storybook-static/assets/select-DDrkxaOg.js +0 -6
  440. package/storybook-static/assets/separator-o5SAUnaJ.js +0 -1
  441. package/storybook-static/assets/settings-2-xWGvvbG6.js +0 -6
  442. package/storybook-static/assets/sheet-C7jhU3XE.js +0 -1
  443. package/storybook-static/assets/shopping-cart-BFlrufvo.js +0 -11
  444. package/storybook-static/assets/sidebar-C8hU1Mxy.js +0 -6
  445. package/storybook-static/assets/skeleton-CjDnQs43.js +0 -1
  446. package/storybook-static/assets/spotify-CUDj7g8m.css +0 -1
  447. package/storybook-static/assets/switch-CKGRuk3u.js +0 -1
  448. package/storybook-static/assets/table-CP3vMqFn.js +0 -1
  449. package/storybook-static/assets/tabs-CopK2m3j.js +0 -1
  450. package/storybook-static/assets/textarea-Dw2vruMl.js +0 -1
  451. package/storybook-static/assets/toggle-DmHbWetf.js +0 -16
  452. package/storybook-static/assets/tooltip-_LqYEYFw.js +0 -1
  453. package/storybook-static/assets/trash-2-xdbApPby.js +0 -11
  454. package/storybook-static/assets/utils-D-KgF5mV.js +0 -1
  455. package/storybook-static/assets/x-B1a4fyWM.js +0 -6
  456. package/storybook-static/favicon-wrapper.svg +0 -46
  457. package/storybook-static/favicon.svg +0 -1
  458. package/storybook-static/iframe.html +0 -687
  459. package/storybook-static/index.d.ts +0 -64
  460. package/storybook-static/index.d.ts.map +0 -1
  461. package/storybook-static/index.html +0 -166
  462. package/storybook-static/index.json +0 -1
  463. package/storybook-static/kit/builder/data-table/components/DataTable.d.ts +0 -37
  464. package/storybook-static/kit/builder/data-table/components/DataTable.d.ts.map +0 -1
  465. package/storybook-static/kit/builder/data-table/components/DataTableColumnHeader.d.ts +0 -8
  466. package/storybook-static/kit/builder/data-table/components/DataTableColumnHeader.d.ts.map +0 -1
  467. package/storybook-static/kit/builder/data-table/components/DataTablePagination.d.ts +0 -6
  468. package/storybook-static/kit/builder/data-table/components/DataTablePagination.d.ts.map +0 -1
  469. package/storybook-static/kit/builder/data-table/components/DataTableViewOptions.d.ts +0 -5
  470. package/storybook-static/kit/builder/data-table/components/DataTableViewOptions.d.ts.map +0 -1
  471. package/storybook-static/kit/builder/data-table/index.d.ts +0 -7
  472. package/storybook-static/kit/builder/data-table/index.d.ts.map +0 -1
  473. package/storybook-static/kit/builder/data-table/types.d.ts +0 -27
  474. package/storybook-static/kit/builder/data-table/types.d.ts.map +0 -1
  475. package/storybook-static/kit/builder/data-table/utils/dotAccessor.d.ts +0 -2
  476. package/storybook-static/kit/builder/data-table/utils/dotAccessor.d.ts.map +0 -1
  477. package/storybook-static/kit/builder/dialog/index.d.ts +0 -3
  478. package/storybook-static/kit/builder/dialog/index.d.ts.map +0 -1
  479. package/storybook-static/kit/builder/dialog/provider.d.ts +0 -26
  480. package/storybook-static/kit/builder/dialog/provider.d.ts.map +0 -1
  481. package/storybook-static/kit/builder/form/components/FormBuilder.d.ts +0 -137
  482. package/storybook-static/kit/builder/form/components/FormBuilder.d.ts.map +0 -1
  483. package/storybook-static/kit/builder/form/components/FormBuilderActions.d.ts +0 -20
  484. package/storybook-static/kit/builder/form/components/FormBuilderActions.d.ts.map +0 -1
  485. package/storybook-static/kit/builder/form/components/FormBuilderField.d.ts +0 -12
  486. package/storybook-static/kit/builder/form/components/FormBuilderField.d.ts.map +0 -1
  487. package/storybook-static/kit/builder/form/components/fields/ArrayField.d.ts +0 -3
  488. package/storybook-static/kit/builder/form/components/fields/ArrayField.d.ts.map +0 -1
  489. package/storybook-static/kit/builder/form/components/fields/AutocompleteField.d.ts +0 -3
  490. package/storybook-static/kit/builder/form/components/fields/AutocompleteField.d.ts.map +0 -1
  491. package/storybook-static/kit/builder/form/components/fields/CheckboxField.d.ts +0 -3
  492. package/storybook-static/kit/builder/form/components/fields/CheckboxField.d.ts.map +0 -1
  493. package/storybook-static/kit/builder/form/components/fields/DateField.d.ts +0 -3
  494. package/storybook-static/kit/builder/form/components/fields/DateField.d.ts.map +0 -1
  495. package/storybook-static/kit/builder/form/components/fields/FileField.d.ts +0 -3
  496. package/storybook-static/kit/builder/form/components/fields/FileField.d.ts.map +0 -1
  497. package/storybook-static/kit/builder/form/components/fields/NumberField.d.ts +0 -3
  498. package/storybook-static/kit/builder/form/components/fields/NumberField.d.ts.map +0 -1
  499. package/storybook-static/kit/builder/form/components/fields/ObjectField.d.ts +0 -3
  500. package/storybook-static/kit/builder/form/components/fields/ObjectField.d.ts.map +0 -1
  501. package/storybook-static/kit/builder/form/components/fields/RadioField.d.ts +0 -3
  502. package/storybook-static/kit/builder/form/components/fields/RadioField.d.ts.map +0 -1
  503. package/storybook-static/kit/builder/form/components/fields/SelectField.d.ts +0 -3
  504. package/storybook-static/kit/builder/form/components/fields/SelectField.d.ts.map +0 -1
  505. package/storybook-static/kit/builder/form/components/fields/SwitchField.d.ts +0 -3
  506. package/storybook-static/kit/builder/form/components/fields/SwitchField.d.ts.map +0 -1
  507. package/storybook-static/kit/builder/form/components/fields/TextField.d.ts +0 -3
  508. package/storybook-static/kit/builder/form/components/fields/TextField.d.ts.map +0 -1
  509. package/storybook-static/kit/builder/form/components/fields/TextareaField.d.ts +0 -3
  510. package/storybook-static/kit/builder/form/components/fields/TextareaField.d.ts.map +0 -1
  511. package/storybook-static/kit/builder/form/components/fields/index.d.ts +0 -14
  512. package/storybook-static/kit/builder/form/components/fields/index.d.ts.map +0 -1
  513. package/storybook-static/kit/builder/form/components/fields/types.d.ts +0 -14
  514. package/storybook-static/kit/builder/form/components/fields/types.d.ts.map +0 -1
  515. package/storybook-static/kit/builder/form/components/index.d.ts +0 -4
  516. package/storybook-static/kit/builder/form/components/index.d.ts.map +0 -1
  517. package/storybook-static/kit/builder/form/index.d.ts +0 -3
  518. package/storybook-static/kit/builder/form/index.d.ts.map +0 -1
  519. package/storybook-static/kit/builder/form/utils/common-forms.d.ts +0 -7
  520. package/storybook-static/kit/builder/form/utils/common-forms.d.ts.map +0 -1
  521. package/storybook-static/kit/builder/form/utils/dependencies.d.ts +0 -41
  522. package/storybook-static/kit/builder/form/utils/dependencies.d.ts.map +0 -1
  523. package/storybook-static/kit/builder/form/utils/field-factories.d.ts +0 -23
  524. package/storybook-static/kit/builder/form/utils/field-factories.d.ts.map +0 -1
  525. package/storybook-static/kit/builder/form/utils/index.d.ts +0 -15
  526. package/storybook-static/kit/builder/form/utils/index.d.ts.map +0 -1
  527. package/storybook-static/kit/builder/form/utils/section-factories.d.ts +0 -7
  528. package/storybook-static/kit/builder/form/utils/section-factories.d.ts.map +0 -1
  529. package/storybook-static/kit/builder/form/utils/transformers.d.ts +0 -6
  530. package/storybook-static/kit/builder/form/utils/transformers.d.ts.map +0 -1
  531. package/storybook-static/kit/builder/form/utils/validations.d.ts +0 -13
  532. package/storybook-static/kit/builder/form/utils/validations.d.ts.map +0 -1
  533. package/storybook-static/kit/builder/form/utils/validators.d.ts +0 -8
  534. package/storybook-static/kit/builder/form/utils/validators.d.ts.map +0 -1
  535. package/storybook-static/kit/builder/page/Page.d.ts +0 -48
  536. package/storybook-static/kit/builder/page/Page.d.ts.map +0 -1
  537. package/storybook-static/kit/builder/page/index.d.ts +0 -2
  538. package/storybook-static/kit/builder/page/index.d.ts.map +0 -1
  539. package/storybook-static/kit/builder/section/SectionBuilder.d.ts +0 -3
  540. package/storybook-static/kit/builder/section/SectionBuilder.d.ts.map +0 -1
  541. package/storybook-static/kit/builder/section/index.d.ts +0 -3
  542. package/storybook-static/kit/builder/section/index.d.ts.map +0 -1
  543. package/storybook-static/kit/builder/section/types.d.ts +0 -70
  544. package/storybook-static/kit/builder/section/types.d.ts.map +0 -1
  545. package/storybook-static/kit/builder/stack-dialog/context.d.ts +0 -3
  546. package/storybook-static/kit/builder/stack-dialog/context.d.ts.map +0 -1
  547. package/storybook-static/kit/builder/stack-dialog/hooks.d.ts +0 -6
  548. package/storybook-static/kit/builder/stack-dialog/hooks.d.ts.map +0 -1
  549. package/storybook-static/kit/builder/stack-dialog/index.d.ts +0 -5
  550. package/storybook-static/kit/builder/stack-dialog/index.d.ts.map +0 -1
  551. package/storybook-static/kit/builder/stack-dialog/provider.d.ts +0 -3
  552. package/storybook-static/kit/builder/stack-dialog/provider.d.ts.map +0 -1
  553. package/storybook-static/kit/builder/stack-dialog/renderer.d.ts +0 -6
  554. package/storybook-static/kit/builder/stack-dialog/renderer.d.ts.map +0 -1
  555. package/storybook-static/kit/builder/stack-dialog/types.d.ts +0 -20
  556. package/storybook-static/kit/builder/stack-dialog/types.d.ts.map +0 -1
  557. package/storybook-static/kit/components/autocomplete/Autocomplete.d.ts +0 -53
  558. package/storybook-static/kit/components/autocomplete/Autocomplete.d.ts.map +0 -1
  559. package/storybook-static/kit/components/autocomplete/index.d.ts +0 -4
  560. package/storybook-static/kit/components/autocomplete/index.d.ts.map +0 -1
  561. package/storybook-static/kit/components/autocomplete/types.d.ts +0 -19
  562. package/storybook-static/kit/components/autocomplete/types.d.ts.map +0 -1
  563. package/storybook-static/kit/components/login/Login.d.ts +0 -29
  564. package/storybook-static/kit/components/login/Login.d.ts.map +0 -1
  565. package/storybook-static/kit/components/login/index.d.ts +0 -2
  566. package/storybook-static/kit/components/login/index.d.ts.map +0 -1
  567. package/storybook-static/kit/components/monthpicker/MonthPicker.d.ts +0 -32
  568. package/storybook-static/kit/components/monthpicker/MonthPicker.d.ts.map +0 -1
  569. package/storybook-static/kit/components/monthpicker/MonthRangePicker.d.ts +0 -48
  570. package/storybook-static/kit/components/monthpicker/MonthRangePicker.d.ts.map +0 -1
  571. package/storybook-static/kit/layouts/admin/components/AdminLayout.d.ts +0 -17
  572. package/storybook-static/kit/layouts/admin/components/AdminLayout.d.ts.map +0 -1
  573. package/storybook-static/kit/layouts/admin/components/ThemeToggle.d.ts +0 -5
  574. package/storybook-static/kit/layouts/admin/components/ThemeToggle.d.ts.map +0 -1
  575. package/storybook-static/kit/layouts/admin/hooks/menu.d.ts +0 -13
  576. package/storybook-static/kit/layouts/admin/hooks/menu.d.ts.map +0 -1
  577. package/storybook-static/kit/layouts/admin/providers/AdminMenuProvider.d.ts +0 -7
  578. package/storybook-static/kit/layouts/admin/providers/AdminMenuProvider.d.ts.map +0 -1
  579. package/storybook-static/kit/layouts/admin/types/index.d.ts +0 -27
  580. package/storybook-static/kit/layouts/admin/types/index.d.ts.map +0 -1
  581. package/storybook-static/kit/providers/ThemeProvider.d.ts +0 -14
  582. package/storybook-static/kit/providers/ThemeProvider.d.ts.map +0 -1
  583. package/storybook-static/nunito-sans-bold-italic.woff2 +0 -0
  584. package/storybook-static/nunito-sans-bold.woff2 +0 -0
  585. package/storybook-static/nunito-sans-italic.woff2 +0 -0
  586. package/storybook-static/nunito-sans-regular.woff2 +0 -0
  587. package/storybook-static/project.json +0 -1
  588. package/storybook-static/sb-addons/a11y-5/manager-bundle.js +0 -5
  589. package/storybook-static/sb-addons/essentials-backgrounds-1/manager-bundle.js +0 -3
  590. package/storybook-static/sb-addons/essentials-measure-2/manager-bundle.js +0 -3
  591. package/storybook-static/sb-addons/essentials-outline-3/manager-bundle.js +0 -3
  592. package/storybook-static/sb-addons/interactions-4/manager-bundle.js +0 -57
  593. package/storybook-static/sb-addons/storybook-core-server-presets-0/common-manager-bundle.js +0 -971
  594. package/storybook-static/sb-addons/storysource-6/manager-bundle.js +0 -3
  595. package/storybook-static/sb-common-assets/favicon-wrapper.svg +0 -46
  596. package/storybook-static/sb-common-assets/favicon.svg +0 -1
  597. package/storybook-static/sb-common-assets/nunito-sans-bold-italic.woff2 +0 -0
  598. package/storybook-static/sb-common-assets/nunito-sans-bold.woff2 +0 -0
  599. package/storybook-static/sb-common-assets/nunito-sans-italic.woff2 +0 -0
  600. package/storybook-static/sb-common-assets/nunito-sans-regular.woff2 +0 -0
  601. package/storybook-static/sb-manager/globals-module-info.js +0 -797
  602. package/storybook-static/sb-manager/globals-runtime.js +0 -69653
  603. package/storybook-static/sb-manager/globals.js +0 -34
  604. package/storybook-static/sb-manager/runtime.js +0 -13181
  605. package/storybook-static/shadcn/hooks/use-mobile.d.ts +0 -2
  606. package/storybook-static/shadcn/hooks/use-mobile.d.ts.map +0 -1
  607. package/storybook-static/shadcn/lib/utils.d.ts +0 -3
  608. package/storybook-static/shadcn/lib/utils.d.ts.map +0 -1
  609. package/storybook-static/shadcn/ui/accordion.d.ts +0 -8
  610. package/storybook-static/shadcn/ui/accordion.d.ts.map +0 -1
  611. package/storybook-static/shadcn/ui/alert-dialog.d.ts +0 -15
  612. package/storybook-static/shadcn/ui/alert-dialog.d.ts.map +0 -1
  613. package/storybook-static/shadcn/ui/alert.d.ts +0 -10
  614. package/storybook-static/shadcn/ui/alert.d.ts.map +0 -1
  615. package/storybook-static/shadcn/ui/aspect-ratio.d.ts +0 -4
  616. package/storybook-static/shadcn/ui/aspect-ratio.d.ts.map +0 -1
  617. package/storybook-static/shadcn/ui/avatar.d.ts +0 -7
  618. package/storybook-static/shadcn/ui/avatar.d.ts.map +0 -1
  619. package/storybook-static/shadcn/ui/badge.d.ts +0 -10
  620. package/storybook-static/shadcn/ui/badge.d.ts.map +0 -1
  621. package/storybook-static/shadcn/ui/breadcrumb.d.ts +0 -12
  622. package/storybook-static/shadcn/ui/breadcrumb.d.ts.map +0 -1
  623. package/storybook-static/shadcn/ui/button.d.ts +0 -11
  624. package/storybook-static/shadcn/ui/button.d.ts.map +0 -1
  625. package/storybook-static/shadcn/ui/calendar.d.ts +0 -9
  626. package/storybook-static/shadcn/ui/calendar.d.ts.map +0 -1
  627. package/storybook-static/shadcn/ui/card.d.ts +0 -10
  628. package/storybook-static/shadcn/ui/card.d.ts.map +0 -1
  629. package/storybook-static/shadcn/ui/carousel.d.ts +0 -20
  630. package/storybook-static/shadcn/ui/carousel.d.ts.map +0 -1
  631. package/storybook-static/shadcn/ui/chart.d.ts +0 -41
  632. package/storybook-static/shadcn/ui/chart.d.ts.map +0 -1
  633. package/storybook-static/shadcn/ui/checkbox.d.ts +0 -5
  634. package/storybook-static/shadcn/ui/checkbox.d.ts.map +0 -1
  635. package/storybook-static/shadcn/ui/collapsible.d.ts +0 -6
  636. package/storybook-static/shadcn/ui/collapsible.d.ts.map +0 -1
  637. package/storybook-static/shadcn/ui/command.d.ts +0 -19
  638. package/storybook-static/shadcn/ui/command.d.ts.map +0 -1
  639. package/storybook-static/shadcn/ui/context-menu.d.ts +0 -26
  640. package/storybook-static/shadcn/ui/context-menu.d.ts.map +0 -1
  641. package/storybook-static/shadcn/ui/dialog.d.ts +0 -16
  642. package/storybook-static/shadcn/ui/dialog.d.ts.map +0 -1
  643. package/storybook-static/shadcn/ui/drawer.d.ts +0 -14
  644. package/storybook-static/shadcn/ui/drawer.d.ts.map +0 -1
  645. package/storybook-static/shadcn/ui/dropdown-menu.d.ts +0 -26
  646. package/storybook-static/shadcn/ui/dropdown-menu.d.ts.map +0 -1
  647. package/storybook-static/shadcn/ui/form.d.ts +0 -25
  648. package/storybook-static/shadcn/ui/form.d.ts.map +0 -1
  649. package/storybook-static/shadcn/ui/hover-card.d.ts +0 -7
  650. package/storybook-static/shadcn/ui/hover-card.d.ts.map +0 -1
  651. package/storybook-static/shadcn/ui/input-otp.d.ts +0 -12
  652. package/storybook-static/shadcn/ui/input-otp.d.ts.map +0 -1
  653. package/storybook-static/shadcn/ui/input.d.ts +0 -4
  654. package/storybook-static/shadcn/ui/input.d.ts.map +0 -1
  655. package/storybook-static/shadcn/ui/label.d.ts +0 -5
  656. package/storybook-static/shadcn/ui/label.d.ts.map +0 -1
  657. package/storybook-static/shadcn/ui/menubar.d.ts +0 -27
  658. package/storybook-static/shadcn/ui/menubar.d.ts.map +0 -1
  659. package/storybook-static/shadcn/ui/navigation-menu.d.ts +0 -15
  660. package/storybook-static/shadcn/ui/navigation-menu.d.ts.map +0 -1
  661. package/storybook-static/shadcn/ui/pagination.d.ts +0 -14
  662. package/storybook-static/shadcn/ui/pagination.d.ts.map +0 -1
  663. package/storybook-static/shadcn/ui/popover.d.ts +0 -8
  664. package/storybook-static/shadcn/ui/popover.d.ts.map +0 -1
  665. package/storybook-static/shadcn/ui/progress.d.ts +0 -5
  666. package/storybook-static/shadcn/ui/progress.d.ts.map +0 -1
  667. package/storybook-static/shadcn/ui/radio-group.d.ts +0 -6
  668. package/storybook-static/shadcn/ui/radio-group.d.ts.map +0 -1
  669. package/storybook-static/shadcn/ui/resizable.d.ts +0 -9
  670. package/storybook-static/shadcn/ui/resizable.d.ts.map +0 -1
  671. package/storybook-static/shadcn/ui/scroll-area.d.ts +0 -6
  672. package/storybook-static/shadcn/ui/scroll-area.d.ts.map +0 -1
  673. package/storybook-static/shadcn/ui/select.d.ts +0 -16
  674. package/storybook-static/shadcn/ui/select.d.ts.map +0 -1
  675. package/storybook-static/shadcn/ui/separator.d.ts +0 -5
  676. package/storybook-static/shadcn/ui/separator.d.ts.map +0 -1
  677. package/storybook-static/shadcn/ui/sheet.d.ts +0 -14
  678. package/storybook-static/shadcn/ui/sheet.d.ts.map +0 -1
  679. package/storybook-static/shadcn/ui/sidebar.d.ts +0 -70
  680. package/storybook-static/shadcn/ui/sidebar.d.ts.map +0 -1
  681. package/storybook-static/shadcn/ui/skeleton.d.ts +0 -3
  682. package/storybook-static/shadcn/ui/skeleton.d.ts.map +0 -1
  683. package/storybook-static/shadcn/ui/slider.d.ts +0 -5
  684. package/storybook-static/shadcn/ui/slider.d.ts.map +0 -1
  685. package/storybook-static/shadcn/ui/sonner.d.ts +0 -4
  686. package/storybook-static/shadcn/ui/sonner.d.ts.map +0 -1
  687. package/storybook-static/shadcn/ui/switch.d.ts +0 -5
  688. package/storybook-static/shadcn/ui/switch.d.ts.map +0 -1
  689. package/storybook-static/shadcn/ui/table.d.ts +0 -11
  690. package/storybook-static/shadcn/ui/table.d.ts.map +0 -1
  691. package/storybook-static/shadcn/ui/tabs.d.ts +0 -8
  692. package/storybook-static/shadcn/ui/tabs.d.ts.map +0 -1
  693. package/storybook-static/shadcn/ui/textarea.d.ts +0 -4
  694. package/storybook-static/shadcn/ui/textarea.d.ts.map +0 -1
  695. package/storybook-static/shadcn/ui/toggle-group.d.ts +0 -8
  696. package/storybook-static/shadcn/ui/toggle-group.d.ts.map +0 -1
  697. package/storybook-static/shadcn/ui/toggle.d.ts +0 -10
  698. package/storybook-static/shadcn/ui/toggle.d.ts.map +0 -1
  699. package/storybook-static/shadcn/ui/tooltip.d.ts +0 -8
  700. package/storybook-static/shadcn/ui/tooltip.d.ts.map +0 -1
  701. package/storybook-static/vite-inject-mocker-entry.js +0 -18
@@ -1,626 +1,626 @@
1
- import { useCallback, useEffect, useRef, useState, memo } from 'react';
2
- import { useDropzone, type Accept } from 'react-dropzone';
3
- import { cn } from '../../../shadcn/lib/utils';
4
- import { Button } from '../../../shadcn/ui/button';
5
- import { Progress } from '../../../shadcn/ui/progress';
6
- import {
7
- Tooltip,
8
- TooltipContent,
9
- TooltipProvider,
10
- TooltipTrigger,
11
- } from '../../../shadcn/ui/tooltip';
12
- import {
13
- Loader2,
14
- UploadCloud,
15
- Image as ImageIcon,
16
- File as FileIcon,
17
- FileText,
18
- FileCode,
19
- Video,
20
- Music,
21
- Archive,
22
- CheckCircle2,
23
- XCircle,
24
- Download,
25
- Trash2,
26
- RotateCcw,
27
- } from 'lucide-react';
28
- import type { FileRecord, FileUploaderProps } from './types';
29
-
30
- // Cache preview URLs per File instance without mutating the File object
31
- const previewUrlMap: WeakMap<File, string> = new WeakMap();
32
-
33
- function formatBytes(bytes?: number) {
34
- if (!bytes && bytes !== 0) return '';
35
- const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB'];
36
- if (bytes === 0) return '0 Byte';
37
- const i = Math.floor(Math.log(bytes) / Math.log(1024));
38
- return `${(bytes / 1024 ** i).toFixed(2)} ${sizes[i]}`;
39
- }
40
-
41
- function isImageType(type?: string, name?: string) {
42
- if (!type && name) {
43
- const ext = name.toLowerCase().split('.').pop();
44
- if (!ext) return false;
45
- return [
46
- 'png',
47
- 'jpg',
48
- 'jpeg',
49
- 'webp',
50
- 'gif',
51
- 'bmp',
52
- 'svg',
53
- 'heic',
54
- 'heif',
55
- ].includes(ext);
56
- }
57
- return !!type?.startsWith('image/');
58
- }
59
-
60
- function pickIconByType(type?: string, name?: string) {
61
- const t = (type || '').toLowerCase();
62
- const ext = (name?.split('.').pop() || '').toLowerCase();
63
- if (
64
- t.startsWith('image/') ||
65
- [
66
- 'png',
67
- 'jpg',
68
- 'jpeg',
69
- 'webp',
70
- 'gif',
71
- 'bmp',
72
- 'svg',
73
- 'heic',
74
- 'heif',
75
- ].includes(ext)
76
- )
77
- return <ImageIcon className="h-8 w-8" />;
78
- if (t.startsWith('video/') || ['mp4', 'mov', 'webm', 'mkv'].includes(ext))
79
- return <Video className="h-8 w-8" />;
80
- if (t.startsWith('audio/') || ['mp3', 'wav', 'aac', 'flac'].includes(ext))
81
- return <Music className="h-8 w-8" />;
82
- if (['zip', 'rar', '7z', 'tar', 'gz'].includes(ext))
83
- return <Archive className="h-8 w-8" />;
84
- if (['txt', 'md', 'rtf'].includes(ext))
85
- return <FileText className="h-8 w-8" />;
86
- if (
87
- ['js', 'ts', 'tsx', 'json', 'yml', 'yaml', 'xml', 'html', 'css'].includes(
88
- ext,
89
- )
90
- )
91
- return <FileCode className="h-8 w-8" />;
92
- return <FileIcon className="h-8 w-8" />;
93
- }
94
-
95
- function getPreviewUrl(file: FileRecord) {
96
- if (file.thumbnailUrl) return file.thumbnailUrl;
97
- if (file.url && isImageType(file.type, file.name)) return file.url;
98
- if (file.file && isImageType(file.file.type, file.file.name)) {
99
- const cached = previewUrlMap.get(file.file);
100
- if (cached) return cached;
101
- const created = URL.createObjectURL(file.file);
102
- previewUrlMap.set(file.file, created);
103
- return created;
104
- }
105
- return undefined;
106
- }
107
-
108
- export function FileUploader({
109
- value,
110
- defaultValue,
111
- onChange,
112
- uploader,
113
- onUploadSuccess,
114
- onUploadError,
115
- onRemove,
116
- withRetry = true,
117
- onRetry,
118
- onRetryAll,
119
- multiple = true,
120
- maxFiles,
121
- accept,
122
- layout = 'grid',
123
- disabled,
124
- withDownload = true,
125
- placeholder = 'Drag and drop files here, or click to select',
126
- className,
127
- }: FileUploaderProps) {
128
- const isControlled = value !== undefined;
129
- const [files, setFiles] = useState<FileRecord[]>(() => defaultValue ?? []);
130
-
131
- const prevUrlsRef = useRef<Set<string>>(new Set());
132
- useEffect(() => {
133
- return () => {
134
- prevUrlsRef.current.forEach((url) => {
135
- URL.revokeObjectURL(url);
136
- });
137
- prevUrlsRef.current.clear();
138
- };
139
- }, []);
140
-
141
- const setFilesAndEmit = useCallback(
142
- (updater: FileRecord[] | ((prev: FileRecord[]) => FileRecord[])) => {
143
- setFiles((prev) => {
144
- let next: FileRecord[];
145
- if (typeof updater === 'function') {
146
- const fn = updater as (p: FileRecord[]) => FileRecord[];
147
- next = fn(prev);
148
- } else {
149
- next = updater;
150
- }
151
- onChange?.(next);
152
- return next;
153
- });
154
- },
155
- [onChange],
156
- );
157
-
158
- useEffect(() => {
159
- if (isControlled && value) setFiles(value);
160
- if (isControlled && !value) setFiles([]);
161
- }, [isControlled, value]);
162
-
163
- // Normalize incoming items (default/controlled): if a file has a URL but no status, mark as success
164
- useEffect(() => {
165
- setFiles((prev) => {
166
- let changed = false;
167
- const next = prev.map((f) => {
168
- if ((f.url || f.thumbnailUrl) && !f.status) {
169
- changed = true;
170
- return { ...f, status: 'success' as const, progress: 100 };
171
- }
172
- return f;
173
- });
174
- return changed ? next : prev;
175
- });
176
- }, []);
177
-
178
- const handleRemove = useCallback(
179
- async (idx: number) => {
180
- const target = files[idx];
181
- try {
182
- await onRemove?.(target);
183
- } catch {
184
- // ignore removal error UX for now
185
- }
186
- setFilesAndEmit((prev) => prev.filter((_, i) => i !== idx));
187
- },
188
- [files, onRemove, setFilesAndEmit],
189
- );
190
-
191
- const startUpload = useCallback(
192
- async (index: number, f: File) => {
193
- if (!uploader) return;
194
- try {
195
- setFilesAndEmit((prev) => {
196
- const n = [...prev];
197
- n[index] = {
198
- ...n[index],
199
- status: 'uploading',
200
- progress: 0,
201
- errorMessage: undefined,
202
- };
203
- return n;
204
- });
205
- const result = await uploader(f, (pct) => {
206
- setFilesAndEmit((prev) => {
207
- const n = [...prev];
208
- if (!n[index]) return prev;
209
- n[index] = {
210
- ...n[index],
211
- progress: Math.min(100, Math.max(0, Math.round(pct))),
212
- status: 'uploading',
213
- };
214
- return n;
215
- });
216
- });
217
- setFilesAndEmit((prev) => {
218
- const n = [...prev];
219
- if (!n[index]) return prev;
220
- n[index] = {
221
- ...n[index],
222
- ...result,
223
- status: 'success',
224
- progress: 100,
225
- };
226
- return n;
227
- });
228
- const uploaded = (isControlled ? value : files)[index] ?? undefined;
229
- if (uploaded) onUploadSuccess?.(uploaded);
230
- } catch (err) {
231
- setFilesAndEmit((prev) => {
232
- const n = [...prev];
233
- if (!n[index]) return prev;
234
- const msg =
235
- err &&
236
- typeof err === 'object' &&
237
- 'message' in (err as Record<string, unknown>)
238
- ? String((err as { message?: unknown }).message)
239
- : 'Upload failed';
240
- n[index] = { ...n[index], status: 'error', errorMessage: msg };
241
- return n;
242
- });
243
- const failed = (isControlled ? value : files)[index] ?? undefined;
244
- if (failed) onUploadError?.(failed, err);
245
- }
246
- },
247
- [
248
- files,
249
- isControlled,
250
- onUploadError,
251
- onUploadSuccess,
252
- setFilesAndEmit,
253
- uploader,
254
- value,
255
- ],
256
- );
257
-
258
- const handleRetry = useCallback(
259
- (idx: number) => {
260
- const fr = files[idx];
261
- if (!uploader || !fr?.file) return;
262
- onRetry?.(fr);
263
- void startUpload(idx, fr.file);
264
- },
265
- [files, onRetry, startUpload, uploader],
266
- );
267
-
268
- const handleRetryAll = useCallback(() => {
269
- if (!uploader) return;
270
- const failedFiles = files.filter((f) => f.status === 'error' && !!f.file);
271
- if (failedFiles.length === 0) return;
272
- onRetryAll?.(failedFiles);
273
- failedFiles.forEach((fr) => {
274
- const idx = files.indexOf(fr);
275
- if (idx >= 0 && fr.file) {
276
- onRetry?.(fr);
277
- void startUpload(idx, fr.file);
278
- }
279
- });
280
- }, [files, onRetry, onRetryAll, startUpload, uploader]);
281
-
282
- const onDrop = useCallback(
283
- (acceptedFiles: File[]) => {
284
- if (!acceptedFiles?.length) return;
285
- setFilesAndEmit((prev) => {
286
- const existing = [...prev];
287
- const capacity =
288
- typeof maxFiles === 'number'
289
- ? Math.max(0, maxFiles - existing.length)
290
- : acceptedFiles.length;
291
- const incoming = acceptedFiles
292
- .slice(0, capacity)
293
- .map<FileRecord>((f) => ({
294
- id: undefined,
295
- url: undefined,
296
- thumbnailUrl: undefined,
297
- file: f,
298
- name: f.name,
299
- size: f.size,
300
- type: f.type,
301
- status: uploader ? 'uploading' : 'idle',
302
- progress: uploader ? 0 : undefined,
303
- }));
304
- const next = multiple
305
- ? [...existing, ...incoming]
306
- : ([incoming[0]].filter(Boolean) as FileRecord[]);
307
- return next;
308
- });
309
-
310
- if (uploader) {
311
- const baseIndex = (isControlled ? value : files)?.length ?? 0;
312
- const capacity =
313
- typeof maxFiles === 'number'
314
- ? Math.max(
315
- 0,
316
- maxFiles - ((isControlled ? value : files)?.length ?? 0),
317
- )
318
- : acceptedFiles.length;
319
- acceptedFiles.slice(0, capacity).forEach((f, i) => {
320
- const index = multiple ? baseIndex + i : 0;
321
- void startUpload(index, f);
322
- });
323
- }
324
- },
325
- [
326
- files,
327
- isControlled,
328
- maxFiles,
329
- multiple,
330
- startUpload,
331
- uploader,
332
- value,
333
- setFilesAndEmit,
334
- ],
335
- );
336
-
337
- const disabledBecauseFull = !!maxFiles && files.length >= maxFiles;
338
-
339
- const { getRootProps, getInputProps, isDragActive, isDragReject, open } =
340
- useDropzone({
341
- onDrop,
342
- multiple,
343
- maxFiles: multiple ? maxFiles : 1,
344
- accept: accept as Accept | undefined,
345
- noClick: true,
346
- noKeyboard: true,
347
- disabled: disabled || disabledBecauseFull,
348
- });
349
-
350
- useEffect(() => {
351
- const urls = new Set<string>();
352
- files.forEach((fr) => {
353
- const preview = getPreviewUrl(fr);
354
- if (preview) urls.add(preview);
355
- });
356
- prevUrlsRef.current.forEach((prev) => {
357
- if (!urls.has(prev)) URL.revokeObjectURL(prev);
358
- });
359
- prevUrlsRef.current = urls;
360
- }, [files]);
361
-
362
- const rootClasses = cn(
363
- 'w-full border border-dashed rounded-md p-4 text-sm transition-colors bg-background',
364
- 'hover:border-foreground/50',
365
- isDragActive && 'border-primary',
366
- isDragReject && 'border-destructive',
367
- (disabled || disabledBecauseFull) && 'opacity-50 pointer-events-none',
368
- );
369
-
370
- // (renderThumb removed; inlined into FileItem for better memoization)
371
-
372
- // Stable handler refs so children don't see new function identities each render
373
- const removeRef = useRef(handleRemove);
374
- useEffect(() => {
375
- removeRef.current = handleRemove;
376
- }, [handleRemove]);
377
- const retryRef = useRef(handleRetry);
378
- useEffect(() => {
379
- retryRef.current = handleRetry;
380
- }, [handleRetry]);
381
- const onRemoveAt = useCallback((i: number) => {
382
- removeRef.current(i);
383
- }, []);
384
- const onRetryAt = useCallback((i: number) => {
385
- retryRef.current(i);
386
- }, []);
387
-
388
- type FileItemProps = {
389
- fr: FileRecord;
390
- idx: number;
391
- layout: 'grid' | 'list';
392
- withDownload: boolean;
393
- uploaderPresent: boolean;
394
- onRemove: (idx: number) => void;
395
- onRetry: (idx: number) => void;
396
- };
397
-
398
- const FileItem = memo(
399
- function FileItem({
400
- fr,
401
- idx,
402
- layout,
403
- withDownload,
404
- uploaderPresent,
405
- onRemove,
406
- onRetry,
407
- }: FileItemProps) {
408
- const name = fr.name;
409
- const size = formatBytes(fr.size);
410
- const error = fr.status === 'error' ? fr.errorMessage : undefined;
411
- const preview = getPreviewUrl(fr);
412
- return (
413
- <div
414
- className={cn(
415
- 'flex items-center gap-3 border rounded-md p-2 bg-card',
416
- layout === 'grid' ? 'flex-col items-stretch' : 'flex-row',
417
- )}
418
- >
419
- <div className={cn(layout === 'grid' ? 'self-center' : '')}>
420
- {
421
- <div
422
- className={cn(
423
- 'relative overflow-hidden bg-muted/40 border rounded-md flex items-center justify-center',
424
- layout === 'grid' ? 'h-28 w-28' : 'h-16 w-16',
425
- )}
426
- >
427
- {preview ? (
428
- <img
429
- src={preview}
430
- alt={fr.name}
431
- className="object-cover w-full h-full"
432
- />
433
- ) : (
434
- <div className="flex items-center justify-center text-muted-foreground">
435
- {pickIconByType(fr.type, fr.name)}
436
- </div>
437
- )}
438
- {fr.status === 'uploading' ? (
439
- <div className="absolute inset-0 bg-black/40 flex items-center justify-center">
440
- <Loader2 className="h-6 w-6 text-white animate-spin" />
441
- </div>
442
- ) : null}
443
- {fr.status === 'success' ? (
444
- <div className="absolute top-1 right-1 text-green-500">
445
- <CheckCircle2 className="h-5 w-5 drop-shadow" />
446
- </div>
447
- ) : null}
448
- {fr.status === 'error' ? (
449
- <div className="absolute top-1 right-1 text-red-500">
450
- <XCircle className="h-5 w-5 drop-shadow" />
451
- </div>
452
- ) : null}
453
- </div>
454
- }
455
- </div>
456
- <div
457
- className={cn('min-w-0 flex-1', layout === 'grid' ? 'mt-2' : '')}
458
- >
459
- <div className="flex items-center justify-between gap-2">
460
- <div className="min-w-0">
461
- <div className="truncate font-medium" title={name}>
462
- {name}
463
- </div>
464
- <div className="text-xs text-muted-foreground">{size}</div>
465
- </div>
466
- <div className="flex items-center gap-1">
467
- {fr.status === 'error' ? (
468
- <Tooltip>
469
- <TooltipTrigger asChild>
470
- <Button
471
- size="icon"
472
- variant="ghost"
473
- onClick={() => onRetry(idx)}
474
- aria-label="Retry upload"
475
- disabled={!uploaderPresent || !fr.file}
476
- >
477
- <RotateCcw className="h-4 w-4" />
478
- </Button>
479
- </TooltipTrigger>
480
- <TooltipContent>Retry</TooltipContent>
481
- </Tooltip>
482
- ) : null}
483
- {withDownload && (fr.url || fr.thumbnailUrl) ? (
484
- <Tooltip>
485
- <TooltipTrigger asChild>
486
- <Button
487
- size="icon"
488
- variant="ghost"
489
- onClick={() => {
490
- const url = fr.url ?? fr.thumbnailUrl;
491
- if (url) {
492
- window.open(url, '_blank', 'noopener,noreferrer');
493
- }
494
- }}
495
- aria-label="Download"
496
- >
497
- <Download className="h-4 w-4" />
498
- </Button>
499
- </TooltipTrigger>
500
- <TooltipContent>Download</TooltipContent>
501
- </Tooltip>
502
- ) : null}
503
- <Tooltip>
504
- <TooltipTrigger asChild>
505
- <Button
506
- size="icon"
507
- variant="ghost"
508
- onClick={() => onRemove(idx)}
509
- aria-label="Remove"
510
- >
511
- <Trash2 className="h-4 w-4" />
512
- </Button>
513
- </TooltipTrigger>
514
- <TooltipContent>Remove</TooltipContent>
515
- </Tooltip>
516
- </div>
517
- </div>
518
- {fr.status === 'uploading' ? (
519
- <div className="mt-2">
520
- <Progress value={fr.progress ?? 0} />
521
- </div>
522
- ) : null}
523
- {error ? (
524
- <div className="mt-2 text-xs text-destructive">{error}</div>
525
- ) : null}
526
- </div>
527
- </div>
528
- );
529
- },
530
- (prev, next) =>
531
- prev.fr === next.fr &&
532
- prev.layout === next.layout &&
533
- prev.withDownload === next.withDownload &&
534
- prev.uploaderPresent === next.uploaderPresent &&
535
- prev.idx === next.idx,
536
- );
537
-
538
- return (
539
- <div className={cn('space-y-3', className)}>
540
- <div {...getRootProps({ className: rootClasses })}>
541
- <input
542
- {...getInputProps({
543
- onClick: (e) => {
544
- (e.target as HTMLInputElement).value = '';
545
- },
546
- })}
547
- />
548
- <div className="flex items-center justify-between gap-3">
549
- <div className="flex items-center gap-2 text-muted-foreground">
550
- <Button
551
- size="sm"
552
- variant="secondary"
553
- disabled={disabled || disabledBecauseFull}
554
- onClick={(e) => {
555
- e.preventDefault();
556
- e.stopPropagation();
557
- open();
558
- }}
559
- >
560
- Select files
561
- </Button>
562
- <UploadCloud className="h-5 w-5" />
563
- <div>
564
- <div className="font-medium">
565
- {disabledBecauseFull
566
- ? 'File limit reached'
567
- : isDragActive
568
- ? 'Drop the files here'
569
- : placeholder}
570
- </div>
571
- <div className="text-xs">
572
- {accept ? 'Specific file types only' : 'Any file type'}
573
- {typeof maxFiles === 'number'
574
- ? ` • Up to ${maxFiles} file${maxFiles > 1 ? 's' : ''}`
575
- : ''}
576
- </div>
577
- </div>
578
- </div>
579
- {withRetry && (
580
- <div className="flex items-center gap-2">
581
- <Button
582
- size="sm"
583
- variant="ghost"
584
- onClick={handleRetryAll}
585
- disabled={
586
- !!disabled ||
587
- !uploader ||
588
- files.every((f) => f.status !== 'error' || !f.file)
589
- }
590
- >
591
- <RotateCcw className="mr-1 h-4 w-4" /> Retry failed
592
- </Button>
593
- </div>
594
- )}
595
- </div>
596
- </div>
597
-
598
- <TooltipProvider>
599
- <div
600
- className={cn(
601
- layout === 'grid' ? 'flex flex-wrap gap-3' : 'flex flex-col gap-2',
602
- )}
603
- >
604
- {files.length === 0 ? (
605
- <div className="text-sm text-muted-foreground">No files</div>
606
- ) : (
607
- files.map((fr, i) => (
608
- <FileItem
609
- key={`${fr.name}-${i}`}
610
- fr={fr}
611
- idx={i}
612
- layout={layout}
613
- withDownload={withDownload}
614
- uploaderPresent={!!uploader}
615
- onRemove={onRemoveAt}
616
- onRetry={onRetryAt}
617
- />
618
- ))
619
- )}
620
- </div>
621
- </TooltipProvider>
622
- </div>
623
- );
624
- }
625
-
626
- export default FileUploader;
1
+ import { useCallback, useEffect, useRef, useState, memo } from 'react';
2
+ import { useDropzone, type Accept } from 'react-dropzone';
3
+ import { cn } from '../../../shadcn/lib/utils';
4
+ import { Button } from '../../../shadcn/ui/button';
5
+ import { Progress } from '../../../shadcn/ui/progress';
6
+ import {
7
+ Tooltip,
8
+ TooltipContent,
9
+ TooltipProvider,
10
+ TooltipTrigger,
11
+ } from '../../../shadcn/ui/tooltip';
12
+ import {
13
+ Loader2,
14
+ UploadCloud,
15
+ Image as ImageIcon,
16
+ File as FileIcon,
17
+ FileText,
18
+ FileCode,
19
+ Video,
20
+ Music,
21
+ Archive,
22
+ CheckCircle2,
23
+ XCircle,
24
+ Download,
25
+ Trash2,
26
+ RotateCcw,
27
+ } from 'lucide-react';
28
+ import type { FileRecord, FileUploaderProps } from './types';
29
+
30
+ // Cache preview URLs per File instance without mutating the File object
31
+ const previewUrlMap: WeakMap<File, string> = new WeakMap();
32
+
33
+ function formatBytes(bytes?: number) {
34
+ if (!bytes && bytes !== 0) return '';
35
+ const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB'];
36
+ if (bytes === 0) return '0 Byte';
37
+ const i = Math.floor(Math.log(bytes) / Math.log(1024));
38
+ return `${(bytes / 1024 ** i).toFixed(2)} ${sizes[i]}`;
39
+ }
40
+
41
+ function isImageType(type?: string, name?: string) {
42
+ if (!type && name) {
43
+ const ext = name.toLowerCase().split('.').pop();
44
+ if (!ext) return false;
45
+ return [
46
+ 'png',
47
+ 'jpg',
48
+ 'jpeg',
49
+ 'webp',
50
+ 'gif',
51
+ 'bmp',
52
+ 'svg',
53
+ 'heic',
54
+ 'heif',
55
+ ].includes(ext);
56
+ }
57
+ return !!type?.startsWith('image/');
58
+ }
59
+
60
+ function pickIconByType(type?: string, name?: string) {
61
+ const t = (type || '').toLowerCase();
62
+ const ext = (name?.split('.').pop() || '').toLowerCase();
63
+ if (
64
+ t.startsWith('image/') ||
65
+ [
66
+ 'png',
67
+ 'jpg',
68
+ 'jpeg',
69
+ 'webp',
70
+ 'gif',
71
+ 'bmp',
72
+ 'svg',
73
+ 'heic',
74
+ 'heif',
75
+ ].includes(ext)
76
+ )
77
+ return <ImageIcon className="h-8 w-8" />;
78
+ if (t.startsWith('video/') || ['mp4', 'mov', 'webm', 'mkv'].includes(ext))
79
+ return <Video className="h-8 w-8" />;
80
+ if (t.startsWith('audio/') || ['mp3', 'wav', 'aac', 'flac'].includes(ext))
81
+ return <Music className="h-8 w-8" />;
82
+ if (['zip', 'rar', '7z', 'tar', 'gz'].includes(ext))
83
+ return <Archive className="h-8 w-8" />;
84
+ if (['txt', 'md', 'rtf'].includes(ext))
85
+ return <FileText className="h-8 w-8" />;
86
+ if (
87
+ ['js', 'ts', 'tsx', 'json', 'yml', 'yaml', 'xml', 'html', 'css'].includes(
88
+ ext,
89
+ )
90
+ )
91
+ return <FileCode className="h-8 w-8" />;
92
+ return <FileIcon className="h-8 w-8" />;
93
+ }
94
+
95
+ function getPreviewUrl(file: FileRecord) {
96
+ if (file.thumbnailUrl) return file.thumbnailUrl;
97
+ if (file.url && isImageType(file.type, file.name)) return file.url;
98
+ if (file.file && isImageType(file.file.type, file.file.name)) {
99
+ const cached = previewUrlMap.get(file.file);
100
+ if (cached) return cached;
101
+ const created = URL.createObjectURL(file.file);
102
+ previewUrlMap.set(file.file, created);
103
+ return created;
104
+ }
105
+ return undefined;
106
+ }
107
+
108
+ export function FileUploader({
109
+ value,
110
+ defaultValue,
111
+ onChange,
112
+ uploader,
113
+ onUploadSuccess,
114
+ onUploadError,
115
+ onRemove,
116
+ withRetry = true,
117
+ onRetry,
118
+ onRetryAll,
119
+ multiple = true,
120
+ maxFiles,
121
+ accept,
122
+ layout = 'grid',
123
+ disabled,
124
+ withDownload = true,
125
+ placeholder = 'Drag and drop files here, or click to select',
126
+ className,
127
+ }: FileUploaderProps) {
128
+ const isControlled = value !== undefined;
129
+ const [files, setFiles] = useState<FileRecord[]>(() => defaultValue ?? []);
130
+
131
+ const prevUrlsRef = useRef<Set<string>>(new Set());
132
+ useEffect(() => {
133
+ return () => {
134
+ prevUrlsRef.current.forEach((url) => {
135
+ URL.revokeObjectURL(url);
136
+ });
137
+ prevUrlsRef.current.clear();
138
+ };
139
+ }, []);
140
+
141
+ const setFilesAndEmit = useCallback(
142
+ (updater: FileRecord[] | ((prev: FileRecord[]) => FileRecord[])) => {
143
+ setFiles((prev) => {
144
+ let next: FileRecord[];
145
+ if (typeof updater === 'function') {
146
+ const fn = updater as (p: FileRecord[]) => FileRecord[];
147
+ next = fn(prev);
148
+ } else {
149
+ next = updater;
150
+ }
151
+ onChange?.(next);
152
+ return next;
153
+ });
154
+ },
155
+ [onChange],
156
+ );
157
+
158
+ useEffect(() => {
159
+ if (isControlled && value) setFiles(value);
160
+ if (isControlled && !value) setFiles([]);
161
+ }, [isControlled, value]);
162
+
163
+ // Normalize incoming items (default/controlled): if a file has a URL but no status, mark as success
164
+ useEffect(() => {
165
+ setFiles((prev) => {
166
+ let changed = false;
167
+ const next = prev.map((f) => {
168
+ if ((f.url || f.thumbnailUrl) && !f.status) {
169
+ changed = true;
170
+ return { ...f, status: 'success' as const, progress: 100 };
171
+ }
172
+ return f;
173
+ });
174
+ return changed ? next : prev;
175
+ });
176
+ }, []);
177
+
178
+ const handleRemove = useCallback(
179
+ async (idx: number) => {
180
+ const target = files[idx];
181
+ try {
182
+ await onRemove?.(target);
183
+ } catch {
184
+ // ignore removal error UX for now
185
+ }
186
+ setFilesAndEmit((prev) => prev.filter((_, i) => i !== idx));
187
+ },
188
+ [files, onRemove, setFilesAndEmit],
189
+ );
190
+
191
+ const startUpload = useCallback(
192
+ async (index: number, f: File) => {
193
+ if (!uploader) return;
194
+ try {
195
+ setFilesAndEmit((prev) => {
196
+ const n = [...prev];
197
+ n[index] = {
198
+ ...n[index],
199
+ status: 'uploading',
200
+ progress: 0,
201
+ errorMessage: undefined,
202
+ };
203
+ return n;
204
+ });
205
+ const result = await uploader(f, (pct) => {
206
+ setFilesAndEmit((prev) => {
207
+ const n = [...prev];
208
+ if (!n[index]) return prev;
209
+ n[index] = {
210
+ ...n[index],
211
+ progress: Math.min(100, Math.max(0, Math.round(pct))),
212
+ status: 'uploading',
213
+ };
214
+ return n;
215
+ });
216
+ });
217
+ setFilesAndEmit((prev) => {
218
+ const n = [...prev];
219
+ if (!n[index]) return prev;
220
+ n[index] = {
221
+ ...n[index],
222
+ ...result,
223
+ status: 'success',
224
+ progress: 100,
225
+ };
226
+ return n;
227
+ });
228
+ const uploaded = (isControlled ? value : files)[index] ?? undefined;
229
+ if (uploaded) onUploadSuccess?.(uploaded);
230
+ } catch (err) {
231
+ setFilesAndEmit((prev) => {
232
+ const n = [...prev];
233
+ if (!n[index]) return prev;
234
+ const msg =
235
+ err &&
236
+ typeof err === 'object' &&
237
+ 'message' in (err as Record<string, unknown>)
238
+ ? String((err as { message?: unknown }).message)
239
+ : 'Upload failed';
240
+ n[index] = { ...n[index], status: 'error', errorMessage: msg };
241
+ return n;
242
+ });
243
+ const failed = (isControlled ? value : files)[index] ?? undefined;
244
+ if (failed) onUploadError?.(failed, err);
245
+ }
246
+ },
247
+ [
248
+ files,
249
+ isControlled,
250
+ onUploadError,
251
+ onUploadSuccess,
252
+ setFilesAndEmit,
253
+ uploader,
254
+ value,
255
+ ],
256
+ );
257
+
258
+ const handleRetry = useCallback(
259
+ (idx: number) => {
260
+ const fr = files[idx];
261
+ if (!uploader || !fr?.file) return;
262
+ onRetry?.(fr);
263
+ void startUpload(idx, fr.file);
264
+ },
265
+ [files, onRetry, startUpload, uploader],
266
+ );
267
+
268
+ const handleRetryAll = useCallback(() => {
269
+ if (!uploader) return;
270
+ const failedFiles = files.filter((f) => f.status === 'error' && !!f.file);
271
+ if (failedFiles.length === 0) return;
272
+ onRetryAll?.(failedFiles);
273
+ failedFiles.forEach((fr) => {
274
+ const idx = files.indexOf(fr);
275
+ if (idx >= 0 && fr.file) {
276
+ onRetry?.(fr);
277
+ void startUpload(idx, fr.file);
278
+ }
279
+ });
280
+ }, [files, onRetry, onRetryAll, startUpload, uploader]);
281
+
282
+ const onDrop = useCallback(
283
+ (acceptedFiles: File[]) => {
284
+ if (!acceptedFiles?.length) return;
285
+ setFilesAndEmit((prev) => {
286
+ const existing = [...prev];
287
+ const capacity =
288
+ typeof maxFiles === 'number'
289
+ ? Math.max(0, maxFiles - existing.length)
290
+ : acceptedFiles.length;
291
+ const incoming = acceptedFiles
292
+ .slice(0, capacity)
293
+ .map<FileRecord>((f) => ({
294
+ id: undefined,
295
+ url: undefined,
296
+ thumbnailUrl: undefined,
297
+ file: f,
298
+ name: f.name,
299
+ size: f.size,
300
+ type: f.type,
301
+ status: uploader ? 'uploading' : 'idle',
302
+ progress: uploader ? 0 : undefined,
303
+ }));
304
+ const next = multiple
305
+ ? [...existing, ...incoming]
306
+ : ([incoming[0]].filter(Boolean) as FileRecord[]);
307
+ return next;
308
+ });
309
+
310
+ if (uploader) {
311
+ const baseIndex = (isControlled ? value : files)?.length ?? 0;
312
+ const capacity =
313
+ typeof maxFiles === 'number'
314
+ ? Math.max(
315
+ 0,
316
+ maxFiles - ((isControlled ? value : files)?.length ?? 0),
317
+ )
318
+ : acceptedFiles.length;
319
+ acceptedFiles.slice(0, capacity).forEach((f, i) => {
320
+ const index = multiple ? baseIndex + i : 0;
321
+ void startUpload(index, f);
322
+ });
323
+ }
324
+ },
325
+ [
326
+ files,
327
+ isControlled,
328
+ maxFiles,
329
+ multiple,
330
+ startUpload,
331
+ uploader,
332
+ value,
333
+ setFilesAndEmit,
334
+ ],
335
+ );
336
+
337
+ const disabledBecauseFull = !!maxFiles && files.length >= maxFiles;
338
+
339
+ const { getRootProps, getInputProps, isDragActive, isDragReject, open } =
340
+ useDropzone({
341
+ onDrop,
342
+ multiple,
343
+ maxFiles: multiple ? maxFiles : 1,
344
+ accept: accept as Accept | undefined,
345
+ noClick: true,
346
+ noKeyboard: true,
347
+ disabled: disabled || disabledBecauseFull,
348
+ });
349
+
350
+ useEffect(() => {
351
+ const urls = new Set<string>();
352
+ files.forEach((fr) => {
353
+ const preview = getPreviewUrl(fr);
354
+ if (preview) urls.add(preview);
355
+ });
356
+ prevUrlsRef.current.forEach((prev) => {
357
+ if (!urls.has(prev)) URL.revokeObjectURL(prev);
358
+ });
359
+ prevUrlsRef.current = urls;
360
+ }, [files]);
361
+
362
+ const rootClasses = cn(
363
+ 'w-full border border-dashed rounded-md p-4 text-sm transition-colors bg-background',
364
+ 'hover:border-foreground/50',
365
+ isDragActive && 'border-primary',
366
+ isDragReject && 'border-destructive',
367
+ (disabled || disabledBecauseFull) && 'opacity-50 pointer-events-none',
368
+ );
369
+
370
+ // (renderThumb removed; inlined into FileItem for better memoization)
371
+
372
+ // Stable handler refs so children don't see new function identities each render
373
+ const removeRef = useRef(handleRemove);
374
+ useEffect(() => {
375
+ removeRef.current = handleRemove;
376
+ }, [handleRemove]);
377
+ const retryRef = useRef(handleRetry);
378
+ useEffect(() => {
379
+ retryRef.current = handleRetry;
380
+ }, [handleRetry]);
381
+ const onRemoveAt = useCallback((i: number) => {
382
+ removeRef.current(i);
383
+ }, []);
384
+ const onRetryAt = useCallback((i: number) => {
385
+ retryRef.current(i);
386
+ }, []);
387
+
388
+ type FileItemProps = {
389
+ fr: FileRecord;
390
+ idx: number;
391
+ layout: 'grid' | 'list';
392
+ withDownload: boolean;
393
+ uploaderPresent: boolean;
394
+ onRemove: (idx: number) => void;
395
+ onRetry: (idx: number) => void;
396
+ };
397
+
398
+ const FileItem = memo(
399
+ function FileItem({
400
+ fr,
401
+ idx,
402
+ layout,
403
+ withDownload,
404
+ uploaderPresent,
405
+ onRemove,
406
+ onRetry,
407
+ }: FileItemProps) {
408
+ const name = fr.name;
409
+ const size = formatBytes(fr.size);
410
+ const error = fr.status === 'error' ? fr.errorMessage : undefined;
411
+ const preview = getPreviewUrl(fr);
412
+ return (
413
+ <div
414
+ className={cn(
415
+ 'flex items-center gap-3 border rounded-md p-2 bg-card',
416
+ layout === 'grid' ? 'flex-col items-stretch' : 'flex-row',
417
+ )}
418
+ >
419
+ <div className={cn(layout === 'grid' ? 'self-center' : '')}>
420
+ {
421
+ <div
422
+ className={cn(
423
+ 'relative overflow-hidden bg-muted/40 border rounded-md flex items-center justify-center',
424
+ layout === 'grid' ? 'h-28 w-28' : 'h-16 w-16',
425
+ )}
426
+ >
427
+ {preview ? (
428
+ <img
429
+ src={preview}
430
+ alt={fr.name}
431
+ className="object-cover w-full h-full"
432
+ />
433
+ ) : (
434
+ <div className="flex items-center justify-center text-muted-foreground">
435
+ {pickIconByType(fr.type, fr.name)}
436
+ </div>
437
+ )}
438
+ {fr.status === 'uploading' ? (
439
+ <div className="absolute inset-0 bg-black/40 flex items-center justify-center">
440
+ <Loader2 className="h-6 w-6 text-white animate-spin" />
441
+ </div>
442
+ ) : null}
443
+ {fr.status === 'success' ? (
444
+ <div className="absolute top-1 right-1 text-green-500">
445
+ <CheckCircle2 className="h-5 w-5 drop-shadow" />
446
+ </div>
447
+ ) : null}
448
+ {fr.status === 'error' ? (
449
+ <div className="absolute top-1 right-1 text-red-500">
450
+ <XCircle className="h-5 w-5 drop-shadow" />
451
+ </div>
452
+ ) : null}
453
+ </div>
454
+ }
455
+ </div>
456
+ <div
457
+ className={cn('min-w-0 flex-1', layout === 'grid' ? 'mt-2' : '')}
458
+ >
459
+ <div className="flex items-center justify-between gap-2">
460
+ <div className="min-w-0">
461
+ <div className="truncate font-medium" title={name}>
462
+ {name}
463
+ </div>
464
+ <div className="text-xs text-muted-foreground">{size}</div>
465
+ </div>
466
+ <div className="flex items-center gap-1">
467
+ {fr.status === 'error' ? (
468
+ <Tooltip>
469
+ <TooltipTrigger asChild>
470
+ <Button
471
+ size="icon"
472
+ variant="ghost"
473
+ onClick={() => onRetry(idx)}
474
+ aria-label="Retry upload"
475
+ disabled={!uploaderPresent || !fr.file}
476
+ >
477
+ <RotateCcw className="h-4 w-4" />
478
+ </Button>
479
+ </TooltipTrigger>
480
+ <TooltipContent>Retry</TooltipContent>
481
+ </Tooltip>
482
+ ) : null}
483
+ {withDownload && (fr.url || fr.thumbnailUrl) ? (
484
+ <Tooltip>
485
+ <TooltipTrigger asChild>
486
+ <Button
487
+ size="icon"
488
+ variant="ghost"
489
+ onClick={() => {
490
+ const url = fr.url ?? fr.thumbnailUrl;
491
+ if (url) {
492
+ window.open(url, '_blank', 'noopener,noreferrer');
493
+ }
494
+ }}
495
+ aria-label="Download"
496
+ >
497
+ <Download className="h-4 w-4" />
498
+ </Button>
499
+ </TooltipTrigger>
500
+ <TooltipContent>Download</TooltipContent>
501
+ </Tooltip>
502
+ ) : null}
503
+ <Tooltip>
504
+ <TooltipTrigger asChild>
505
+ <Button
506
+ size="icon"
507
+ variant="ghost"
508
+ onClick={() => onRemove(idx)}
509
+ aria-label="Remove"
510
+ >
511
+ <Trash2 className="h-4 w-4" />
512
+ </Button>
513
+ </TooltipTrigger>
514
+ <TooltipContent>Remove</TooltipContent>
515
+ </Tooltip>
516
+ </div>
517
+ </div>
518
+ {fr.status === 'uploading' ? (
519
+ <div className="mt-2">
520
+ <Progress value={fr.progress ?? 0} />
521
+ </div>
522
+ ) : null}
523
+ {error ? (
524
+ <div className="mt-2 text-xs text-destructive">{error}</div>
525
+ ) : null}
526
+ </div>
527
+ </div>
528
+ );
529
+ },
530
+ (prev, next) =>
531
+ prev.fr === next.fr &&
532
+ prev.layout === next.layout &&
533
+ prev.withDownload === next.withDownload &&
534
+ prev.uploaderPresent === next.uploaderPresent &&
535
+ prev.idx === next.idx,
536
+ );
537
+
538
+ return (
539
+ <div className={cn('space-y-3', className)}>
540
+ <div {...getRootProps({ className: rootClasses })}>
541
+ <input
542
+ {...getInputProps({
543
+ onClick: (e) => {
544
+ (e.target as HTMLInputElement).value = '';
545
+ },
546
+ })}
547
+ />
548
+ <div className="flex items-center justify-between gap-3">
549
+ <div className="flex items-center gap-2 text-muted-foreground">
550
+ <Button
551
+ size="sm"
552
+ variant="secondary"
553
+ disabled={disabled || disabledBecauseFull}
554
+ onClick={(e) => {
555
+ e.preventDefault();
556
+ e.stopPropagation();
557
+ open();
558
+ }}
559
+ >
560
+ Select files
561
+ </Button>
562
+ <UploadCloud className="h-5 w-5" />
563
+ <div>
564
+ <div className="font-medium">
565
+ {disabledBecauseFull
566
+ ? 'File limit reached'
567
+ : isDragActive
568
+ ? 'Drop the files here'
569
+ : placeholder}
570
+ </div>
571
+ <div className="text-xs">
572
+ {accept ? 'Specific file types only' : 'Any file type'}
573
+ {typeof maxFiles === 'number'
574
+ ? ` • Up to ${maxFiles} file${maxFiles > 1 ? 's' : ''}`
575
+ : ''}
576
+ </div>
577
+ </div>
578
+ </div>
579
+ {withRetry && (
580
+ <div className="flex items-center gap-2">
581
+ <Button
582
+ size="sm"
583
+ variant="ghost"
584
+ onClick={handleRetryAll}
585
+ disabled={
586
+ !!disabled ||
587
+ !uploader ||
588
+ files.every((f) => f.status !== 'error' || !f.file)
589
+ }
590
+ >
591
+ <RotateCcw className="mr-1 h-4 w-4" /> Retry failed
592
+ </Button>
593
+ </div>
594
+ )}
595
+ </div>
596
+ </div>
597
+
598
+ <TooltipProvider>
599
+ <div
600
+ className={cn(
601
+ layout === 'grid' ? 'flex flex-wrap gap-3' : 'flex flex-col gap-2',
602
+ )}
603
+ >
604
+ {files.length === 0 ? (
605
+ <div className="text-sm text-muted-foreground">No files</div>
606
+ ) : (
607
+ files.map((fr, i) => (
608
+ <FileItem
609
+ key={`${fr.name}-${i}`}
610
+ fr={fr}
611
+ idx={i}
612
+ layout={layout}
613
+ withDownload={withDownload}
614
+ uploaderPresent={!!uploader}
615
+ onRemove={onRemoveAt}
616
+ onRetry={onRetryAt}
617
+ />
618
+ ))
619
+ )}
620
+ </div>
621
+ </TooltipProvider>
622
+ </div>
623
+ );
624
+ }
625
+
626
+ export default FileUploader;