@firecms/core 3.0.1 → 3.1.0-canary.02232f4

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 (334) hide show
  1. package/README.md +1 -1
  2. package/dist/components/AIIcon.d.ts +16 -0
  3. package/dist/components/EntityCollectionTable/EntityCollectionRowActions.d.ts +7 -1
  4. package/dist/components/EntityCollectionTable/EntityCollectionTable.d.ts +1 -1
  5. package/dist/components/EntityCollectionTable/EntityCollectionTableProps.d.ts +14 -0
  6. package/dist/components/EntityCollectionTable/PropertyTableCell.d.ts +6 -0
  7. package/dist/components/EntityCollectionTable/internal/CollectionTableToolbar.d.ts +5 -4
  8. package/dist/components/EntityCollectionTable/internal/EntityTableCell.d.ts +6 -0
  9. package/dist/components/EntityCollectionTable/internal/popup_field/useDraggable.d.ts +2 -2
  10. package/dist/components/EntityCollectionView/Board.d.ts +2 -0
  11. package/dist/components/EntityCollectionView/BoardColumn.d.ts +42 -0
  12. package/dist/components/EntityCollectionView/BoardColumnTitle.d.ts +9 -0
  13. package/dist/components/EntityCollectionView/BoardSortableList.d.ts +14 -0
  14. package/dist/components/EntityCollectionView/CollectionDataErrorBanner.d.ts +4 -0
  15. package/dist/components/EntityCollectionView/EntityBoardCard.d.ts +26 -0
  16. package/dist/components/EntityCollectionView/EntityCard.d.ts +19 -0
  17. package/dist/components/EntityCollectionView/EntityCollectionBoardView.d.ts +20 -0
  18. package/dist/components/EntityCollectionView/EntityCollectionCardView.d.ts +31 -0
  19. package/dist/components/EntityCollectionView/EntityCollectionViewActions.d.ts +2 -2
  20. package/dist/components/EntityCollectionView/EntityCollectionViewStartActions.d.ts +7 -3
  21. package/dist/components/EntityCollectionView/FiltersDialog.d.ts +14 -0
  22. package/dist/components/EntityCollectionView/ViewModeToggle.d.ts +44 -0
  23. package/dist/components/EntityCollectionView/board_types.d.ts +105 -0
  24. package/dist/components/EntityCollectionView/useBoardDataController.d.ts +60 -0
  25. package/dist/components/ErrorBoundary.d.ts +4 -2
  26. package/dist/components/HomePage/DefaultHomePage.d.ts +0 -1
  27. package/dist/components/LanguageToggle.d.ts +1 -0
  28. package/dist/components/SelectableTable/SelectableTable.d.ts +5 -1
  29. package/dist/components/SelectableTable/filters/DateTimeFilterField.d.ts +2 -1
  30. package/dist/components/UnsavedChangesDialog.d.ts +1 -0
  31. package/dist/components/VirtualTable/VirtualTableCell.d.ts +6 -0
  32. package/dist/components/VirtualTable/VirtualTableHeader.d.ts +4 -1
  33. package/dist/components/VirtualTable/VirtualTableHeaderRow.d.ts +1 -1
  34. package/dist/components/VirtualTable/VirtualTableProps.d.ts +17 -1
  35. package/dist/components/VirtualTable/fields/VirtualTableDateField.d.ts +1 -0
  36. package/dist/components/VirtualTable/types.d.ts +3 -0
  37. package/dist/components/index.d.ts +4 -0
  38. package/dist/contexts/index.d.ts +10 -0
  39. package/dist/core/DrawerNavigationGroup.d.ts +45 -0
  40. package/dist/core/index.d.ts +1 -0
  41. package/dist/editor/components/SlashCommandMenu.d.ts +6 -0
  42. package/dist/editor/components/editor-bubble-item.d.ts +8 -0
  43. package/dist/editor/components/editor-bubble.d.ts +8 -0
  44. package/dist/editor/components/image-bubble.d.ts +5 -0
  45. package/dist/editor/components/index.d.ts +16 -0
  46. package/dist/editor/components/table-bubble.d.ts +5 -0
  47. package/dist/editor/editor.d.ts +30 -0
  48. package/dist/editor/extensions/HighlightDecorationExtension.d.ts +24 -0
  49. package/dist/editor/extensions/Image/index.d.ts +6 -0
  50. package/dist/editor/extensions/Image.d.ts +6 -0
  51. package/dist/editor/extensions/TextLoadingDecorationExtension.d.ts +16 -0
  52. package/dist/editor/extensions/clipboard.d.ts +7 -0
  53. package/dist/editor/extensions/custom-keymap.d.ts +1 -0
  54. package/dist/editor/extensions/drag-and-drop.d.ts +9 -0
  55. package/dist/editor/hooks/useProseMirror.d.ts +13 -0
  56. package/dist/editor/hooks/useProseMirrorContext.d.ts +9 -0
  57. package/dist/editor/index.d.ts +2 -0
  58. package/dist/editor/markdown.d.ts +5 -0
  59. package/dist/editor/nodeViews/ImageComponent.d.ts +3 -0
  60. package/dist/editor/nodeViews/ReactNodeView.d.ts +29 -0
  61. package/dist/editor/nodeViews/TaskItemComponent.d.ts +3 -0
  62. package/dist/editor/nodeViews/index.d.ts +6 -0
  63. package/dist/editor/plugins/index.d.ts +2 -0
  64. package/dist/editor/plugins/inputrules.d.ts +6 -0
  65. package/dist/editor/plugins/placeholderPlugin.d.ts +3 -0
  66. package/dist/editor/plugins/slashCommandPlugin.d.ts +12 -0
  67. package/dist/editor/schema.d.ts +2 -0
  68. package/dist/editor/selectors/ai-selector.d.ts +0 -0
  69. package/dist/editor/selectors/color-selector.d.ts +10 -0
  70. package/dist/editor/selectors/link-selector.d.ts +8 -0
  71. package/dist/editor/selectors/node-selector.d.ts +15 -0
  72. package/dist/editor/selectors/text-buttons.d.ts +1 -0
  73. package/dist/editor/types.d.ts +5 -0
  74. package/dist/editor/useProseMirror.d.ts +16 -0
  75. package/dist/editor/utils/prosemirror-utils.d.ts +6 -0
  76. package/dist/editor/utils/remove_classes.d.ts +1 -0
  77. package/dist/editor/utils/useDebouncedCallback.d.ts +1 -0
  78. package/dist/form/components/ErrorFocus.d.ts +1 -1
  79. package/dist/form/field_bindings/MapFieldBinding.d.ts +1 -1
  80. package/dist/form/field_bindings/MarkdownEditorFieldBinding.d.ts +1 -1
  81. package/dist/form/validation.d.ts +3 -2
  82. package/dist/hooks/index.d.ts +1 -0
  83. package/dist/hooks/useBreadcrumbsController.d.ts +16 -0
  84. package/dist/hooks/useBuildNavigationController.d.ts +0 -1
  85. package/dist/hooks/useCollapsedGroups.d.ts +6 -3
  86. package/dist/hooks/useTranslation.d.ts +17 -0
  87. package/dist/i18n/FireCMSi18nProvider.d.ts +33 -0
  88. package/dist/index.d.ts +5 -0
  89. package/dist/index.es.js +30146 -15178
  90. package/dist/index.es.js.map +1 -1
  91. package/dist/index.umd.js +30032 -15085
  92. package/dist/index.umd.js.map +1 -1
  93. package/dist/internal/useRestoreScroll.d.ts +1 -1
  94. package/dist/locales/de.d.ts +2 -0
  95. package/dist/locales/en.d.ts +10 -0
  96. package/dist/locales/es.d.ts +10 -0
  97. package/dist/locales/fr.d.ts +2 -0
  98. package/dist/locales/hi.d.ts +2 -0
  99. package/dist/locales/it.d.ts +2 -0
  100. package/dist/locales/pt.d.ts +7 -0
  101. package/dist/preview/PropertyPreviewProps.d.ts +5 -0
  102. package/dist/preview/components/DatePreview.d.ts +13 -3
  103. package/dist/preview/components/ImagePreview.d.ts +5 -1
  104. package/dist/preview/components/StorageThumbnail.d.ts +2 -1
  105. package/dist/preview/components/UrlComponentPreview.d.ts +2 -1
  106. package/dist/preview/property_previews/ArrayOfStorageComponentsPreview.d.ts +1 -1
  107. package/dist/preview/property_previews/ArrayOfStringsPreview.d.ts +1 -1
  108. package/dist/preview/property_previews/SkeletonPropertyComponent.d.ts +1 -1
  109. package/dist/types/analytics.d.ts +1 -1
  110. package/dist/types/collections.d.ts +88 -2
  111. package/dist/types/customization_controller.d.ts +2 -1
  112. package/dist/types/datasource.d.ts +0 -1
  113. package/dist/types/firecms.d.ts +2 -1
  114. package/dist/types/index.d.ts +1 -0
  115. package/dist/types/navigation.d.ts +2 -2
  116. package/dist/types/plugins.d.ts +69 -1
  117. package/dist/types/properties.d.ts +268 -12
  118. package/dist/types/storage.d.ts +1 -0
  119. package/dist/types/translations.d.ts +669 -0
  120. package/dist/util/__tests__/conditions.test.d.ts +1 -0
  121. package/dist/util/__tests__/objects.test.d.ts +1 -0
  122. package/dist/util/conditions.d.ts +26 -0
  123. package/dist/util/entities.d.ts +2 -3
  124. package/dist/util/index.d.ts +3 -1
  125. package/dist/util/lazy_eager.d.ts +7 -0
  126. package/dist/util/objects.d.ts +1 -0
  127. package/dist/util/property_utils.d.ts +2 -1
  128. package/dist/util/resolutions.d.ts +3 -3
  129. package/dist/util/useStorageUploadController.d.ts +10 -1
  130. package/package.json +51 -12
  131. package/src/app/Scaffold.tsx +20 -19
  132. package/src/components/AIIcon.tsx +41 -0
  133. package/src/components/ArrayContainer.tsx +7 -8
  134. package/src/components/ClearFilterSortButton.tsx +25 -19
  135. package/src/components/ConfirmationDialog.tsx +4 -4
  136. package/src/components/DeleteEntityDialog.tsx +12 -11
  137. package/src/components/EntityCollectionTable/EntityCollectionRowActions.tsx +82 -43
  138. package/src/components/EntityCollectionTable/EntityCollectionTable.tsx +130 -79
  139. package/src/components/EntityCollectionTable/EntityCollectionTableProps.tsx +121 -104
  140. package/src/components/EntityCollectionTable/PropertyTableCell.tsx +132 -103
  141. package/src/components/EntityCollectionTable/fields/TableReferenceField.tsx +6 -3
  142. package/src/components/EntityCollectionTable/internal/CollectionTableToolbar.tsx +24 -44
  143. package/src/components/EntityCollectionTable/internal/EntityTableCell.tsx +90 -49
  144. package/src/components/EntityCollectionTable/internal/EntityTableCellActions.tsx +1 -1
  145. package/src/components/EntityCollectionTable/internal/popup_field/PopupFormField.tsx +3 -2
  146. package/src/components/EntityCollectionTable/internal/popup_field/useDraggable.tsx +11 -11
  147. package/src/components/EntityCollectionView/Board.tsx +324 -0
  148. package/src/components/EntityCollectionView/BoardColumn.tsx +158 -0
  149. package/src/components/EntityCollectionView/BoardColumnTitle.tsx +45 -0
  150. package/src/components/EntityCollectionView/BoardSortableList.tsx +174 -0
  151. package/src/components/EntityCollectionView/CollectionDataErrorBanner.tsx +43 -0
  152. package/src/components/EntityCollectionView/EntityBoardCard.tsx +212 -0
  153. package/src/components/EntityCollectionView/EntityCard.tsx +235 -0
  154. package/src/components/EntityCollectionView/EntityCollectionBoardView.tsx +706 -0
  155. package/src/components/EntityCollectionView/EntityCollectionCardView.tsx +236 -0
  156. package/src/components/EntityCollectionView/EntityCollectionView.tsx +531 -209
  157. package/src/components/EntityCollectionView/EntityCollectionViewActions.tsx +35 -22
  158. package/src/components/EntityCollectionView/EntityCollectionViewStartActions.tsx +86 -15
  159. package/src/components/EntityCollectionView/FiltersDialog.tsx +252 -0
  160. package/src/components/EntityCollectionView/ViewModeToggle.tsx +202 -0
  161. package/src/components/EntityCollectionView/board_types.ts +113 -0
  162. package/src/components/EntityCollectionView/useBoardDataController.tsx +490 -0
  163. package/src/components/EntityJsonPreview.tsx +2 -1
  164. package/src/components/EntityView.tsx +3 -2
  165. package/src/components/ErrorBoundary.tsx +27 -15
  166. package/src/components/ErrorTooltip.tsx +2 -1
  167. package/src/components/HomePage/DefaultHomePage.tsx +65 -22
  168. package/src/components/HomePage/HomePageDnD.tsx +59 -42
  169. package/src/components/HomePage/NavigationCard.tsx +20 -18
  170. package/src/components/HomePage/NavigationGroup.tsx +20 -17
  171. package/src/components/HomePage/RenameGroupDialog.tsx +15 -15
  172. package/src/components/HomePage/SmallNavigationCard.tsx +10 -9
  173. package/src/components/LanguageToggle.tsx +66 -0
  174. package/src/components/NotFoundPage.tsx +5 -3
  175. package/src/components/ReferenceTable/ReferenceSelectionTable.tsx +12 -17
  176. package/src/components/ReferenceWidget.tsx +5 -6
  177. package/src/components/SearchIconsView.tsx +3 -1
  178. package/src/components/SelectableTable/SelectableTable.tsx +75 -67
  179. package/src/components/SelectableTable/filters/BooleanFilterField.tsx +7 -6
  180. package/src/components/SelectableTable/filters/DateTimeFilterField.tsx +50 -40
  181. package/src/components/SelectableTable/filters/ReferenceFilterField.tsx +53 -40
  182. package/src/components/SelectableTable/filters/StringNumberFilterField.tsx +60 -58
  183. package/src/components/UnsavedChangesDialog.tsx +6 -6
  184. package/src/components/UserDisplay.tsx +4 -4
  185. package/src/components/VirtualTable/VirtualTable.performance.test.tsx +1 -0
  186. package/src/components/VirtualTable/VirtualTable.tsx +275 -119
  187. package/src/components/VirtualTable/VirtualTableCell.tsx +18 -2
  188. package/src/components/VirtualTable/VirtualTableHeader.tsx +76 -64
  189. package/src/components/VirtualTable/VirtualTableHeaderRow.tsx +163 -42
  190. package/src/components/VirtualTable/VirtualTableProps.tsx +21 -2
  191. package/src/components/VirtualTable/VirtualTableRow.tsx +1 -1
  192. package/src/components/VirtualTable/fields/VirtualTableDateField.tsx +3 -0
  193. package/src/components/VirtualTable/fields/VirtualTableSelect.tsx +19 -6
  194. package/src/components/VirtualTable/types.tsx +3 -0
  195. package/src/components/common/default_entity_actions.tsx +4 -0
  196. package/src/components/common/useColumnsIds.tsx +95 -3
  197. package/src/components/common/useDataSourceTableController.tsx +12 -4
  198. package/src/components/index.tsx +5 -0
  199. package/src/contexts/BreacrumbsContext.tsx +15 -8
  200. package/src/contexts/index.ts +10 -0
  201. package/src/core/DefaultAppBar.tsx +49 -32
  202. package/src/core/DefaultDrawer.tsx +49 -57
  203. package/src/core/DrawerNavigationGroup.tsx +120 -0
  204. package/src/core/DrawerNavigationItem.tsx +4 -3
  205. package/src/core/EntityEditView.tsx +94 -50
  206. package/src/core/EntityEditViewFormActions.tsx +24 -17
  207. package/src/core/EntitySidePanel.tsx +34 -30
  208. package/src/core/FireCMS.tsx +33 -6
  209. package/src/core/SideDialogs.tsx +4 -2
  210. package/src/core/field_configs.tsx +18 -11
  211. package/src/core/index.tsx +1 -0
  212. package/src/editor/components/SlashCommandMenu.tsx +516 -0
  213. package/src/editor/components/editor-bubble-item.tsx +32 -0
  214. package/src/editor/components/editor-bubble.tsx +118 -0
  215. package/src/editor/components/image-bubble.tsx +156 -0
  216. package/src/editor/components/index.ts +14 -0
  217. package/src/editor/components/table-bubble.tsx +165 -0
  218. package/src/editor/editor.tsx +455 -0
  219. package/src/editor/extensions/HighlightDecorationExtension.ts +114 -0
  220. package/src/editor/extensions/Image/index.ts +133 -0
  221. package/src/editor/extensions/Image.ts +159 -0
  222. package/src/editor/extensions/TextLoadingDecorationExtension.tsx +107 -0
  223. package/src/editor/extensions/clipboard.ts +72 -0
  224. package/src/editor/extensions/custom-keymap.ts +24 -0
  225. package/src/editor/extensions/drag-and-drop.tsx +480 -0
  226. package/src/editor/hooks/useProseMirror.ts +124 -0
  227. package/src/editor/hooks/useProseMirrorContext.ts +15 -0
  228. package/src/editor/index.ts +2 -0
  229. package/src/editor/markdown.ts +172 -0
  230. package/src/editor/nodeViews/ImageComponent.tsx +20 -0
  231. package/src/editor/nodeViews/ReactNodeView.tsx +89 -0
  232. package/src/editor/nodeViews/TaskItemComponent.tsx +29 -0
  233. package/src/editor/nodeViews/index.ts +35 -0
  234. package/src/editor/plugins/index.ts +58 -0
  235. package/src/editor/plugins/inputrules.ts +82 -0
  236. package/src/editor/plugins/placeholderPlugin.ts +55 -0
  237. package/src/editor/plugins/slashCommandPlugin.ts +61 -0
  238. package/src/editor/schema.ts +240 -0
  239. package/src/editor/selectors/ai-selector.tsx +111 -0
  240. package/src/editor/selectors/color-selector.tsx +200 -0
  241. package/src/editor/selectors/link-selector.tsx +118 -0
  242. package/src/editor/selectors/node-selector.tsx +157 -0
  243. package/src/editor/selectors/text-buttons.tsx +86 -0
  244. package/src/editor/types.ts +6 -0
  245. package/src/editor/useProseMirror.ts +126 -0
  246. package/src/editor/utils/prosemirror-utils.ts +108 -0
  247. package/src/editor/utils/remove_classes.ts +17 -0
  248. package/src/editor/utils/useDebouncedCallback.ts +25 -0
  249. package/src/form/EntityForm.tsx +149 -67
  250. package/src/form/EntityFormActions.tsx +19 -12
  251. package/src/form/PropertyFieldBinding.tsx +68 -51
  252. package/src/form/components/ErrorFocus.tsx +3 -3
  253. package/src/form/components/LocalChangesMenu.tsx +13 -13
  254. package/src/form/components/StorageItemPreview.tsx +5 -3
  255. package/src/form/components/StorageUploadProgress.tsx +18 -3
  256. package/src/form/field_bindings/ArrayCustomShapedFieldBinding.tsx +18 -5
  257. package/src/form/field_bindings/ArrayOfReferencesFieldBinding.tsx +22 -10
  258. package/src/form/field_bindings/BlockFieldBinding.tsx +26 -9
  259. package/src/form/field_bindings/DateTimeFieldBinding.tsx +18 -17
  260. package/src/form/field_bindings/KeyValueFieldBinding.tsx +46 -25
  261. package/src/form/field_bindings/MapFieldBinding.tsx +88 -70
  262. package/src/form/field_bindings/MarkdownEditorFieldBinding.tsx +93 -52
  263. package/src/form/field_bindings/MultiSelectFieldBinding.tsx +15 -1
  264. package/src/form/field_bindings/ReferenceAsStringFieldBinding.tsx +25 -11
  265. package/src/form/field_bindings/ReferenceFieldBinding.tsx +25 -11
  266. package/src/form/field_bindings/RepeatFieldBinding.tsx +21 -6
  267. package/src/form/field_bindings/SelectFieldBinding.tsx +7 -5
  268. package/src/form/field_bindings/StorageUploadFieldBinding.tsx +110 -92
  269. package/src/form/field_bindings/SwitchFieldBinding.tsx +31 -14
  270. package/src/form/field_bindings/TextFieldBinding.tsx +77 -38
  271. package/src/form/field_bindings/UserSelectFieldBinding.tsx +7 -5
  272. package/src/form/validation.ts +245 -160
  273. package/src/hooks/index.tsx +1 -0
  274. package/src/hooks/useBreadcrumbsController.tsx +18 -0
  275. package/src/hooks/useBuildNavigationController.tsx +91 -41
  276. package/src/hooks/useCollapsedGroups.ts +18 -9
  277. package/src/hooks/useTranslation.ts +31 -0
  278. package/src/hooks/useValidateAuthenticator.tsx +1 -1
  279. package/src/i18n/FireCMSi18nProvider.tsx +160 -0
  280. package/src/index.ts +5 -0
  281. package/src/internal/useBuildDataSource.ts +68 -34
  282. package/src/internal/useBuildSideDialogsController.tsx +11 -8
  283. package/src/internal/useBuildSideEntityController.tsx +24 -24
  284. package/src/internal/useRestoreScroll.tsx +26 -14
  285. package/src/locales/de.ts +718 -0
  286. package/src/locales/en.ts +730 -0
  287. package/src/locales/es.ts +730 -0
  288. package/src/locales/fr.ts +718 -0
  289. package/src/locales/hi.ts +718 -0
  290. package/src/locales/it.ts +718 -0
  291. package/src/locales/pt.ts +727 -0
  292. package/src/preview/PropertyPreview.tsx +43 -33
  293. package/src/preview/PropertyPreviewProps.tsx +6 -0
  294. package/src/preview/components/DatePreview.tsx +72 -4
  295. package/src/preview/components/EmptyValue.tsx +1 -1
  296. package/src/preview/components/ImagePreview.tsx +37 -21
  297. package/src/preview/components/ReferencePreview.tsx +2 -1
  298. package/src/preview/components/StorageThumbnail.tsx +16 -12
  299. package/src/preview/components/UrlComponentPreview.tsx +32 -27
  300. package/src/preview/components/UserPreview.tsx +3 -1
  301. package/src/preview/property_previews/ArrayOfStorageComponentsPreview.tsx +9 -7
  302. package/src/preview/property_previews/ArrayOfStringsPreview.tsx +11 -9
  303. package/src/preview/property_previews/ArrayPropertyPreview.tsx +26 -24
  304. package/src/preview/property_previews/MapPropertyPreview.tsx +49 -27
  305. package/src/preview/property_previews/SkeletonPropertyComponent.tsx +61 -56
  306. package/src/routes/CustomCMSRoute.tsx +1 -0
  307. package/src/routes/FireCMSRoute.tsx +87 -65
  308. package/src/types/analytics.ts +10 -0
  309. package/src/types/collections.ts +97 -3
  310. package/src/types/customization_controller.tsx +2 -1
  311. package/src/types/datasource.ts +54 -56
  312. package/src/types/firecms.tsx +2 -1
  313. package/src/types/index.ts +1 -0
  314. package/src/types/navigation.ts +2 -2
  315. package/src/types/plugins.tsx +77 -1
  316. package/src/types/properties.ts +359 -37
  317. package/src/types/storage.ts +2 -1
  318. package/src/types/translations.ts +752 -0
  319. package/src/util/__tests__/conditions.test.ts +506 -0
  320. package/src/util/__tests__/objects.test.ts +196 -0
  321. package/src/util/callbacks.ts +6 -3
  322. package/src/util/collections.ts +51 -6
  323. package/src/util/conditions.ts +339 -0
  324. package/src/util/entities.ts +29 -30
  325. package/src/util/entity_cache.ts +2 -1
  326. package/src/util/index.ts +3 -1
  327. package/src/util/join_collections.ts +10 -8
  328. package/src/util/lazy_eager.tsx +33 -0
  329. package/src/util/objects.ts +46 -13
  330. package/src/util/{references.ts → previews.ts} +16 -2
  331. package/src/util/property_utils.tsx +37 -11
  332. package/src/util/resolutions.ts +62 -58
  333. package/src/util/useStorageUploadController.tsx +23 -29
  334. /package/dist/util/{references.d.ts → previews.d.ts} +0 -0
@@ -4,8 +4,10 @@ import { FieldProps, MapProperty, Properties, PropertyFieldBindingProps, Resolve
4
4
  import { ErrorBoundary } from "../../components";
5
5
  import { getIconForProperty, isHidden, isReadOnly, pick } from "../../util";
6
6
  import { FieldHelperText, LabelWithIconAndTooltip } from "../components";
7
+ import { FormEntry } from "../components/FormEntry";
7
8
  import { PropertyFieldBinding } from "../PropertyFieldBinding";
8
- import { cls, ExpandablePanel, InputLabel, Select, SelectItem } from "@firecms/ui";
9
+ import { cls, ExpandablePanel, InputLabel, Select, SelectItem, IconButton, CloseIcon } from "@firecms/ui";
10
+ import { useTranslation } from "../../hooks";
9
11
 
10
12
  /**
11
13
  * Field that renders the children property fields
@@ -15,19 +17,20 @@ import { cls, ExpandablePanel, InputLabel, Select, SelectItem } from "@firecms/u
15
17
  * @group Form fields
16
18
  */
17
19
  export function MapFieldBinding({
18
- propertyKey,
19
- value,
20
- showError,
21
- error,
22
- disabled,
23
- property,
24
- partOfArray,
25
- minimalistView: minimalistViewProp,
26
- includeDescription,
27
- autoFocus,
28
- context,
29
- onPropertyChange
30
- }: FieldProps<Record<string, any>>) {
20
+ propertyKey,
21
+ value,
22
+ showError,
23
+ error,
24
+ disabled,
25
+ property,
26
+ partOfArray,
27
+ minimalistView: minimalistViewProp,
28
+ includeDescription,
29
+ autoFocus,
30
+ context,
31
+ onPropertyChange,
32
+ setValue
33
+ }: FieldProps<Record<string, any>>) {
31
34
 
32
35
  const pickOnlySomeKeys = property.pickOnlySomeKeys || false;
33
36
  const expanded = property.expanded === undefined ? true : property.expanded;
@@ -52,81 +55,96 @@ export function MapFieldBinding({
52
55
  }
53
56
 
54
57
  const mapFormView = <>
55
- <div
56
- className={cls("py-1 flex flex-col space-y-2", minimalistView && property.widthPercentage !== undefined ? "mt-8" : undefined)}>
57
- {Object.entries(mapProperties)
58
- .filter(([_, property]) => !isHidden(property))
59
- .map(([entryKey, childProperty], index) => {
60
- const thisDisabled = isReadOnly(childProperty) || Boolean(childProperty.disabled);
61
- const fieldBindingProps: PropertyFieldBindingProps<any> = {
62
- propertyKey: `${propertyKey}.${entryKey}`,
63
- disabled: disabled || thisDisabled,
64
- property: childProperty,
65
- includeDescription,
66
- context,
67
- partOfArray: false,
68
- minimalistView: false,
69
- autoFocus: autoFocus && index === 0,
70
- onPropertyChange: function (updatedProperty) {
71
- onPropertyChange?.({
72
- properties: {
73
- [entryKey]: updatedProperty
74
- }
75
- } as Partial<MapProperty>);
58
+ <div
59
+ className={cls("py-1 flex flex-wrap gap-x-4 w-full space-y-2", minimalistView && property.widthPercentage !== undefined ? "mt-8" : undefined)}>
60
+ {Object.entries(mapProperties)
61
+ .filter(([_, property]) => !isHidden(property))
62
+ .map(([entryKey, childProperty], index) => {
63
+ const thisDisabled = isReadOnly(childProperty) || Boolean(childProperty.disabled);
64
+ const fieldBindingProps: PropertyFieldBindingProps<any> = {
65
+ propertyKey: `${propertyKey}.${entryKey}`,
66
+ disabled: disabled || thisDisabled,
67
+ property: childProperty,
68
+ includeDescription,
69
+ context,
70
+ partOfArray: false,
71
+ minimalistView: false,
72
+ autoFocus: autoFocus && index === 0,
73
+ onPropertyChange: function (updatedProperty) {
74
+ onPropertyChange?.({
75
+ properties: {
76
+ [entryKey]: updatedProperty
76
77
  }
77
- };
78
-
79
- return (
80
- <div key={`map-${propertyKey}-${index}`} className={"relative"}>
81
- <ErrorBoundary>
82
- <PropertyFieldBinding
83
- {...fieldBindingProps}/>
84
- </ErrorBoundary>
85
- </div>
86
- ) ;
78
+ } as Partial<MapProperty>);
87
79
  }
88
- )
80
+ };
81
+
82
+ const widthPercentage = childProperty.widthPercentage ?? 100;
83
+ return (
84
+ <FormEntry propertyKey={`${propertyKey}.${entryKey}`}
85
+ widthPercentage={widthPercentage}
86
+ key={`map-${propertyKey}-${index}`}>
87
+ <PropertyFieldBinding
88
+ {...fieldBindingProps} />
89
+ </FormEntry>
90
+ );
89
91
  }
90
- </div>
92
+ )
93
+ }
94
+ </div>
91
95
 
92
- {/*{pickOnlySomeKeys && buildPickKeysSelect(disabled, property.properties, setValue, value)}*/}
96
+ {/*{pickOnlySomeKeys && buildPickKeysSelect(disabled, property.properties, setValue, value, t)}*/}
93
97
 
94
- </>
95
- ;
98
+ </>
99
+ ;
96
100
 
97
101
  return (
98
102
  <ErrorBoundary>
99
103
 
100
104
  {!minimalistView && <ExpandablePanel initiallyExpanded={expanded}
101
- onExpandedChange={(expanded) => {
102
- onPropertyChange?.({
103
- expanded
104
- });
105
- }}
106
- className={property.widthPercentage !== undefined ? "mt-8" : undefined}
107
- innerClassName={"px-2 md:px-4 pb-2 md:pb-4 pt-1 md:pt-2 bg-white dark:bg-surface-900"}
108
- title={<LabelWithIconAndTooltip
109
- propertyKey={propertyKey}
110
- icon={getIconForProperty(property, "small")}
111
- required={property.validation?.required}
112
- title={property.name}
113
- className={"text-text-secondary dark:text-text-secondary-dark"}/>}>
105
+ onExpandedChange={(expanded) => {
106
+ onPropertyChange?.({
107
+ expanded
108
+ });
109
+ }}
110
+ className={property.widthPercentage !== undefined ? "mt-8" : undefined}
111
+ innerClassName={"px-2 md:px-4 pb-2 md:pb-4 pt-1 md:pt-2 bg-white dark:bg-surface-900"}
112
+ title={<div className="flex items-center w-full">
113
+ <LabelWithIconAndTooltip
114
+ propertyKey={propertyKey}
115
+ icon={getIconForProperty(property, "small")}
116
+ required={property.validation?.required}
117
+ title={property.name}
118
+ className={"text-text-secondary dark:text-text-secondary-dark flex-grow"} />
119
+ {(property.nullable || property.clearable) && !disabled && (
120
+ <IconButton
121
+ size="small"
122
+ onClick={(e) => {
123
+ e.stopPropagation();
124
+ e.preventDefault();
125
+ setValue(null);
126
+ }}
127
+ >
128
+ <CloseIcon size={"small"}/>
129
+ </IconButton>
130
+ )}
131
+ </div>}>
114
132
  {mapFormView}
115
133
  </ExpandablePanel>}
116
134
 
117
135
  {minimalistView && mapFormView}
118
136
 
119
137
  <FieldHelperText includeDescription={includeDescription}
120
- showError={showError ?? false}
121
- error={error && !partOfArray ? (typeof error === "string" ? error : "A property of this map has an error") : undefined}
122
- disabled={disabled}
123
- property={property}/>
138
+ showError={showError ?? false}
139
+ error={error && !partOfArray ? (typeof error === "string" ? error : "A property of this map has an error") : undefined}
140
+ disabled={disabled}
141
+ property={property} />
124
142
 
125
143
  </ErrorBoundary>
126
144
  );
127
145
  }
128
146
 
129
- const buildPickKeysSelect = (disabled: boolean, properties: Properties, setValue: (value: any) => void, value: any) => {
147
+ const buildPickKeysSelect = (disabled: boolean, properties: Properties, setValue: (value: any) => void, value: any, t: any) => {
130
148
 
131
149
  const keys = Object.keys(properties)
132
150
  .filter((key) => !value || !(key in value));
@@ -141,7 +159,7 @@ const buildPickKeysSelect = (disabled: boolean, properties: Properties, setValue
141
159
  if (!keys.length) return <></>;
142
160
 
143
161
  return <div className={"m-4"}>
144
- <InputLabel>Add property</InputLabel>
162
+ <InputLabel>{t("add_property")}</InputLabel>
145
163
  <Select
146
164
  value={""}
147
165
  size={"large"}
@@ -150,7 +168,7 @@ const buildPickKeysSelect = (disabled: boolean, properties: Properties, setValue
150
168
  onValueChange={handleAddProperty}
151
169
  renderValue={(key) => (properties as Properties)[key].name || key}>
152
170
  {keys.map((key) => <SelectItem key={key}
153
- value={key}>{(properties as Properties)[key].name || key}</SelectItem>)}
171
+ value={key}>{(properties as Properties)[key].name || key}</SelectItem>)}
154
172
  </Select>
155
173
  </div>;
156
174
  };
@@ -11,9 +11,14 @@ import {
11
11
  useAuthController,
12
12
  useStorageSource
13
13
  } from "../../index";
14
- import { cls, fieldBackgroundDisabledMixin, fieldBackgroundHoverMixin, fieldBackgroundMixin } from "@firecms/ui";
15
- import { FireCMSEditor, FireCMSEditorProps } from "@firecms/editor";
14
+ import { cls, fieldBackgroundDisabledMixin, fieldBackgroundHoverMixin, fieldBackgroundMixin, IconButton, CloseIcon } from "@firecms/ui";
15
+ import type { FireCMSEditorProps } from "../../editor";
16
16
  import { resolveProperty, resolveStorageFilenameString, resolveStoragePathString } from "../../util";
17
+ import { isImageFile, resizeImage } from "../../util/useStorageUploadController";
18
+ import { lazyEager } from "../../util/lazy_eager";
19
+ import { CircularProgressCenter } from "../../components/CircularProgressCenter";
20
+
21
+ const FireCMSEditor = lazyEager<typeof import("../../editor/editor")["FireCMSEditor"]>(() => import("../../editor/editor"), "FireCMSEditor");
17
22
 
18
23
  interface MarkdownEditorFieldProps {
19
24
  highlight?: { from: number, to: number };
@@ -21,19 +26,19 @@ interface MarkdownEditorFieldProps {
21
26
  }
22
27
 
23
28
  export function MarkdownEditorFieldBinding({
24
- property,
25
- propertyKey,
26
- value,
27
- setValue,
28
- includeDescription,
29
- showError,
30
- error,
31
- minimalistView,
32
- disabled: disabledProp,
33
- isSubmitting,
34
- context,
35
- customProps,
36
- }: FieldProps<string, MarkdownEditorFieldProps>) {
29
+ property,
30
+ propertyKey,
31
+ value,
32
+ setValue,
33
+ includeDescription,
34
+ showError,
35
+ error,
36
+ minimalistView,
37
+ disabled: disabledProp,
38
+ isSubmitting,
39
+ context,
40
+ customProps,
41
+ }: FieldProps<string, MarkdownEditorFieldProps>) {
37
42
 
38
43
  const authController = useAuthController();
39
44
  const disabled = disabledProp || isSubmitting;
@@ -50,22 +55,22 @@ export function MarkdownEditorFieldBinding({
50
55
  const internalValue = useRef<string | null>(value);
51
56
 
52
57
  const onContentChange = useCallback((content: string) => {
53
- if (content === value || (value === null && content === "")) {
58
+ if (content === value || ((value === null || value === undefined) && content === "")) {
54
59
  return;
55
60
  }
56
61
  internalValue.current = content;
57
62
  setValue(content);
58
- }, [setValue]);
63
+ }, [setValue, value]);
59
64
 
60
65
  useEffect(() => {
61
66
  if (internalValue.current !== value) {
62
67
  internalValue.current = value;
63
- setFieldVersion(fieldVersion + 1);
68
+ setFieldVersion(v => v + 1);
64
69
  }
65
70
  }, [value]);
66
71
 
67
72
  const resolvedProperty = resolveProperty({
68
- propertyOrBuilder: property as PropertyOrBuilder,
73
+ propertyOrBuilder: property as PropertyOrBuilder<string>,
69
74
  values: entityValues,
70
75
  authController
71
76
  }) as ResolvedStringProperty | ResolvedArrayProperty<string[]>;
@@ -109,49 +114,85 @@ export function MarkdownEditorFieldBinding({
109
114
  }) ?? "/";
110
115
  }, [entityId, entityValues, path, property, propertyKey, storage]);
111
116
 
112
- const editor = <FireCMSEditor
113
- content={value}
114
- onMarkdownContentChange={onContentChange}
115
- version={context.formex.version + fieldVersion}
116
- highlight={highlight}
117
- disabled={disabled}
118
- handleImageUpload={async (file: File) => {
119
- const storagePath = storagePathBuilder(file);
120
- const fileName = await fileNameBuilder(file);
121
- const result = await storageSource.uploadFile({
122
- file,
123
- fileName,
124
- path: storagePath,
125
- });
126
- const downloadConfig = await storageSource.getDownloadURL(result.path);
127
- const url = downloadConfig.url;
128
- if (!url) {
129
- throw new Error("Error uploading image");
130
- }
131
- return url;
132
- }}
133
- {...editorProps}
134
- />;
117
+ // Extract markdown config from property - can be boolean or object
118
+ const markdownConfig = typeof property.markdown === 'object' ? property.markdown : undefined;
119
+
120
+ const handleImageUpload = async (file: File) => {
121
+ const imageResize = storage?.imageResize;
122
+ const legacyCompression = storage?.imageCompression;
123
+ if ((imageResize || legacyCompression) && isImageFile(file)) {
124
+ file = await resizeImage(file, imageResize, legacyCompression);
125
+ }
126
+
127
+ const storagePath = storagePathBuilder(file);
128
+ const fileName = await fileNameBuilder(file);
129
+ const result = await storageSource.uploadFile({
130
+ file,
131
+ fileName,
132
+ path: storagePath,
133
+ });
134
+ const downloadConfig = await storageSource.getDownloadURL(result.path);
135
+ const url = downloadConfig.url;
136
+ if (!url) {
137
+ throw new Error("Error uploading image");
138
+ }
139
+ return url;
140
+ };
141
+
142
+ const editor = (
143
+ <React.Suspense fallback={<CircularProgressCenter />}>
144
+ <FireCMSEditor
145
+ key={context.formex.version + fieldVersion}
146
+ content={value}
147
+ onMarkdownContentChange={onContentChange}
148
+ version={context.formex.version + fieldVersion}
149
+ highlight={highlight}
150
+ disabled={disabled}
151
+ markdownConfig={markdownConfig}
152
+ handleImageUpload={handleImageUpload}
153
+ {...editorProps}
154
+ />
155
+ </React.Suspense>
156
+ );
135
157
 
136
158
  if (minimalistView)
137
- return editor;
159
+ return (
160
+ <>
161
+ {editor}
162
+ </>
163
+ );
138
164
 
139
165
  return (
140
166
  <>
141
- <LabelWithIconAndTooltip
142
- propertyKey={propertyKey}
143
- icon={getIconForProperty(property, "small")}
144
- required={property.validation?.required}
145
- title={property.name}
146
- className={"h-8 text-text-secondary dark:text-text-secondary-dark ml-3.5"}/>
167
+ <div className="flex items-center w-full">
168
+ <LabelWithIconAndTooltip
169
+ propertyKey={propertyKey}
170
+ icon={getIconForProperty(property, "small")}
171
+ required={property.validation?.required}
172
+ title={property.name}
173
+ className={"h-8 text-text-secondary dark:text-text-secondary-dark ml-3.5"} />
174
+ <div className="flex-grow"/>
175
+ {(property.nullable || property.clearable) && !disabled && (
176
+ <IconButton
177
+ size="small"
178
+ onClick={(e) => {
179
+ e.stopPropagation();
180
+ e.preventDefault();
181
+ setValue(null);
182
+ }}
183
+ >
184
+ <CloseIcon size={"small"}/>
185
+ </IconButton>
186
+ )}
187
+ </div>
147
188
  <div
148
189
  className={cls("rounded-md", fieldBackgroundMixin, disabled ? fieldBackgroundDisabledMixin : fieldBackgroundHoverMixin)}>
149
190
  {editor}
150
191
  </div>
151
192
  <FieldHelperText includeDescription={includeDescription}
152
- showError={showError}
153
- error={error}
154
- property={property}/>
193
+ showError={showError}
194
+ error={error}
195
+ property={property} />
155
196
  </>
156
197
 
157
198
  );
@@ -4,7 +4,7 @@ import { EnumType, FieldProps, ResolvedProperty } from "../../types";
4
4
  import { FieldHelperText, LabelWithIconAndTooltip } from "../components";
5
5
  import { EnumValuesChip } from "../../preview";
6
6
  import { enumToObjectEntries, getIconForProperty, getLabelOrConfigFrom } from "../../util";
7
- import { CloseIcon, MultiSelect, MultiSelectItem } from "@firecms/ui";
7
+ import { CloseIcon, MultiSelect, MultiSelectItem, IconButton } from "@firecms/ui";
8
8
  import { useClearRestoreValue } from "../useClearRestoreValue";
9
9
 
10
10
  /**
@@ -93,6 +93,20 @@ export function MultiSelectFieldBinding({
93
93
  required={property.validation?.required}
94
94
  title={property.name}
95
95
  className={"h-8 text-text-secondary dark:text-text-secondary-dark ml-3.5"}/>}
96
+ endAdornment={
97
+ (property.nullable || property.clearable) && !disabled && value !== null && value !== undefined ? (
98
+ <IconButton
99
+ size="small"
100
+ onClick={(e) => {
101
+ e.stopPropagation();
102
+ e.preventDefault();
103
+ setValue(null);
104
+ }}
105
+ >
106
+ <CloseIcon size={"small"}/>
107
+ </IconButton>
108
+ ) : undefined
109
+ }
96
110
  onValueChange={(updatedValue: string[]) => {
97
111
  let newValue: EnumType[] | null;
98
112
  if (of && (of as ResolvedProperty)?.dataType === "number") {
@@ -8,7 +8,7 @@ import { ReferencePreview } from "../../preview";
8
8
  import { getIconForProperty, IconForView } from "../../util";
9
9
  import { useClearRestoreValue } from "../useClearRestoreValue";
10
10
  import { EntityPreviewContainer } from "../../components/EntityPreview";
11
- import { cls } from "@firecms/ui";
11
+ import { cls, IconButton, CloseIcon } from "@firecms/ui";
12
12
 
13
13
  /**
14
14
  * Field that opens a reference selection dialog and stores the entity ID as a string.
@@ -98,16 +98,30 @@ function ReferenceAsStringFieldBindingInternal({
98
98
 
99
99
  {collection && <>
100
100
 
101
- {referenceValue && <ReferencePreview
102
- disabled={!path}
103
- previewProperties={property.reference?.previewProperties}
104
- hover={!disabled}
105
- size={size}
106
- onClick={disabled || isSubmitting ? undefined : onEntryClick}
107
- reference={referenceValue}
108
- includeEntityLink={property.reference?.includeEntityLink}
109
- includeId={property.reference?.includeId}
110
- />}
101
+ {referenceValue && <div className="flex items-center gap-2">
102
+ <ReferencePreview
103
+ disabled={!path}
104
+ previewProperties={property.reference?.previewProperties}
105
+ hover={!disabled}
106
+ size={size}
107
+ onClick={disabled || isSubmitting ? undefined : onEntryClick}
108
+ reference={referenceValue}
109
+ includeEntityLink={property.reference?.includeEntityLink}
110
+ includeId={property.reference?.includeId}
111
+ />
112
+ {(property.nullable || property.clearable) && !disabled && (
113
+ <IconButton
114
+ size="small"
115
+ onClick={(e) => {
116
+ e.stopPropagation();
117
+ e.preventDefault();
118
+ setValue(null);
119
+ }}
120
+ >
121
+ <CloseIcon size={"small"}/>
122
+ </IconButton>
123
+ )}
124
+ </div>}
111
125
 
112
126
  {!value && <div className="justify-center text-left">
113
127
  <EntityPreviewContainer
@@ -9,7 +9,7 @@ import { EmptyValue, ReferencePreview } from "../../preview";
9
9
  import { getIconForProperty, getReferenceFrom, IconForView } from "../../util";
10
10
  import { useClearRestoreValue } from "../useClearRestoreValue";
11
11
  import { EntityPreviewContainer } from "../../components/EntityPreview";
12
- import { cls } from "@firecms/ui";
12
+ import { cls, IconButton, CloseIcon } from "@firecms/ui";
13
13
 
14
14
  /**
15
15
  * Field that opens a reference selection dialog.
@@ -97,16 +97,30 @@ function ReferenceFieldBindingInternal({
97
97
 
98
98
  {collection && <>
99
99
 
100
- {value && <ReferencePreview
101
- disabled={!property.path}
102
- previewProperties={property.previewProperties}
103
- hover={!disabled}
104
- size={size}
105
- onClick={disabled || isSubmitting ? undefined : onEntryClick}
106
- reference={value}
107
- includeEntityLink={property.includeEntityLink}
108
- includeId={property.includeId}
109
- />}
100
+ {value && <div className="flex items-center gap-2">
101
+ <ReferencePreview
102
+ disabled={!property.path}
103
+ previewProperties={property.previewProperties}
104
+ hover={!disabled}
105
+ size={size}
106
+ onClick={disabled || isSubmitting ? undefined : onEntryClick}
107
+ reference={value}
108
+ includeEntityLink={property.includeEntityLink}
109
+ includeId={property.includeId}
110
+ />
111
+ {(property.nullable || property.clearable) && !disabled && (
112
+ <IconButton
113
+ size="small"
114
+ onClick={(e) => {
115
+ e.stopPropagation();
116
+ e.preventDefault();
117
+ setValue(null);
118
+ }}
119
+ >
120
+ <CloseIcon size={"small"}/>
121
+ </IconButton>
122
+ )}
123
+ </div>}
110
124
 
111
125
  {!value && <div className="justify-center text-left">
112
126
  <EntityPreviewContainer className={cls("px-6 h-16 text-sm font-medium flex items-center gap-6",
@@ -4,9 +4,10 @@ import { FieldHelperText, LabelWithIconAndTooltip } from "../components";
4
4
  import { ArrayContainer, ArrayEntryParams, ErrorBoundary } from "../../components";
5
5
  import { getArrayResolvedProperties, getDefaultValueFor, getIconForProperty, mergeDeep } from "../../util";
6
6
  import { PropertyFieldBinding } from "../PropertyFieldBinding";
7
- import { ExpandablePanel, Typography } from "@firecms/ui";
7
+ import { ExpandablePanel, Typography, IconButton, CloseIcon } from "@firecms/ui";
8
8
  import { useClearRestoreValue } from "../useClearRestoreValue";
9
9
  import { useAuthController } from "../../hooks";
10
+ import { useTranslation } from "../../hooks/useTranslation";
10
11
 
11
12
  /**
12
13
  * Generic array field that allows reordering and renders the child property
@@ -34,6 +35,7 @@ export function RepeatFieldBinding<T extends Array<any>>({
34
35
 
35
36
  const authController = useAuthController();
36
37
  const minimalistView = minimalistViewProp || property.minimalistView;
38
+ const { t } = useTranslation();
37
39
 
38
40
  if (!property.of)
39
41
  throw Error("RepeatFieldBinding misconfiguration. Property `of` not set");
@@ -87,7 +89,7 @@ export function RepeatFieldBinding<T extends Array<any>>({
87
89
  const canAddElements = !property.disabled && !isSubmitting && !disabled && (property.canAddElements || property.canAddElements === undefined);
88
90
  const sortable = property.sortable === undefined ? true : property.sortable;
89
91
  const arrayContainer = <ArrayContainer droppableId={propertyKey}
90
- addLabel={property.name ? "Add entry to " + property.name : "Add entry"}
92
+ addLabel={property.name ? t("add_to_field", { fieldName: property.name }) : t("add_entry")}
91
93
  value={value}
92
94
  buildEntry={buildEntry}
93
95
  onInternalIdAdded={setLastAddedId}
@@ -99,15 +101,28 @@ export function RepeatFieldBinding<T extends Array<any>>({
99
101
  className={property.widthPercentage !== undefined ? "mt-8" : undefined}
100
102
  />;
101
103
 
102
- const title = (<>
104
+ const title = (<div className="flex items-center w-full">
103
105
  <LabelWithIconAndTooltip
104
106
  propertyKey={propertyKey}
105
107
  icon={getIconForProperty(property, "small")}
106
108
  required={property.validation?.required}
107
109
  title={property.name}
108
- className={"h-8 flex flex-grow text-text-secondary dark:text-text-secondary-dark"}/>
109
- {Array.isArray(value) && <Typography variant={"caption"} className={"px-4"}>({value.length})</Typography>}
110
- </>);
110
+ className={"text-text-secondary dark:text-text-secondary-dark"}/>
111
+ {Array.isArray(value) && <span className={"text-sm text-text-secondary dark:text-text-secondary-dark ml-1"}>({value.length})</span>}
112
+ <div className="flex-grow"/>
113
+ {(property.nullable || property.clearable) && !disabled && (
114
+ <IconButton
115
+ size="small"
116
+ onClick={(e) => {
117
+ e.stopPropagation();
118
+ e.preventDefault();
119
+ setValue(null);
120
+ }}
121
+ >
122
+ <CloseIcon size={"small"}/>
123
+ </IconButton>
124
+ )}
125
+ </div>);
111
126
 
112
127
  return (
113
128
 
@@ -66,11 +66,13 @@ export function SelectFieldBinding<T extends EnumType>({
66
66
  />
67
67
  </PropertyIdCopyTooltip>}
68
68
  endAdornment={
69
- property.clearable && !disabled && <IconButton
70
- size="small"
71
- onClick={handleClearClick}>
72
- <CloseIcon size={"small"}/>
73
- </IconButton>
69
+ (property.nullable || property.clearable) && !disabled && value !== null && value !== undefined ? (
70
+ <IconButton
71
+ size="small"
72
+ onClick={handleClearClick}>
73
+ <CloseIcon size={"small"}/>
74
+ </IconButton>
75
+ ) : undefined
74
76
  }
75
77
  onValueChange={(updatedValue: string) => {
76
78
  const newValue = updatedValue