@firecms/core 3.0.0-canary.29 → 3.0.0-canary.290

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 (433) hide show
  1. package/README.md +3 -3
  2. package/dist/app/AppBar.d.ts +12 -0
  3. package/dist/app/Drawer.d.ts +16 -0
  4. package/dist/app/Scaffold.d.ts +34 -0
  5. package/dist/app/index.d.ts +4 -0
  6. package/dist/app/useApp.d.ts +16 -0
  7. package/dist/components/ArrayContainer.d.ts +31 -12
  8. package/dist/components/CircularProgressCenter.d.ts +1 -1
  9. package/dist/components/ClearFilterSortButton.d.ts +5 -0
  10. package/dist/components/{DeleteConfirmationDialog.d.ts → ConfirmationDialog.d.ts} +1 -1
  11. package/dist/components/EntityCollectionTable/EntityCollectionRowActions.d.ts +14 -13
  12. package/dist/components/EntityCollectionTable/EntityCollectionTable.d.ts +2 -2
  13. package/dist/components/EntityCollectionTable/EntityCollectionTableProps.d.ts +22 -6
  14. package/dist/components/EntityCollectionTable/PropertyTableCell.d.ts +1 -0
  15. package/dist/components/EntityCollectionTable/column_utils.d.ts +1 -2
  16. package/dist/components/EntityCollectionTable/fields/TableReferenceField.d.ts +3 -1
  17. package/dist/components/EntityCollectionTable/index.d.ts +1 -1
  18. package/dist/components/EntityCollectionTable/internal/CollectionTableToolbar.d.ts +1 -4
  19. package/dist/components/EntityCollectionTable/internal/EntityTableCell.d.ts +2 -2
  20. package/dist/components/EntityCollectionTable/internal/popup_field/PopupFormField.d.ts +7 -4
  21. package/dist/components/EntityCollectionView/EntityCollectionView.d.ts +20 -2
  22. package/dist/components/EntityCollectionView/EntityCollectionViewStartActions.d.ts +11 -0
  23. package/dist/components/EntityCollectionView/utils.d.ts +3 -0
  24. package/dist/components/EntityJsonPreview.d.ts +3 -0
  25. package/dist/components/EntityPreview.d.ts +10 -7
  26. package/dist/components/ErrorView.d.ts +1 -1
  27. package/dist/components/HomePage/DefaultHomePage.d.ts +2 -15
  28. package/dist/components/HomePage/HomePageDnD.d.ts +77 -0
  29. package/dist/components/HomePage/NavigationCard.d.ts +3 -1
  30. package/dist/components/HomePage/NavigationCardBinding.d.ts +4 -3
  31. package/dist/components/HomePage/NavigationGroup.d.ts +8 -1
  32. package/dist/components/HomePage/RenameGroupDialog.d.ts +9 -0
  33. package/dist/components/PropertyCollectionView.d.ts +23 -0
  34. package/dist/components/PropertyConfigBadge.d.ts +2 -1
  35. package/dist/components/PropertyIdCopyTooltip.d.ts +8 -0
  36. package/dist/components/ReferenceWidget.d.ts +3 -1
  37. package/dist/components/SelectableTable/SelectableTable.d.ts +14 -4
  38. package/dist/components/SelectableTable/filters/ReferenceFilterField.d.ts +2 -1
  39. package/dist/components/UnsavedChangesDialog.d.ts +8 -0
  40. package/dist/components/UserDisplay.d.ts +7 -0
  41. package/dist/components/VirtualTable/VirtualTableProps.d.ts +24 -12
  42. package/dist/components/VirtualTable/fields/VirtualTableUserSelect.d.ts +12 -0
  43. package/dist/components/VirtualTable/types.d.ts +3 -3
  44. package/dist/components/{EntityCollectionTable/internal → common}/default_entity_actions.d.ts +1 -3
  45. package/dist/components/common/index.d.ts +2 -1
  46. package/dist/components/common/table_height.d.ts +5 -0
  47. package/dist/components/common/types.d.ts +4 -6
  48. package/dist/components/common/useColumnsIds.d.ts +3 -1
  49. package/dist/components/common/{useDataSourceEntityCollectionTableController.d.ts → useDataSourceTableController.d.ts} +13 -2
  50. package/dist/components/common/useDebouncedCallback.d.ts +1 -0
  51. package/dist/components/common/useScrollRestoration.d.ts +14 -0
  52. package/dist/components/index.d.ts +5 -2
  53. package/dist/contexts/BreacrumbsContext.d.ts +8 -0
  54. package/dist/contexts/InternalUserManagementContext.d.ts +3 -0
  55. package/dist/core/DefaultAppBar.d.ts +29 -0
  56. package/dist/core/DefaultDrawer.d.ts +19 -0
  57. package/dist/core/DrawerNavigationItem.d.ts +10 -0
  58. package/dist/core/EntityEditView.d.ts +49 -11
  59. package/dist/core/EntityEditViewFormActions.d.ts +2 -0
  60. package/dist/core/FireCMS.d.ts +2 -3
  61. package/dist/core/FireCMSRouter.d.ts +4 -0
  62. package/dist/core/NavigationRoutes.d.ts +2 -3
  63. package/dist/core/SideDialogs.d.ts +4 -2
  64. package/dist/core/field_configs.d.ts +1 -1
  65. package/dist/core/index.d.ts +4 -4
  66. package/dist/form/EntityForm.d.ts +40 -64
  67. package/dist/form/EntityFormActions.d.ts +21 -0
  68. package/dist/form/PropertyFieldBinding.d.ts +1 -1
  69. package/dist/form/components/ErrorFocus.d.ts +1 -1
  70. package/dist/form/components/FieldHelperText.d.ts +3 -3
  71. package/dist/form/components/FormEntry.d.ts +6 -0
  72. package/dist/form/components/FormLayout.d.ts +5 -0
  73. package/dist/form/components/LabelWithIcon.d.ts +1 -1
  74. package/dist/form/components/LabelWithIconAndTooltip.d.ts +15 -0
  75. package/dist/form/components/LocalChangesMenu.d.ts +11 -0
  76. package/dist/form/components/StorageItemPreview.d.ts +4 -4
  77. package/dist/form/components/index.d.ts +3 -1
  78. package/dist/form/field_bindings/ArrayCustomShapedFieldBinding.d.ts +1 -1
  79. package/dist/form/field_bindings/ArrayOfReferencesFieldBinding.d.ts +1 -1
  80. package/dist/form/field_bindings/BlockFieldBinding.d.ts +1 -1
  81. package/dist/form/field_bindings/KeyValueFieldBinding.d.ts +1 -1
  82. package/dist/form/field_bindings/MapFieldBinding.d.ts +1 -1
  83. package/dist/form/field_bindings/MarkdownEditorFieldBinding.d.ts +11 -0
  84. package/dist/form/field_bindings/{MultiSelectBinding.d.ts → MultiSelectFieldBinding.d.ts} +1 -1
  85. package/dist/form/field_bindings/ReadOnlyFieldBinding.d.ts +1 -1
  86. package/dist/form/field_bindings/ReferenceAsStringFieldBinding.d.ts +9 -0
  87. package/dist/form/field_bindings/ReferenceFieldBinding.d.ts +2 -2
  88. package/dist/form/field_bindings/RepeatFieldBinding.d.ts +1 -1
  89. package/dist/form/field_bindings/SelectFieldBinding.d.ts +1 -1
  90. package/dist/form/field_bindings/StorageUploadFieldBinding.d.ts +5 -13
  91. package/dist/form/field_bindings/SwitchFieldBinding.d.ts +1 -2
  92. package/dist/form/field_bindings/TextFieldBinding.d.ts +1 -1
  93. package/dist/form/field_bindings/UserSelectFieldBinding.d.ts +12 -0
  94. package/dist/form/index.d.ts +18 -18
  95. package/dist/form/useClearRestoreValue.d.ts +2 -2
  96. package/dist/hooks/data/delete.d.ts +4 -4
  97. package/dist/hooks/data/save.d.ts +4 -5
  98. package/dist/hooks/data/useCollectionFetch.d.ts +1 -1
  99. package/dist/hooks/data/useEntityFetch.d.ts +4 -3
  100. package/dist/hooks/index.d.ts +3 -0
  101. package/dist/hooks/useAuthController.d.ts +1 -1
  102. package/dist/hooks/useBreadcrumbsController.d.ts +26 -0
  103. package/dist/hooks/useBuildNavigationController.d.ts +57 -13
  104. package/dist/hooks/useCollapsedGroups.d.ts +9 -0
  105. package/dist/hooks/useFireCMSContext.d.ts +1 -1
  106. package/dist/hooks/useInternalUserManagementController.d.ts +12 -0
  107. package/dist/hooks/useModeController.d.ts +1 -2
  108. package/dist/hooks/useProjectLog.d.ts +8 -2
  109. package/dist/hooks/useResolvedNavigationFrom.d.ts +3 -3
  110. package/dist/hooks/useValidateAuthenticator.d.ts +4 -8
  111. package/dist/index.d.ts +1 -0
  112. package/dist/index.es.js +24546 -13965
  113. package/dist/index.es.js.map +1 -1
  114. package/dist/index.umd.js +27256 -588
  115. package/dist/index.umd.js.map +1 -1
  116. package/dist/internal/useBuildDataSource.d.ts +3 -17
  117. package/dist/internal/useBuildSideEntityController.d.ts +3 -3
  118. package/dist/internal/useUnsavedChangesDialog.d.ts +7 -9
  119. package/dist/preview/PropertyPreviewProps.d.ts +6 -1
  120. package/dist/preview/components/EnumValuesChip.d.ts +1 -1
  121. package/dist/preview/components/ReferencePreview.d.ts +4 -3
  122. package/dist/preview/components/StorageThumbnail.d.ts +2 -1
  123. package/dist/preview/components/UrlComponentPreview.d.ts +2 -1
  124. package/dist/preview/components/UserPreview.d.ts +8 -0
  125. package/dist/preview/index.d.ts +1 -0
  126. package/dist/preview/util.d.ts +3 -3
  127. package/dist/routes/CustomCMSRoute.d.ts +4 -0
  128. package/dist/routes/FireCMSRoute.d.ts +1 -0
  129. package/dist/routes/HomePageRoute.d.ts +3 -0
  130. package/dist/types/analytics.d.ts +1 -1
  131. package/dist/types/auth.d.ts +8 -10
  132. package/dist/types/collections.d.ts +123 -25
  133. package/dist/types/customization_controller.d.ts +8 -0
  134. package/dist/types/datasource.d.ts +52 -36
  135. package/dist/types/dialogs_controller.d.ts +7 -3
  136. package/dist/types/entities.d.ts +12 -3
  137. package/dist/types/entity_actions.d.ts +72 -8
  138. package/dist/types/entity_callbacks.d.ts +16 -16
  139. package/dist/types/entity_overrides.d.ts +2 -2
  140. package/dist/types/export_import.d.ts +4 -4
  141. package/dist/types/fields.d.ts +79 -39
  142. package/dist/types/firecms.d.ts +31 -3
  143. package/dist/types/firecms_context.d.ts +17 -1
  144. package/dist/types/index.d.ts +1 -1
  145. package/dist/types/internal_user_management.d.ts +20 -0
  146. package/dist/types/navigation.d.ts +62 -19
  147. package/dist/types/permissions.d.ts +4 -4
  148. package/dist/types/plugins.d.ts +58 -13
  149. package/dist/types/properties.d.ts +122 -31
  150. package/dist/types/property_config.d.ts +1 -3
  151. package/dist/types/roles.d.ts +3 -0
  152. package/dist/types/side_dialogs_controller.d.ts +10 -0
  153. package/dist/types/side_entity_controller.d.ts +14 -1
  154. package/dist/types/storage.d.ts +75 -0
  155. package/dist/types/user.d.ts +2 -1
  156. package/dist/util/builders.d.ts +3 -3
  157. package/dist/util/callbacks.d.ts +2 -0
  158. package/dist/util/collections.d.ts +1 -0
  159. package/dist/util/createFormexStub.d.ts +2 -0
  160. package/dist/util/entities.d.ts +3 -3
  161. package/dist/util/entity_actions.d.ts +2 -0
  162. package/dist/util/entity_cache.d.ts +28 -0
  163. package/dist/util/icon_list.d.ts +5 -1
  164. package/dist/util/icon_synonyms.d.ts +1 -98
  165. package/dist/util/icons.d.ts +7 -4
  166. package/dist/util/index.d.ts +3 -0
  167. package/dist/util/make_properties_editable.d.ts +1 -2
  168. package/dist/util/navigation_from_path.d.ts +10 -1
  169. package/dist/util/navigation_utils.d.ts +15 -3
  170. package/dist/util/objects.d.ts +3 -1
  171. package/dist/util/permissions.d.ts +4 -4
  172. package/dist/util/plurals.d.ts +0 -2
  173. package/dist/util/property_utils.d.ts +4 -4
  174. package/dist/util/references.d.ts +2 -2
  175. package/dist/util/resolutions.d.ts +42 -17
  176. package/dist/util/storage.d.ts +23 -2
  177. package/dist/util/useStorageUploadController.d.ts +4 -3
  178. package/package.json +70 -53
  179. package/src/app/AppBar.tsx +18 -0
  180. package/src/app/Drawer.tsx +24 -0
  181. package/src/app/Scaffold.tsx +253 -0
  182. package/src/app/index.ts +4 -0
  183. package/src/app/useApp.tsx +32 -0
  184. package/src/components/ArrayContainer.tsx +447 -229
  185. package/src/components/CircularProgressCenter.tsx +2 -2
  186. package/src/components/ClearFilterSortButton.tsx +41 -0
  187. package/src/components/{DeleteConfirmationDialog.tsx → ConfirmationDialog.tsx} +12 -11
  188. package/src/components/DeleteEntityDialog.tsx +13 -20
  189. package/src/components/EntityCollectionTable/EntityCollectionRowActions.tsx +87 -62
  190. package/src/components/EntityCollectionTable/EntityCollectionTable.tsx +38 -31
  191. package/src/components/EntityCollectionTable/EntityCollectionTableProps.tsx +30 -9
  192. package/src/components/EntityCollectionTable/PropertyTableCell.tsx +84 -42
  193. package/src/components/EntityCollectionTable/column_utils.tsx +3 -3
  194. package/src/components/EntityCollectionTable/fields/TableReferenceField.tsx +30 -16
  195. package/src/components/EntityCollectionTable/fields/TableStorageUpload.tsx +19 -17
  196. package/src/components/EntityCollectionTable/index.tsx +1 -1
  197. package/src/components/EntityCollectionTable/internal/CollectionTableToolbar.tsx +34 -39
  198. package/src/components/EntityCollectionTable/internal/EntityTableCell.tsx +49 -36
  199. package/src/components/EntityCollectionTable/internal/EntityTableCellActions.tsx +20 -8
  200. package/src/components/EntityCollectionTable/internal/popup_field/PopupFormField.tsx +135 -105
  201. package/src/components/EntityCollectionTable/internal/popup_field/useDraggable.tsx +9 -9
  202. package/src/components/EntityCollectionView/EntityCollectionView.tsx +241 -119
  203. package/src/components/EntityCollectionView/EntityCollectionViewActions.tsx +7 -4
  204. package/src/components/EntityCollectionView/EntityCollectionViewStartActions.tsx +68 -0
  205. package/src/components/EntityCollectionView/useSelectionController.tsx +20 -7
  206. package/src/components/EntityCollectionView/utils.ts +19 -0
  207. package/src/components/EntityJsonPreview.tsx +66 -0
  208. package/src/components/EntityPreview.tsx +83 -62
  209. package/src/components/EntityView.tsx +34 -42
  210. package/src/components/ErrorView.tsx +4 -4
  211. package/src/components/FireCMSLogo.tsx +7 -51
  212. package/src/components/HomePage/DefaultHomePage.tsx +516 -158
  213. package/src/components/HomePage/FavouritesView.tsx +9 -14
  214. package/src/components/HomePage/HomePageDnD.tsx +702 -0
  215. package/src/components/HomePage/NavigationCard.tsx +48 -39
  216. package/src/components/HomePage/NavigationCardBinding.tsx +17 -16
  217. package/src/components/HomePage/NavigationGroup.tsx +144 -30
  218. package/src/components/HomePage/RenameGroupDialog.tsx +123 -0
  219. package/src/components/HomePage/SmallNavigationCard.tsx +5 -6
  220. package/src/components/NotFoundPage.tsx +2 -2
  221. package/src/components/PropertyCollectionView.tsx +329 -0
  222. package/src/components/PropertyConfigBadge.tsx +10 -4
  223. package/src/components/PropertyIdCopyTooltip.tsx +47 -0
  224. package/src/components/ReferenceTable/ReferenceSelectionTable.tsx +23 -13
  225. package/src/components/ReferenceWidget.tsx +21 -11
  226. package/src/components/SearchIconsView.tsx +10 -7
  227. package/src/components/SelectableTable/SelectableTable.tsx +157 -157
  228. package/src/components/SelectableTable/filters/BooleanFilterField.tsx +2 -3
  229. package/src/components/SelectableTable/filters/DateTimeFilterField.tsx +27 -9
  230. package/src/components/SelectableTable/filters/ReferenceFilterField.tsx +36 -12
  231. package/src/components/SelectableTable/filters/StringNumberFilterField.tsx +92 -24
  232. package/src/components/UnsavedChangesDialog.tsx +46 -0
  233. package/src/components/UserDisplay.tsx +55 -0
  234. package/src/components/VirtualTable/VirtualTable.tsx +105 -51
  235. package/src/components/VirtualTable/VirtualTableCell.tsx +1 -9
  236. package/src/components/VirtualTable/VirtualTableHeader.tsx +10 -10
  237. package/src/components/VirtualTable/VirtualTableHeaderRow.tsx +2 -2
  238. package/src/components/VirtualTable/VirtualTableProps.tsx +28 -14
  239. package/src/components/VirtualTable/VirtualTableRow.tsx +5 -6
  240. package/src/components/VirtualTable/fields/VirtualTableDateField.tsx +5 -5
  241. package/src/components/VirtualTable/fields/VirtualTableInput.tsx +2 -2
  242. package/src/components/VirtualTable/fields/VirtualTableNumberInput.tsx +2 -1
  243. package/src/components/VirtualTable/fields/VirtualTableSelect.tsx +16 -28
  244. package/src/components/VirtualTable/fields/VirtualTableUserSelect.tsx +99 -0
  245. package/src/components/VirtualTable/types.tsx +2 -3
  246. package/src/components/{EntityCollectionTable/internal → common}/default_entity_actions.tsx +64 -44
  247. package/src/components/common/index.ts +2 -1
  248. package/src/components/{VirtualTable/common.tsx → common/table_height.tsx} +5 -2
  249. package/src/components/common/types.tsx +4 -6
  250. package/src/components/common/useColumnsIds.tsx +16 -2
  251. package/src/components/common/useDataSourceTableController.tsx +420 -0
  252. package/src/components/common/useDebouncedCallback.tsx +20 -0
  253. package/src/components/common/useScrollRestoration.tsx +68 -0
  254. package/src/components/common/useTableSearchHelper.ts +53 -12
  255. package/src/components/index.tsx +6 -2
  256. package/src/contexts/BreacrumbsContext.tsx +38 -0
  257. package/src/contexts/DialogsProvider.tsx +5 -4
  258. package/src/contexts/InternalUserManagementContext.tsx +4 -0
  259. package/src/contexts/ModeController.tsx +1 -3
  260. package/src/contexts/SnackbarProvider.tsx +2 -0
  261. package/src/core/DefaultAppBar.tsx +219 -0
  262. package/src/core/DefaultDrawer.tsx +185 -0
  263. package/src/core/DrawerNavigationItem.tsx +66 -0
  264. package/src/core/EntityEditView.tsx +447 -469
  265. package/src/core/EntityEditViewFormActions.tsx +344 -0
  266. package/src/core/EntitySidePanel.tsx +96 -23
  267. package/src/core/FireCMS.tsx +85 -60
  268. package/src/core/FireCMSRouter.tsx +17 -0
  269. package/src/core/NavigationRoutes.tsx +28 -38
  270. package/src/core/SideDialogs.tsx +22 -12
  271. package/src/core/field_configs.tsx +41 -14
  272. package/src/core/index.tsx +6 -5
  273. package/src/form/EntityForm.tsx +740 -523
  274. package/src/form/EntityFormActions.tsx +226 -0
  275. package/src/form/PropertyFieldBinding.tsx +88 -41
  276. package/src/form/components/CustomIdField.tsx +9 -3
  277. package/src/form/components/ErrorFocus.tsx +22 -29
  278. package/src/form/components/FieldHelperText.tsx +4 -4
  279. package/src/form/components/FormEntry.tsx +22 -0
  280. package/src/form/components/FormLayout.tsx +16 -0
  281. package/src/form/components/LabelWithIcon.tsx +30 -19
  282. package/src/form/components/LabelWithIconAndTooltip.tsx +28 -0
  283. package/src/form/components/LocalChangesMenu.tsx +144 -0
  284. package/src/form/components/StorageItemPreview.tsx +23 -13
  285. package/src/form/components/StorageUploadProgress.tsx +5 -6
  286. package/src/form/components/index.tsx +3 -1
  287. package/src/form/field_bindings/ArrayCustomShapedFieldBinding.tsx +34 -19
  288. package/src/form/field_bindings/ArrayOfReferencesFieldBinding.tsx +50 -36
  289. package/src/form/field_bindings/BlockFieldBinding.tsx +56 -33
  290. package/src/form/field_bindings/DateTimeFieldBinding.tsx +18 -14
  291. package/src/form/field_bindings/KeyValueFieldBinding.tsx +61 -52
  292. package/src/form/field_bindings/MapFieldBinding.tsx +73 -55
  293. package/src/form/field_bindings/MarkdownEditorFieldBinding.tsx +159 -0
  294. package/src/form/field_bindings/{MultiSelectBinding.tsx → MultiSelectFieldBinding.tsx} +26 -21
  295. package/src/form/field_bindings/ReadOnlyFieldBinding.tsx +11 -16
  296. package/src/form/field_bindings/ReferenceAsStringFieldBinding.tsx +135 -0
  297. package/src/form/field_bindings/ReferenceFieldBinding.tsx +42 -31
  298. package/src/form/field_bindings/RepeatFieldBinding.tsx +62 -35
  299. package/src/form/field_bindings/SelectFieldBinding.tsx +24 -15
  300. package/src/form/field_bindings/StorageUploadFieldBinding.tsx +257 -199
  301. package/src/form/field_bindings/SwitchFieldBinding.tsx +29 -24
  302. package/src/form/field_bindings/TextFieldBinding.tsx +28 -24
  303. package/src/form/field_bindings/UserSelectFieldBinding.tsx +94 -0
  304. package/src/form/index.tsx +21 -37
  305. package/src/form/useClearRestoreValue.tsx +2 -2
  306. package/src/form/validation.ts +13 -23
  307. package/src/hooks/data/delete.ts +6 -5
  308. package/src/hooks/data/save.ts +26 -33
  309. package/src/hooks/data/useCollectionFetch.tsx +3 -3
  310. package/src/hooks/data/useDataSource.tsx +11 -3
  311. package/src/hooks/data/useEntityFetch.tsx +10 -6
  312. package/src/hooks/index.tsx +4 -0
  313. package/src/hooks/useAuthController.tsx +1 -1
  314. package/src/hooks/useBreadcrumbsController.tsx +31 -0
  315. package/src/hooks/useBrowserTitleAndIcon.tsx +1 -1
  316. package/src/hooks/useBuildLocalConfigurationPersistence.tsx +8 -10
  317. package/src/hooks/useBuildModeController.tsx +22 -29
  318. package/src/hooks/useBuildNavigationController.tsx +515 -121
  319. package/src/hooks/useCollapsedGroups.ts +64 -0
  320. package/src/hooks/useFireCMSContext.tsx +9 -35
  321. package/src/hooks/useInternalUserManagementController.tsx +16 -0
  322. package/src/hooks/useLargeLayout.tsx +0 -35
  323. package/src/hooks/useModeController.tsx +1 -2
  324. package/src/hooks/useProjectLog.tsx +32 -10
  325. package/src/hooks/useResolvedNavigationFrom.tsx +10 -12
  326. package/src/hooks/useValidateAuthenticator.tsx +17 -37
  327. package/src/index.ts +1 -0
  328. package/src/internal/useBuildDataSource.ts +79 -85
  329. package/src/internal/useBuildSideDialogsController.tsx +4 -2
  330. package/src/internal/useBuildSideEntityController.tsx +204 -77
  331. package/src/internal/useUnsavedChangesDialog.tsx +127 -91
  332. package/src/preview/PropertyPreview.tsx +42 -25
  333. package/src/preview/PropertyPreviewProps.tsx +7 -1
  334. package/src/preview/components/BooleanPreview.tsx +2 -2
  335. package/src/preview/components/EmptyValue.tsx +1 -1
  336. package/src/preview/components/EnumValuesChip.tsx +2 -2
  337. package/src/preview/components/ImagePreview.tsx +26 -37
  338. package/src/preview/components/ReferencePreview.tsx +30 -38
  339. package/src/preview/components/StorageThumbnail.tsx +5 -1
  340. package/src/preview/components/UrlComponentPreview.tsx +60 -28
  341. package/src/preview/components/UserPreview.tsx +27 -0
  342. package/src/preview/index.ts +1 -0
  343. package/src/preview/property_previews/ArrayOfMapsPreview.tsx +6 -6
  344. package/src/preview/property_previews/ArrayOfReferencesPreview.tsx +7 -5
  345. package/src/preview/property_previews/ArrayOfStorageComponentsPreview.tsx +5 -4
  346. package/src/preview/property_previews/ArrayOfStringsPreview.tsx +4 -4
  347. package/src/preview/property_previews/ArrayOneOfPreview.tsx +7 -6
  348. package/src/preview/property_previews/ArrayPropertyPreview.tsx +8 -7
  349. package/src/preview/property_previews/MapPropertyPreview.tsx +14 -13
  350. package/src/preview/property_previews/NumberPropertyPreview.tsx +2 -2
  351. package/src/preview/property_previews/SkeletonPropertyComponent.tsx +13 -13
  352. package/src/preview/property_previews/StringPropertyPreview.tsx +3 -3
  353. package/src/preview/util.ts +10 -10
  354. package/src/routes/CustomCMSRoute.tsx +21 -0
  355. package/src/routes/FireCMSRoute.tsx +246 -0
  356. package/src/routes/HomePageRoute.tsx +17 -0
  357. package/src/types/analytics.ts +3 -0
  358. package/src/types/auth.tsx +9 -13
  359. package/src/types/collections.ts +146 -30
  360. package/src/types/customization_controller.tsx +9 -1
  361. package/src/types/datasource.ts +61 -43
  362. package/src/types/dialogs_controller.tsx +7 -3
  363. package/src/types/entities.ts +19 -3
  364. package/src/types/entity_actions.tsx +86 -10
  365. package/src/types/entity_callbacks.ts +18 -18
  366. package/src/types/entity_overrides.tsx +2 -2
  367. package/src/types/export_import.ts +4 -4
  368. package/src/types/fields.tsx +91 -42
  369. package/src/types/firecms.tsx +34 -4
  370. package/src/types/firecms_context.tsx +18 -1
  371. package/src/types/index.ts +1 -1
  372. package/src/types/internal_user_management.ts +24 -0
  373. package/src/types/navigation.ts +77 -24
  374. package/src/types/permissions.ts +5 -5
  375. package/src/types/plugins.tsx +69 -15
  376. package/src/types/properties.ts +141 -33
  377. package/src/types/property_config.tsx +2 -2
  378. package/src/types/roles.ts +3 -0
  379. package/src/types/side_dialogs_controller.tsx +15 -0
  380. package/src/types/side_entity_controller.tsx +16 -1
  381. package/src/types/storage.ts +83 -1
  382. package/src/types/user.ts +3 -1
  383. package/src/util/builders.ts +10 -8
  384. package/src/util/callbacks.ts +119 -0
  385. package/src/util/collections.ts +8 -0
  386. package/src/util/createFormexStub.tsx +66 -0
  387. package/src/util/entities.ts +11 -8
  388. package/src/util/entity_actions.ts +28 -0
  389. package/src/util/entity_cache.ts +223 -0
  390. package/src/util/enums.ts +1 -1
  391. package/src/util/icon_list.ts +16 -10
  392. package/src/util/icon_synonyms.ts +3 -100
  393. package/src/util/icons.tsx +36 -11
  394. package/src/util/index.ts +3 -0
  395. package/src/util/join_collections.ts +11 -4
  396. package/src/util/make_properties_editable.ts +5 -19
  397. package/src/util/navigation_from_path.ts +33 -12
  398. package/src/util/navigation_utils.ts +141 -25
  399. package/src/util/objects.ts +128 -33
  400. package/src/util/parent_references_from_path.ts +3 -3
  401. package/src/util/permissions.ts +9 -8
  402. package/src/util/plurals.ts +0 -2
  403. package/src/util/property_utils.tsx +17 -6
  404. package/src/util/references.ts +19 -8
  405. package/src/util/resolutions.ts +122 -48
  406. package/src/util/storage.ts +79 -21
  407. package/src/util/strings.ts +2 -2
  408. package/src/util/useStorageUploadController.tsx +162 -62
  409. package/dist/components/EntityCollectionTable/internal/popup_field/ElementResizeListener.d.ts +0 -5
  410. package/dist/components/FireCMSAppBar.d.ts +0 -26
  411. package/dist/components/PropertyIdCopyTooltipContent.d.ts +0 -3
  412. package/dist/components/VirtualTable/common.d.ts +0 -2
  413. package/dist/core/Drawer.d.ts +0 -23
  414. package/dist/core/Scaffold.d.ts +0 -55
  415. package/dist/core/SideEntityView.d.ts +0 -7
  416. package/dist/form/components/FormikArrayContainer.d.ts +0 -18
  417. package/dist/form/field_bindings/MarkdownFieldBinding.d.ts +0 -9
  418. package/dist/internal/useBuildCustomizationController.d.ts +0 -2
  419. package/dist/internal/useLocaleConfig.d.ts +0 -1
  420. package/dist/types/appcheck.d.ts +0 -26
  421. package/src/components/EntityCollectionTable/internal/popup_field/ElementResizeListener.tsx +0 -59
  422. package/src/components/FireCMSAppBar.tsx +0 -165
  423. package/src/components/PropertyIdCopyTooltipContent.tsx +0 -28
  424. package/src/components/common/useDataSourceEntityCollectionTableController.tsx +0 -225
  425. package/src/core/Drawer.tsx +0 -191
  426. package/src/core/Scaffold.tsx +0 -281
  427. package/src/core/SideEntityView.tsx +0 -38
  428. package/src/form/components/FormikArrayContainer.tsx +0 -44
  429. package/src/form/field_bindings/MarkdownFieldBinding.tsx +0 -695
  430. package/src/internal/useBuildCustomizationController.tsx +0 -5
  431. package/src/internal/useLocaleConfig.tsx +0 -18
  432. package/src/types/appcheck.ts +0 -29
  433. /package/src/util/{common.tsx → common.ts} +0 -0
@@ -1,47 +1,46 @@
1
- import React, { useCallback, useEffect, useState } from "react";
2
-
3
- import { useCustomizationController, useFireCMSContext, useNavigationController } from "../../hooks";
4
- import { CMSAnalyticsEvent, PluginGenericProps, PluginHomePageAdditionalCardsProps } from "../../types";
5
-
1
+ import React, { useCallback, useEffect, useMemo, useRef, useState } from "react";
2
+ import Fuse from "fuse.js";
3
+ import { Container, SearchBar } from "@firecms/ui";
4
+ import {
5
+ useCollapsedGroups,
6
+ useCustomizationController,
7
+ useFireCMSContext,
8
+ useNavigationController
9
+ } from "../../hooks";
10
+ import {
11
+ CMSAnalyticsEvent,
12
+ NavigationEntry,
13
+ NavigationGroupMapping,
14
+ PluginGenericProps,
15
+ PluginHomePageAdditionalCardsProps
16
+ } from "../../types";
6
17
  import { toArray } from "../../util/arrays";
18
+ import { FavouritesView } from "./FavouritesView";
19
+ import { useRestoreScroll } from "../../internal/useRestoreScroll";
7
20
  import { NavigationGroup } from "./NavigationGroup";
21
+ import {
22
+ NavigationGroupDroppable,
23
+ NewGroupDropZone,
24
+ SortableNavigationCard,
25
+ SortableNavigationGroup,
26
+ useHomePageDnd
27
+ } from "./HomePageDnD";
28
+ import { DndContext, DragOverlay, MeasuringStrategy } from "@dnd-kit/core";
8
29
  import { NavigationCardBinding } from "./NavigationCardBinding";
30
+ import { SortableContext, verticalListSortingStrategy } from "@dnd-kit/sortable";
31
+ import { restrictToVerticalAxis, restrictToWindowEdges } from "@dnd-kit/modifiers";
32
+ import { RenameGroupDialog } from "./RenameGroupDialog";
9
33
 
10
- // @ts-ignore
11
- import * as JsSearch from "js-search";
12
-
13
- import { Container, SearchBar } from "@firecms/ui";
14
- import { FavouritesView } from "./FavouritesView";
15
- import { useRestoreScroll } from "../../internal/useRestoreScroll";
34
+ export const DEFAULT_GROUP_NAME = "Views";
35
+ export const ADMIN_GROUP_NAME = "Admin";
16
36
 
17
- const search = new JsSearch.Search("url");
18
- search.addIndex("name");
19
- search.addIndex("description");
20
- search.addIndex("group");
21
- search.addIndex("path");
22
-
23
- /**
24
- * Default entry view for the CMS. This component renders navigation cards
25
- * for each collection defined in the navigation.
26
- * @constructor
27
- * @group Components
28
- */
29
37
  export function DefaultHomePage({
30
38
  additionalActions,
31
39
  additionalChildrenStart,
32
40
  additionalChildrenEnd
33
41
  }: {
34
- /**
35
- * Additional actions to be rendered in the home page, close to the search bar.
36
- */
37
42
  additionalActions?: React.ReactNode;
38
- /**
39
- * Additional children to be rendered in the beginning of the home page.
40
- */
41
43
  additionalChildrenStart?: React.ReactNode;
42
- /**
43
- * Additional children to be rendered at the end of the home page.
44
- */
45
44
  additionalChildrenEnd?: React.ReactNode;
46
45
  }) {
47
46
 
@@ -50,165 +49,524 @@ export function DefaultHomePage({
50
49
  const navigationController = useNavigationController();
51
50
 
52
51
  if (!navigationController.topLevelNavigation)
53
- throw Error("Navigation not ready in FireCMSHomePage");
54
-
55
- const {
56
- containerRef,
57
- scroll,
58
- direction
59
- } = useRestoreScroll();
52
+ throw Error("Navigation not ready");
60
53
 
61
54
  const {
62
- navigationEntries,
63
- groups
55
+ allowDragAndDrop,
56
+ navigationEntries: rawNavigationEntries,
57
+ groups: groupOrderFromNavController,
58
+ onNavigationEntriesUpdate
64
59
  } = navigationController.topLevelNavigation;
65
60
 
61
+ const fuse = useRef<Fuse<NavigationEntry> | null>(null);
66
62
  const [filteredUrls, setFilteredUrls] = useState<string[] | null>(null);
63
+ const performingSearch = Boolean(filteredUrls);
67
64
 
68
- const filteredNavigationEntries = filteredUrls
69
- ? navigationEntries.filter((entry) => filteredUrls.includes(entry.url))
70
- : navigationEntries;
65
+ // Memoize filtered navigation entries to prevent unnecessary recalculations
66
+ const filteredNavigationEntries = useMemo(() => {
67
+ return filteredUrls
68
+ ? rawNavigationEntries.filter((e) => filteredUrls.includes(e.url))
69
+ : rawNavigationEntries;
70
+ }, [filteredUrls, rawNavigationEntries]);
71
71
 
72
72
  useEffect(() => {
73
- search.addDocuments(navigationEntries);
74
- }, [navigationEntries]);
75
-
76
- const updateSearchResults = useCallback(
77
- (value?: string) => {
78
- if (!value || value === "") {
79
- setFilteredUrls(null);
80
- } else {
81
- const searchResult = search.search(value);
82
- setFilteredUrls(searchResult.map((e: any) => e.url));
73
+ fuse.current = new Fuse(rawNavigationEntries, {
74
+ keys: ["name", "description", "group", "path"]
75
+ });
76
+ }, [rawNavigationEntries]);
77
+
78
+ const updateSearch = useCallback((v?: string) => {
79
+ if (!v?.trim()) {
80
+ setFilteredUrls(null);
81
+ return;
82
+ }
83
+ const results = fuse.current?.search(v.trim());
84
+ setFilteredUrls(results ? results.map((x) => x.item.url) : []);
85
+ }, []);
86
+
87
+ /* ───────────────────────────────────────────────────────────────
88
+ Build groups (all items) + isolate Admin
89
+ ─────────────────────────────────────────────────────────────── */
90
+ const [items, setItems] = useState<
91
+ { name: string; entries: NavigationEntry[] }[]
92
+ >([]);
93
+ const [adminGroupData, setAdminGroupData] = useState<{
94
+ name: string;
95
+ entries: NavigationEntry[];
96
+ } | null>(null);
97
+
98
+ // Memoize the processed groups to avoid unnecessary recalculations
99
+ const processedGroups = useMemo(() => {
100
+ const src = filteredNavigationEntries;
101
+ const entriesByGroup: Record<string, NavigationEntry[]> = {};
102
+
103
+ src.forEach((e) => {
104
+ const g =
105
+ e.type === "admin"
106
+ ? ADMIN_GROUP_NAME
107
+ : e.group ?? DEFAULT_GROUP_NAME;
108
+ (entriesByGroup[g] ??= []).push(e);
109
+ });
110
+
111
+ // Check if there are custom actions from plugins that should show in the default group
112
+ const hasPluginAdditionalCards = customizationController.plugins?.some(p => p.homePage?.AdditionalCards);
113
+
114
+ let allProcessed: { name: string; entries: NavigationEntry[] }[];
115
+
116
+ if (performingSearch) {
117
+ const ordered = [
118
+ ...new Set(src.map((e) => e.group ?? DEFAULT_GROUP_NAME))
119
+ ];
120
+ allProcessed = ordered
121
+ .map((name) => ({
122
+ name,
123
+ entries: entriesByGroup[name] || []
124
+ }))
125
+ .filter((g) => g.entries.length);
126
+ } else {
127
+ allProcessed = groupOrderFromNavController.map((g) => ({
128
+ name: g,
129
+ entries: entriesByGroup[g] || []
130
+ }));
131
+ Object.keys(entriesByGroup).forEach((g) => {
132
+ if (!groupOrderFromNavController.includes(g))
133
+ allProcessed.push({
134
+ name: g,
135
+ entries: entriesByGroup[g]
136
+ });
137
+ });
138
+
139
+ // Ensure default group exists if there are plugin additional cards but no collections
140
+ if (hasPluginAdditionalCards && !allProcessed.some(g => g.name === DEFAULT_GROUP_NAME)) {
141
+ allProcessed.push({
142
+ name: DEFAULT_GROUP_NAME,
143
+ entries: []
144
+ });
83
145
  }
84
- }, []);
85
146
 
86
- const allGroups: Array<string | undefined> = [...groups];
87
- if (filteredNavigationEntries.filter(e => !e.group).length > 0 || filteredNavigationEntries.length === 0) {
88
- allGroups.push(undefined);
89
- }
147
+ allProcessed = allProcessed.filter(
148
+ (g) =>
149
+ g.entries.length ||
150
+ (g.name === DEFAULT_GROUP_NAME && hasPluginAdditionalCards)
151
+ );
152
+ }
153
+
154
+ const admin = allProcessed.find((g) => g.name === ADMIN_GROUP_NAME);
155
+ return {
156
+ adminGroupData: admin || null,
157
+ items: allProcessed.filter((g) => g.name !== ADMIN_GROUP_NAME)
158
+ };
159
+ }, [filteredNavigationEntries, performingSearch, groupOrderFromNavController, customizationController.plugins]);
160
+
161
+ // Update state only when processedGroups actually changes
162
+ useEffect(() => {
163
+ setAdminGroupData(processedGroups.adminGroupData);
164
+ setItems(processedGroups.items);
165
+ }, [processedGroups]);
166
+
167
+ /* ───────────────────────────────────────────────────────────────
168
+ Local update vs. persistence helpers
169
+ ─────────────────────────────────────────────────────────────── */
170
+ const updateItems = (
171
+ updater:
172
+ | { name: string; entries: NavigationEntry[] }[]
173
+ | ((
174
+ prev: { name: string; entries: NavigationEntry[] }[]
175
+ ) => { name: string; entries: NavigationEntry[] }[])
176
+ ) => {
177
+ setItems(updater); // local only
178
+ };
179
+
180
+ const persistNavigationGroups = (
181
+ latest: { name: string; entries: NavigationEntry[] }[]
182
+ ) => {
183
+ // Map ALL groups including "Views"
184
+ const draggable: NavigationGroupMapping[] = latest.map((g) => ({
185
+ name: g.name,
186
+ entries: g.entries.map((e) => e.path)
187
+ }));
188
+
189
+ const all: NavigationGroupMapping[] = adminGroupData
190
+ ? [
191
+ ...draggable,
192
+ {
193
+ name: adminGroupData.name,
194
+ entries: adminGroupData.entries.map((e) => e.path)
195
+ }
196
+ ]
197
+ : draggable;
198
+
199
+ onNavigationEntriesUpdate(all);
200
+ };
90
201
 
202
+ // Use custom hook for collapsed groups with localStorage persistence
203
+ const groupNames = useMemo(() => [
204
+ ...items.map(item => item.name),
205
+ ...(adminGroupData ? [adminGroupData.name] : [])
206
+ ], [items, adminGroupData]);
207
+
208
+ const { isGroupCollapsed, toggleGroupCollapsed } = useCollapsedGroups(groupNames);
209
+
210
+
211
+ const {
212
+ sensors,
213
+ collisionDetection,
214
+ onDragStart,
215
+ onDragOver,
216
+ onDragEnd,
217
+ dropAnimation,
218
+ activeItemForOverlay,
219
+ activeGroupData,
220
+ draggingGroupId,
221
+ containers,
222
+ dndKitActiveNode,
223
+ onDragCancel,
224
+ isDraggingCardOnly,
225
+ dialogOpenForGroup,
226
+ setDialogOpenForGroup,
227
+ handleRenameGroup,
228
+ handleDialogClose,
229
+ isHoveringNewGroupDropZone,
230
+ setIsHoveringNewGroupDropZone
231
+ } = useHomePageDnd({
232
+ items,
233
+ setItems: updateItems,
234
+ disabled: !allowDragAndDrop || performingSearch,
235
+ onPersist: persistNavigationGroups,
236
+ onGroupMoved: (g) =>
237
+ context.analyticsController?.onAnalyticsEvent?.("home_move_group", {
238
+ name: g
239
+ }),
240
+ onCardMovedBetweenGroups: (card) =>
241
+ context.analyticsController?.onAnalyticsEvent?.("home_move_card", {
242
+ id: card.id
243
+ }),
244
+ onNewGroupDrop: () =>
245
+ context.analyticsController?.onAnalyticsEvent?.(
246
+ "home_drop_new_group"
247
+ )
248
+ });
249
+
250
+ const {
251
+ containerRef,
252
+ direction
253
+ } = useRestoreScroll();
254
+
255
+ const dndDisabled = !allowDragAndDrop || performingSearch;
256
+
257
+ const dndModifiers = dndKitActiveNode?.data.current?.type === "group"
258
+ ? [restrictToVerticalAxis, restrictToWindowEdges]
259
+ : [restrictToWindowEdges];
260
+
261
+ /* ───────────────────────────────────────────────────────────────
262
+ Plugin extras
263
+ ─────────────────────────────────────────────────────────────── */
91
264
  let additionalPluginChildrenStart: React.ReactNode | undefined;
92
265
  let additionalPluginChildrenEnd: React.ReactNode | undefined;
93
266
  let additionalPluginSections: React.ReactNode | undefined;
267
+
94
268
  if (customizationController.plugins) {
95
- const sectionProps: PluginGenericProps = {
96
- context
97
- };
98
- additionalPluginSections = <>
99
- {customizationController.plugins.filter(plugin => plugin.homePage?.includeSection)
100
- .map((plugin, i) => {
101
- const section = plugin.homePage!.includeSection!(sectionProps)
102
- return (
103
- <NavigationGroup
104
- group={section.title}
105
- key={`plugin_section_${plugin.key}`}>
106
- {section.children}
107
- </NavigationGroup>
108
- );
109
- })}
110
- </>;
111
- additionalPluginChildrenStart = <div className={"flex flex-col gap-2"}>
112
- {customizationController.plugins.filter(plugin => plugin.homePage?.additionalChildrenStart)
113
- .map((plugin, i) => {
114
- return <div key={`plugin_children_start_${i}`}>{plugin.homePage!.additionalChildrenStart}</div>;
115
- })}
116
- </div>;
117
-
118
- additionalPluginChildrenEnd = <div className={"flex flex-col gap-2"}>
119
- {customizationController.plugins.filter(plugin => plugin.homePage?.additionalChildrenEnd)
120
- .map((plugin, i) => {
121
- return <div key={`plugin_children_start_${i}`}>{plugin.homePage!.additionalChildrenEnd}</div>;
122
- })}
123
- </div>;
269
+ const sectionProps: PluginGenericProps = { context };
270
+
271
+ additionalPluginSections = (
272
+ <>
273
+ {customizationController.plugins
274
+ .filter((p) => p.homePage?.includeSection)
275
+ .map((plugin) => {
276
+ const section = plugin.homePage!.includeSection!(
277
+ sectionProps
278
+ );
279
+ return (
280
+ <NavigationGroup
281
+ group={section.title}
282
+ key={`plugin_section_${plugin.key}`}
283
+ >
284
+ {section.children}
285
+ </NavigationGroup>
286
+ );
287
+ })}
288
+ </>
289
+ );
290
+
291
+ additionalPluginChildrenStart = (
292
+ <div className="flex flex-col gap-2">
293
+ {customizationController.plugins
294
+ .filter((p) => p.homePage?.additionalChildrenStart)
295
+ .map((plugin, i) => (
296
+ <div key={`plugin_children_start_${i}`}>
297
+ {plugin.homePage!.additionalChildrenStart}
298
+ </div>
299
+ ))}
300
+ </div>
301
+ );
302
+
303
+ additionalPluginChildrenEnd = (
304
+ <div className="flex flex-col gap-2">
305
+ {customizationController.plugins
306
+ .filter((p) => p.homePage?.additionalChildrenEnd)
307
+ .map((plugin, i) => (
308
+ <div key={`plugin_children_end_${i}`}>
309
+ {plugin.homePage!.additionalChildrenEnd}
310
+ </div>
311
+ ))}
312
+ </div>
313
+ );
124
314
  }
125
315
 
316
+ /* ───────────────────────────────────────────────────────────────
317
+ Render
318
+ ─────────────────────────────────────────────────────────────── */
126
319
  return (
127
- <div id="home_page"
128
- ref={containerRef}
129
- className="py-2 overflow-auto h-full w-full">
130
- <Container maxWidth={"6xl"}>
320
+ <div ref={containerRef} className="py-2 overflow-auto h-full w-full">
321
+ <Container maxWidth="6xl">
322
+ {/* search & actions */}
131
323
  <div
132
324
  className="w-full sticky py-4 transition-all duration-400 ease-in-out top-0 z-10 flex flex-row gap-4"
133
- style={{ top: direction === "down" ? -84 : 0 }}>
134
- <SearchBar onTextSearch={updateSearchResults}
135
- placeholder={"Search collections"}
136
- large={false}
137
- innerClassName={"w-full"}
138
- className={"w-full flex-grow"}/>
325
+ style={{ top: direction === "down" ? -84 : 0 }}
326
+ >
327
+ <SearchBar
328
+ onTextSearch={updateSearch}
329
+ placeholder="Search collections"
330
+ autoFocus
331
+ innerClassName="w-full"
332
+ className="w-full flex-grow"
333
+ />
139
334
  {additionalActions}
140
335
  </div>
141
336
 
142
- <FavouritesView hidden={Boolean(filteredUrls)}/>
337
+ <FavouritesView hidden={performingSearch}/>
143
338
 
144
339
  {additionalChildrenStart}
145
-
146
340
  {additionalPluginChildrenStart}
147
341
 
148
- {allGroups.map((group, index) => {
149
-
150
- const AdditionalCards: React.ComponentType<PluginHomePageAdditionalCardsProps>[] = [];
151
- const actionProps: PluginHomePageAdditionalCardsProps = {
152
- group,
153
- context
154
- };
155
-
156
- if (customizationController.plugins) {
157
- customizationController.plugins.forEach(plugin => {
158
- if (plugin.homePage?.AdditionalCards) {
159
- AdditionalCards.push(...toArray(plugin.homePage?.AdditionalCards));
160
- }
161
- });
162
- }
163
-
164
- const thisGroupCollections = filteredNavigationEntries
165
- .filter((entry) => entry.group === group || (!entry.group && group === undefined));
166
- if (thisGroupCollections.length === 0 && AdditionalCards.length === 0)
167
- return null;
168
- return (
169
- <NavigationGroup
170
- group={group}
171
- key={`plugin_section_${group}`}>
172
-
173
- <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
174
- {thisGroupCollections.map((entry) => (
175
- <div key={`nav_${entry.group}_${entry.name}`} className="col-span-1">
176
- <NavigationCardBinding
177
- {...entry}
178
- onClick={() => {
179
- let event: CMSAnalyticsEvent;
180
- if (entry.type === "collection") {
181
- event = "home_navigate_to_collection";
182
- } else if (entry.type === "view") {
183
- event = "home_navigate_to_view";
184
- } else if (entry.type === "admin") {
185
- event = "home_navigate_to_admin_view";
186
- } else {
187
- event = "unmapped_event";
188
- }
189
- context.analyticsController?.onAnalyticsEvent?.(event, { path: entry.path });
190
- }}
191
- />
192
- </div>
193
- ))}
194
- {AdditionalCards &&
195
- AdditionalCards.map((AdditionalCard, i) => (
196
- <div key={`nav_${group}_add_${i}`}>
197
- <AdditionalCard {...actionProps} />
198
- </div>
199
- ))}
342
+ {/* ─────── DND context ─────── */}
343
+ <DndContext
344
+ sensors={sensors}
345
+ collisionDetection={collisionDetection}
346
+ measuring={{
347
+ droppable: {
348
+ strategy: MeasuringStrategy.Always,
349
+ frequency: 500
350
+ }
351
+ }}
352
+ onDragStart={onDragStart}
353
+ onDragOver={onDragOver}
354
+ onDragEnd={onDragEnd}
355
+ onDragCancel={onDragCancel}
356
+ modifiers={dndModifiers}
357
+ >
358
+ <SortableContext
359
+ key={JSON.stringify(containers)}
360
+ items={containers}
361
+ strategy={verticalListSortingStrategy}
362
+ >
363
+ {items.map((groupData, groupIndex) => {
364
+ const groupKey = groupData.name;
365
+ const entriesInGroup = groupData.entries;
366
+
367
+ const AdditionalCards: React.ComponentType<PluginHomePageAdditionalCardsProps>[] =
368
+ [];
369
+ customizationController.plugins?.forEach((p) => {
370
+ if (p.homePage?.AdditionalCards)
371
+ AdditionalCards.push(
372
+ ...toArray(p.homePage.AdditionalCards)
373
+ );
374
+ });
375
+
376
+ const actionProps: PluginHomePageAdditionalCardsProps = {
377
+ group:
378
+ groupKey === DEFAULT_GROUP_NAME
379
+ ? undefined
380
+ : groupKey,
381
+ context
382
+ };
383
+
384
+ if (
385
+ entriesInGroup.length === 0 &&
386
+ (AdditionalCards.length === 0 || performingSearch)
387
+ )
388
+ return null;
389
+
390
+ return (
391
+ <SortableNavigationGroup
392
+ key={`group-${groupIndex}`}
393
+ groupName={groupKey}
394
+ disabled={dndDisabled}
395
+ >
396
+ <NavigationGroup
397
+ group={
398
+ groupKey === DEFAULT_GROUP_NAME
399
+ ? undefined
400
+ : groupKey
401
+ }
402
+ minimised={
403
+ draggingGroupId === groupKey &&
404
+ !isDraggingCardOnly
405
+ }
406
+ isPotentialCardDropTarget={
407
+ isDraggingCardOnly
408
+ }
409
+ dndDisabled={dndDisabled}
410
+ onEditGroup={() => {
411
+ if (dndDisabled) return;
412
+ setDialogOpenForGroup(groupKey);
413
+ }}
414
+ collapsed={isGroupCollapsed(groupKey)}
415
+ onToggleCollapsed={() => toggleGroupCollapsed(groupKey)}
416
+ >
417
+ <NavigationGroupDroppable
418
+ id={groupKey}
419
+ itemIds={entriesInGroup.map(
420
+ (e) => e.url
421
+ )}
422
+ isPotentialCardDropTarget={
423
+ isDraggingCardOnly
424
+ }
425
+ >
426
+ <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 ">
427
+ {entriesInGroup.map(
428
+ (entry) => (
429
+ <SortableNavigationCard
430
+ key={entry.url}
431
+ entry={entry}
432
+ onClick={() => {
433
+ let event: CMSAnalyticsEvent =
434
+ "unmapped_event";
435
+ if (
436
+ entry.type ===
437
+ "collection"
438
+ )
439
+ event =
440
+ "home_navigate_to_collection";
441
+ else if (
442
+ entry.type ===
443
+ "view"
444
+ )
445
+ event =
446
+ "home_navigate_to_view";
447
+ else if (
448
+ entry.type ===
449
+ "admin"
450
+ )
451
+ event =
452
+ "home_navigate_to_admin_view";
453
+
454
+ context.analyticsController?.onAnalyticsEvent?.(
455
+ event,
456
+ {
457
+ path: entry.path
458
+ }
459
+ );
460
+ }}
461
+ />
462
+ )
463
+ )}
464
+ {!performingSearch &&
465
+ groupKey.toLowerCase() !==
466
+ ADMIN_GROUP_NAME.toLowerCase() &&
467
+ AdditionalCards.map(
468
+ (C, i) => (
469
+ <C
470
+ key={`extra_${groupKey}_${i}`}
471
+ {...actionProps}
472
+ />
473
+ )
474
+ )}
475
+ </div>
476
+ </NavigationGroupDroppable>
477
+ </NavigationGroup>
478
+ </SortableNavigationGroup>
479
+ );
480
+ })}
481
+ </SortableContext>
482
+
483
+ <NewGroupDropZone
484
+ disabled={dndDisabled}
485
+ setIsHovering={setIsHoveringNewGroupDropZone}
486
+ />
487
+
488
+ <DragOverlay adjustScale={false} dropAnimation={dropAnimation}>
489
+ {activeGroupData &&
490
+ draggingGroupId === activeGroupData.name ? (
491
+ <div
492
+ className="rounded-lg bg-transparent"
493
+ style={{
494
+ padding: 0,
495
+ margin: 0
496
+ }}
497
+ >
498
+ <NavigationGroup
499
+ group={
500
+ activeGroupData.name ===
501
+ DEFAULT_GROUP_NAME
502
+ ? undefined
503
+ : activeGroupData.name
504
+ }
505
+ isPreview={false}
506
+ minimised
507
+ />
200
508
  </div>
201
- </NavigationGroup>
202
- );
203
- })}
509
+ ) : activeItemForOverlay ? (
510
+ <NavigationCardBinding
511
+ {...activeItemForOverlay}
512
+ shrink={isHoveringNewGroupDropZone}
513
+ />
514
+ ) : null}
515
+ </DragOverlay>
516
+ </DndContext>
204
517
 
205
- {additionalPluginSections}
518
+ {!performingSearch && adminGroupData && (
519
+ <NavigationGroup
520
+ group={adminGroupData.name}
521
+ collapsed={isGroupCollapsed(adminGroupData.name)}
522
+ onToggleCollapsed={() => toggleGroupCollapsed(adminGroupData.name)}
523
+ >
524
+ <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 ">
525
+ {adminGroupData.entries.map((entry) => (
526
+ <NavigationCardBinding
527
+ key={entry.url}
528
+ {...entry}
529
+ onClick={() => {
530
+ let event: CMSAnalyticsEvent =
531
+ "unmapped_event";
532
+ if (entry.type === "collection")
533
+ event =
534
+ "home_navigate_to_collection";
535
+ else if (entry.type === "view")
536
+ event = "home_navigate_to_view";
537
+ else if (entry.type === "admin")
538
+ event =
539
+ "home_navigate_to_admin_view";
206
540
 
207
- {additionalPluginChildrenEnd}
541
+ context.analyticsController?.onAnalyticsEvent?.(
542
+ event,
543
+ { path: entry.path }
544
+ );
545
+ }}
546
+ />
547
+ ))}
548
+ </div>
549
+ </NavigationGroup>
550
+ )}
208
551
 
552
+ {additionalPluginSections}
553
+ {additionalPluginChildrenEnd}
209
554
  {additionalChildrenEnd}
210
-
211
555
  </Container>
556
+
557
+ {dialogOpenForGroup && (
558
+ <RenameGroupDialog
559
+ open
560
+ initialName={dialogOpenForGroup}
561
+ existingGroupNames={items
562
+ .map((g) => g.name)
563
+ .filter((n) => n !== dialogOpenForGroup)}
564
+ onClose={handleDialogClose}
565
+ onRename={(newName) => {
566
+ handleRenameGroup(dialogOpenForGroup, newName);
567
+ }}
568
+ />
569
+ )}
212
570
  </div>
213
571
  );
214
572
  }