@rebasepro/admin 0.2.3 → 0.2.5

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 (225) hide show
  1. package/dist/{CollectionEditorDialog-CmGXXSY9.js → CollectionEditorDialog-Cn8-tGyL.js} +89 -79
  2. package/dist/CollectionEditorDialog-Cn8-tGyL.js.map +1 -0
  3. package/dist/{CollectionsStudioView-DcLHT5bU.js → CollectionsStudioView-C-Ts1rZt.js} +5 -4
  4. package/dist/{CollectionsStudioView-DcLHT5bU.js.map → CollectionsStudioView-C-Ts1rZt.js.map} +1 -1
  5. package/dist/{ExportCollectionAction-BfN34eWX.js → ExportCollectionAction-BRdKM3DF.js} +4 -3
  6. package/dist/ExportCollectionAction-BRdKM3DF.js.map +1 -0
  7. package/dist/{ImportCollectionAction-SZrInjhx.js → ImportCollectionAction-U-v7lGxO.js} +3 -2
  8. package/dist/{ImportCollectionAction-SZrInjhx.js.map → ImportCollectionAction-U-v7lGxO.js.map} +1 -1
  9. package/dist/{PropertyEditView-Cvryrb3B.js → PropertyEditView-BDNYkfNf.js} +128 -121
  10. package/dist/PropertyEditView-BDNYkfNf.js.map +1 -0
  11. package/dist/collection_editor/ConfigControllerProvider.d.ts +0 -5
  12. package/dist/collection_editor/index.d.ts +0 -1
  13. package/dist/collection_editor/types/collection_editor_controller.d.ts +0 -2
  14. package/dist/collection_editor/ui/collection_editor/CollectionPropertiesEditorForm.d.ts +3 -3
  15. package/dist/collection_editor_ui.js +3 -3
  16. package/dist/components/ArrayContainer.d.ts +2 -2
  17. package/dist/components/DefaultAppBar.d.ts +18 -1
  18. package/dist/components/DefaultDrawer.d.ts +51 -3
  19. package/dist/components/EntityCollectionTable/fields/TableStorageUpload.d.ts +2 -2
  20. package/dist/components/EntityCollectionTable/internal/EntityTableCell.d.ts +2 -2
  21. package/dist/components/EntityCollectionTable/table_bindings.d.ts +4 -3
  22. package/dist/components/EntityCollectionView/hooks/useKanbanDragAndDrop.d.ts +4 -3
  23. package/dist/components/EntityEditView.d.ts +2 -1
  24. package/dist/components/HomePage/HomePageDnD.d.ts +3 -3
  25. package/dist/components/PropertyCollectionView.d.ts +1 -1
  26. package/dist/components/PropertyIdCopyTooltip.d.ts +1 -1
  27. package/dist/components/RebaseRouteDefs.d.ts +1 -1
  28. package/dist/components/SelectableTable/SelectionStore.d.ts +4 -1
  29. package/dist/components/SelectableTable/filters/BooleanFilterField.d.ts +2 -2
  30. package/dist/components/SelectableTable/filters/DateTimeFilterField.d.ts +2 -2
  31. package/dist/components/SelectableTable/filters/ReferenceFilterField.d.ts +2 -2
  32. package/dist/components/SelectableTable/filters/StringNumberFilterField.d.ts +2 -2
  33. package/dist/components/admin/index.d.ts +1 -3
  34. package/dist/components/app/Drawer.d.ts +8 -1
  35. package/dist/data_export/export/export.d.ts +3 -3
  36. package/dist/editor/components/editor-bubble.d.ts +5 -1
  37. package/dist/editor/components/image-bubble.d.ts +5 -1
  38. package/dist/editor/components/index.d.ts +3 -3
  39. package/dist/editor/components/table-bubble.d.ts +5 -1
  40. package/dist/editor/nodeViews/ReactNodeView.d.ts +4 -1
  41. package/dist/editor/useProseMirror.d.ts +2 -2
  42. package/dist/editor/utils/remove_classes.d.ts +1 -1
  43. package/dist/editor.js +15 -14
  44. package/dist/editor.js.map +1 -1
  45. package/dist/form/EntityForm.d.ts +2 -2
  46. package/dist/form/components/StorageUploadProgress.d.ts +2 -2
  47. package/dist/form/field_bindings/MultiSelectFieldBinding.d.ts +1 -1
  48. package/dist/form/field_bindings/StorageUploadFieldBinding.d.ts +1 -1
  49. package/dist/form/validation.d.ts +3 -3
  50. package/dist/hooks/navigation/useBuildNavigationStateController.d.ts +1 -1
  51. package/dist/hooks/navigation/useResolvedCollections.d.ts +6 -0
  52. package/dist/hooks/navigation/useResolvedViews.d.ts +3 -7
  53. package/dist/{index-DjduZG1T.js → index-DHaOV-7A.js} +3 -3
  54. package/dist/index-DHaOV-7A.js.map +1 -0
  55. package/dist/{index-MKPc70-v.js → index-DJSL_SCr.js} +3 -3
  56. package/dist/index-DJSL_SCr.js.map +1 -0
  57. package/dist/{index-PLIQXpTt.js → index-XMII4H3d.js} +3 -2
  58. package/dist/{index-PLIQXpTt.js.map → index-XMII4H3d.js.map} +1 -1
  59. package/dist/index.d.ts +1 -3
  60. package/dist/index.js +2688 -452
  61. package/dist/index.js.map +1 -1
  62. package/dist/{markdown-z2Ir7Cgo.js → markdown-DD2JDU1X.js} +2 -2
  63. package/dist/markdown-DD2JDU1X.js.map +1 -0
  64. package/dist/preview/components/UrlComponentPreview.d.ts +1 -0
  65. package/dist/types/components/EntityFormActionsProps.d.ts +1 -1
  66. package/dist/types/components/EntityFormProps.d.ts +2 -2
  67. package/dist/types/fields.d.ts +1 -1
  68. package/dist/{util-DbWax_sV.js → util-0GYaJqL_.js} +1505 -2043
  69. package/dist/util-0GYaJqL_.js.map +1 -0
  70. package/package.json +8 -8
  71. package/src/collection_editor/ConfigControllerProvider.tsx +3 -13
  72. package/src/collection_editor/index.ts +1 -3
  73. package/src/collection_editor/pgColumnToProperty.ts +19 -2
  74. package/src/collection_editor/types/collection_editor_controller.tsx +0 -3
  75. package/src/collection_editor/ui/EditorCollectionAction.tsx +1 -6
  76. package/src/collection_editor/ui/EditorCollectionActionStart.tsx +1 -6
  77. package/src/collection_editor/ui/EditorEntityAction.tsx +1 -6
  78. package/src/collection_editor/ui/HomePageEditorCollectionAction.tsx +7 -14
  79. package/src/collection_editor/ui/NewCollectionCard.tsx +1 -5
  80. package/src/collection_editor/ui/PropertyAddColumnComponent.tsx +3 -8
  81. package/src/collection_editor/ui/collection_editor/CollectionJsonImportDialog.tsx +8 -12
  82. package/src/collection_editor/ui/collection_editor/CollectionPropertiesEditorForm.tsx +21 -21
  83. package/src/collection_editor/ui/collection_editor/CollectionRLSTab.tsx +4 -4
  84. package/src/collection_editor/ui/collection_editor/EnumForm.tsx +1 -1
  85. package/src/collection_editor/ui/collection_editor/properties/BlockPropertyField.tsx +3 -3
  86. package/src/collection_editor/ui/collection_editor/properties/CommonPropertyFields.tsx +3 -3
  87. package/src/collection_editor/ui/collection_editor/properties/DateTimePropertyField.tsx +8 -8
  88. package/src/collection_editor/ui/collection_editor/properties/EnumPropertyField.tsx +5 -5
  89. package/src/collection_editor/ui/collection_editor/properties/MapPropertyField.tsx +2 -2
  90. package/src/collection_editor/ui/collection_editor/properties/MarkdownPropertyField.tsx +5 -5
  91. package/src/collection_editor/ui/collection_editor/properties/NumberPropertyField.tsx +5 -5
  92. package/src/collection_editor/ui/collection_editor/properties/ReferencePropertyField.tsx +2 -2
  93. package/src/collection_editor/ui/collection_editor/properties/RepeatPropertyField.tsx +2 -2
  94. package/src/collection_editor/ui/collection_editor/properties/StoragePropertyField.tsx +8 -8
  95. package/src/collection_editor/ui/collection_editor/properties/StringPropertyField.tsx +5 -5
  96. package/src/collection_editor/ui/collection_editor/properties/UrlPropertyField.tsx +3 -2
  97. package/src/collection_editor/ui/collection_editor/properties/VectorPropertyField.tsx +2 -2
  98. package/src/collection_editor/ui/collection_editor/properties/validation/ArrayPropertyValidation.tsx +2 -2
  99. package/src/collection_editor/ui/collection_editor/properties/validation/GeneralPropertyValidation.tsx +1 -1
  100. package/src/collection_editor/ui/collection_editor/properties/validation/NumberPropertyValidation.tsx +4 -7
  101. package/src/collection_editor/ui/collection_editor/properties/validation/StringPropertyValidation.tsx +4 -4
  102. package/src/components/ArrayContainer.tsx +3 -3
  103. package/src/components/DefaultAppBar.tsx +52 -31
  104. package/src/components/DefaultDrawer.tsx +280 -67
  105. package/src/components/DrawerNavigationItem.tsx +1 -1
  106. package/src/components/EntityCollectionTable/EntityCollectionTable.tsx +6 -5
  107. package/src/components/EntityCollectionTable/PropertyTableCell.tsx +9 -7
  108. package/src/components/EntityCollectionTable/fields/TableStorageUpload.tsx +5 -5
  109. package/src/components/EntityCollectionTable/fields/VirtualTableNumberInput.tsx +12 -9
  110. package/src/components/EntityCollectionTable/internal/EntityTableCell.tsx +2 -2
  111. package/src/components/EntityCollectionTable/internal/popup_field/PopupFormField.tsx +1 -1
  112. package/src/components/EntityCollectionTable/table_bindings.tsx +5 -4
  113. package/src/components/EntityCollectionView/EntityCollectionCardView.tsx +4 -4
  114. package/src/components/EntityCollectionView/EntityCollectionListView.tsx +7 -0
  115. package/src/components/EntityCollectionView/EntityCollectionView.tsx +10 -5
  116. package/src/components/EntityCollectionView/hooks/useCollectionInlineEditor.ts +1 -1
  117. package/src/components/EntityCollectionView/hooks/useKanbanDragAndDrop.ts +7 -6
  118. package/src/components/EntityDetailView.tsx +46 -24
  119. package/src/components/EntityEditView.tsx +51 -28
  120. package/src/components/EntityEditViewFormActions.tsx +4 -4
  121. package/src/components/EntityPreview.tsx +9 -4
  122. package/src/components/HomePage/HomePageDnD.tsx +3 -2
  123. package/src/components/PropertyCollectionView.tsx +1 -1
  124. package/src/components/PropertyIdCopyTooltip.tsx +1 -1
  125. package/src/components/RebaseLayout.tsx +5 -1
  126. package/src/components/RebaseNavigation.tsx +2 -2
  127. package/src/components/RebaseRouteDefs.tsx +6 -11
  128. package/src/components/RebaseShell.tsx +16 -13
  129. package/src/components/SearchIconsView.tsx +1 -8
  130. package/src/components/SelectableTable/SelectableTable.tsx +8 -11
  131. package/src/components/SelectableTable/SelectionStore.ts +1 -1
  132. package/src/components/SelectableTable/filters/BooleanFilterField.tsx +3 -3
  133. package/src/components/SelectableTable/filters/DateTimeFilterField.tsx +3 -3
  134. package/src/components/SelectableTable/filters/ReferenceFilterField.tsx +5 -5
  135. package/src/components/SelectableTable/filters/StringNumberFilterField.tsx +3 -3
  136. package/src/components/SideEntityProvider.tsx +2 -1
  137. package/src/components/admin/index.ts +1 -3
  138. package/src/components/app/Drawer.tsx +9 -1
  139. package/src/components/app/Scaffold.tsx +5 -1
  140. package/src/components/index.ts +1 -3
  141. package/src/data_export/export/export.ts +17 -17
  142. package/src/data_import/components/DataNewPropertiesMapping.tsx +1 -1
  143. package/src/editor/components/editor-bubble.tsx +32 -9
  144. package/src/editor/components/image-bubble.tsx +27 -11
  145. package/src/editor/components/index.ts +3 -3
  146. package/src/editor/components/table-bubble.tsx +79 -17
  147. package/src/editor/extensions/HighlightDecorationExtension.ts +3 -2
  148. package/src/editor/nodeViews/ReactNodeView.tsx +1 -1
  149. package/src/editor/nodeViews/TaskItemComponent.tsx +9 -8
  150. package/src/editor/schema.ts +135 -59
  151. package/src/editor/selectors/link-selector.tsx +8 -5
  152. package/src/editor/useProseMirror.ts +2 -2
  153. package/src/editor/utils/remove_classes.ts +6 -5
  154. package/src/form/EntityForm.tsx +15 -15
  155. package/src/form/EntityFormActions.tsx +2 -2
  156. package/src/form/PropertyFieldBinding.tsx +64 -64
  157. package/src/form/components/FieldHelperText.tsx +4 -4
  158. package/src/form/components/StorageUploadProgress.tsx +2 -2
  159. package/src/form/field_bindings/ArrayCustomShapedFieldBinding.tsx +1 -1
  160. package/src/form/field_bindings/ArrayOfReferencesFieldBinding.tsx +1 -1
  161. package/src/form/field_bindings/BlockFieldBinding.tsx +54 -53
  162. package/src/form/field_bindings/KeyValueFieldBinding.tsx +290 -289
  163. package/src/form/field_bindings/MapFieldBinding.tsx +2 -2
  164. package/src/form/field_bindings/MultiSelectFieldBinding.tsx +2 -2
  165. package/src/form/field_bindings/MultipleRelationFieldBinding.tsx +1 -1
  166. package/src/form/field_bindings/ReferenceAsStringFieldBinding.tsx +1 -1
  167. package/src/form/field_bindings/ReferenceFieldBinding.tsx +8 -6
  168. package/src/form/field_bindings/RelationFieldBinding.tsx +4 -4
  169. package/src/form/field_bindings/RepeatFieldBinding.tsx +1 -1
  170. package/src/form/field_bindings/SelectFieldBinding.tsx +1 -1
  171. package/src/form/field_bindings/StorageUploadFieldBinding.tsx +84 -84
  172. package/src/form/field_bindings/SwitchFieldBinding.tsx +16 -16
  173. package/src/form/field_bindings/TextFieldBinding.tsx +77 -73
  174. package/src/form/field_bindings/UserSelectFieldBinding.tsx +17 -17
  175. package/src/form/validation.ts +43 -43
  176. package/src/hooks/navigation/useBuildNavigationStateController.tsx +4 -7
  177. package/src/hooks/navigation/useResolvedCollections.ts +27 -7
  178. package/src/hooks/navigation/useResolvedViews.tsx +8 -70
  179. package/src/index.ts +4 -3
  180. package/src/preview/PropertyPreview.tsx +2 -2
  181. package/src/preview/components/ImagePreview.tsx +2 -1
  182. package/src/preview/components/UrlComponentPreview.tsx +11 -2
  183. package/src/preview/components/UserPreview.tsx +1 -1
  184. package/src/preview/property_previews/ArrayOfMapsPreview.tsx +2 -2
  185. package/src/preview/property_previews/ArrayOfReferencesPreview.tsx +4 -4
  186. package/src/preview/property_previews/ArrayOfRelationsPreview.tsx +3 -3
  187. package/src/preview/property_previews/ArrayOfStorageComponentsPreview.tsx +3 -3
  188. package/src/preview/property_previews/ArrayOfStringsPreview.tsx +3 -2
  189. package/src/preview/property_previews/ArrayOneOfPreview.tsx +6 -8
  190. package/src/preview/property_previews/ArrayPropertyEnumPreview.tsx +1 -1
  191. package/src/preview/property_previews/ArrayPropertyPreview.tsx +3 -3
  192. package/src/preview/property_previews/MapPropertyPreview.tsx +4 -3
  193. package/src/preview/property_previews/NumberPropertyPreview.tsx +5 -3
  194. package/src/preview/property_previews/StringPropertyPreview.tsx +10 -8
  195. package/src/types/components/EntityFormActionsProps.tsx +1 -1
  196. package/src/types/components/EntityFormProps.tsx +2 -2
  197. package/src/types/fields.tsx +2 -2
  198. package/src/util/previews.ts +9 -1
  199. package/dist/CollectionEditorDialog-CmGXXSY9.js.map +0 -1
  200. package/dist/ContentHomePage-C7vFqKSe.js +0 -1784
  201. package/dist/ContentHomePage-C7vFqKSe.js.map +0 -1
  202. package/dist/ExportCollectionAction-BfN34eWX.js.map +0 -1
  203. package/dist/PropertyEditView-Cvryrb3B.js.map +0 -1
  204. package/dist/RoleChip-QtUFXeTp.js +0 -67
  205. package/dist/RoleChip-QtUFXeTp.js.map +0 -1
  206. package/dist/RolesView-BCb7qwWs.js +0 -437
  207. package/dist/RolesView-BCb7qwWs.js.map +0 -1
  208. package/dist/UsersView-Cex24r8H.js +0 -408
  209. package/dist/UsersView-Cex24r8H.js.map +0 -1
  210. package/dist/collection_editor/types/config_permissions.d.ts +0 -19
  211. package/dist/components/admin/RoleChip.d.ts +0 -4
  212. package/dist/components/admin/RolesFilterSelect.d.ts +0 -2
  213. package/dist/components/admin/RolesView.d.ts +0 -4
  214. package/dist/components/admin/UserRolesSelectField.d.ts +0 -2
  215. package/dist/components/admin/UsersView.d.ts +0 -4
  216. package/dist/index-DjduZG1T.js.map +0 -1
  217. package/dist/index-MKPc70-v.js.map +0 -1
  218. package/dist/markdown-z2Ir7Cgo.js.map +0 -1
  219. package/dist/util-DbWax_sV.js.map +0 -1
  220. package/src/collection_editor/types/config_permissions.ts +0 -20
  221. package/src/components/admin/RoleChip.tsx +0 -23
  222. package/src/components/admin/RolesFilterSelect.tsx +0 -45
  223. package/src/components/admin/RolesView.tsx +0 -465
  224. package/src/components/admin/UserRolesSelectField.tsx +0 -50
  225. package/src/components/admin/UsersView.tsx +0 -687
@@ -1,408 +0,0 @@
1
- import { jsxs, jsx, Fragment } from "react/jsx-runtime";
2
- import React__default, { useState, useRef, useCallback, useEffect } from "react";
3
- import { useSnackbarController, useAuthController, useTranslation, BootstrapAdminBanner, ConfirmationDialog } from "@rebasepro/core";
4
- import "react-compiler-runtime";
5
- import "fast-equals";
6
- import { Container, Typography, Select, SelectItem, SearchBar, Button, PlusIcon, Table, TableHeader, TableCell, TableBody, TableRow, Skeleton, Tooltip, IconButton, KeyRoundIcon, iconSize, Trash2Icon, CenteredView, ChevronLeftIcon, ChevronRightIcon, Dialog, DialogTitle, DialogContent, TextField, MultiSelect, MultiSelectItem, DialogActions, LoadingButton } from "@rebasepro/ui";
7
- import "@rebasepro/common";
8
- import "@rebasepro/formex";
9
- import "zod";
10
- import "@rebasepro/utils";
11
- import { b9 as useBreadcrumbsController, m as CreationResultDialog } from "./util-DbWax_sV.js";
12
- import "date-fns";
13
- import "date-fns/locale";
14
- import "@rebasepro/types";
15
- import "@dnd-kit/core";
16
- import "@dnd-kit/modifiers";
17
- import "@dnd-kit/sortable";
18
- import "@dnd-kit/utilities";
19
- import "react-dropzone";
20
- import "react-router-dom";
21
- import "exceljs";
22
- import "@rebasepro/schema-inference";
23
- import { R as RoleChip } from "./RoleChip-QtUFXeTp.js";
24
- const PAGE_SIZE = 25;
25
- function UsersView({
26
- userManagement
27
- }) {
28
- const {
29
- roles,
30
- saveUser,
31
- createUser,
32
- deleteUser,
33
- resetPassword,
34
- loading: delegateLoading,
35
- bootstrapAdmin,
36
- usersError
37
- } = userManagement;
38
- const snackbarController = useSnackbarController();
39
- const {
40
- user: loggedInUser
41
- } = useAuthController();
42
- const {
43
- t
44
- } = useTranslation();
45
- const breadcrumbs = useBreadcrumbsController();
46
- React__default.useEffect(() => {
47
- breadcrumbs.set({
48
- breadcrumbs: [{
49
- title: t("users"),
50
- url: "/users"
51
- }]
52
- });
53
- }, []);
54
- const [dialogOpen, setDialogOpen] = useState(false);
55
- const [selectedUser, setSelectedUser] = useState();
56
- const [deleteConfirmOpen, setDeleteConfirmOpen] = useState(false);
57
- const [userToDelete, setUserToDelete] = useState();
58
- const [deleteInProgress, setDeleteInProgress] = useState(false);
59
- const [formKey, setFormKey] = useState(0);
60
- const [bootstrapping, setBootstrapping] = useState(false);
61
- const [creationResult, setCreationResult] = useState(null);
62
- const [resetConfirmOpen, setResetConfirmOpen] = useState(false);
63
- const [userToReset, setUserToReset] = useState();
64
- const [resetInProgress, setResetInProgress] = useState(false);
65
- const hasServerSearch = !!userManagement.searchUsers;
66
- const [searchQuery, setSearchQuery] = useState("");
67
- const [roleFilter, setRoleFilter] = useState("");
68
- const [page, setPage] = useState(0);
69
- const [paginatedUsers, setPaginatedUsers] = useState([]);
70
- const [totalUsers, setTotalUsers] = useState(0);
71
- const [tableLoading, setTableLoading] = useState(hasServerSearch);
72
- const searchTimerRef = useRef(null);
73
- const allUsers = userManagement.users;
74
- const fetchPage = useCallback(async (pageNum, search, filterRole, forceLoading = false) => {
75
- if (!userManagement.searchUsers) return;
76
- if (forceLoading) {
77
- setTableLoading(true);
78
- }
79
- try {
80
- const result = await userManagement.searchUsers({
81
- search: search || void 0,
82
- roleId: filterRole || void 0,
83
- limit: PAGE_SIZE,
84
- offset: pageNum * PAGE_SIZE,
85
- orderBy: "createdAt",
86
- orderDir: "desc"
87
- });
88
- setPaginatedUsers(result.users);
89
- setTotalUsers(result.total);
90
- } catch (error) {
91
- console.error("Failed to fetch users:", error);
92
- snackbarController.open({
93
- type: "error",
94
- message: error instanceof Error ? error.message : "Failed to load users"
95
- });
96
- } finally {
97
- setTableLoading(false);
98
- }
99
- }, [userManagement.searchUsers, snackbarController]);
100
- const fetchPageRef = useRef(fetchPage);
101
- fetchPageRef.current = fetchPage;
102
- const initialFetchDone = useRef(false);
103
- useEffect(() => {
104
- if (!delegateLoading && !usersError && hasServerSearch && !initialFetchDone.current) {
105
- initialFetchDone.current = true;
106
- fetchPageRef.current(0, "", roleFilter, true);
107
- }
108
- }, [delegateLoading, usersError, hasServerSearch, roleFilter]);
109
- const handleSearch = useCallback((value) => {
110
- setSearchQuery(value);
111
- setPage(0);
112
- if (searchTimerRef.current) {
113
- clearTimeout(searchTimerRef.current);
114
- }
115
- if (hasServerSearch) {
116
- searchTimerRef.current = setTimeout(() => {
117
- fetchPage(0, value, roleFilter, true);
118
- }, 300);
119
- }
120
- }, [hasServerSearch, fetchPage, roleFilter]);
121
- const handleRoleFilterChange = useCallback((newRole) => {
122
- setRoleFilter(newRole);
123
- setPage(0);
124
- if (hasServerSearch) {
125
- fetchPage(0, searchQuery, newRole, true);
126
- }
127
- }, [hasServerSearch, fetchPage, searchQuery]);
128
- const handlePageChange = useCallback((newPage) => {
129
- setPage(newPage);
130
- if (hasServerSearch) {
131
- fetchPage(newPage, searchQuery, roleFilter, true);
132
- }
133
- }, [hasServerSearch, fetchPage, searchQuery, roleFilter]);
134
- const refreshCurrentPage = useCallback(() => {
135
- if (hasServerSearch) {
136
- fetchPage(page, searchQuery, roleFilter);
137
- }
138
- }, [hasServerSearch, fetchPage, page, searchQuery, roleFilter]);
139
- let displayUsers;
140
- let displayTotal;
141
- if (hasServerSearch) {
142
- displayUsers = paginatedUsers;
143
- displayTotal = totalUsers;
144
- } else {
145
- const filtered = allUsers.filter((u) => {
146
- let matches = true;
147
- if (searchQuery) {
148
- const q = searchQuery.toLowerCase();
149
- matches = !!(u.email?.toLowerCase().includes(q) || u.displayName?.toLowerCase().includes(q));
150
- }
151
- if (matches && roleFilter) {
152
- matches = !!u.roles?.includes(roleFilter);
153
- }
154
- return matches;
155
- });
156
- displayTotal = filtered.length;
157
- displayUsers = filtered.slice(page * PAGE_SIZE, (page + 1) * PAGE_SIZE);
158
- }
159
- const totalPages = Math.max(1, Math.ceil(displayTotal / PAGE_SIZE));
160
- allUsers.some((u_0) => u_0.roles?.includes("admin"));
161
- const handleAddUser = () => {
162
- setSelectedUser(void 0);
163
- setFormKey((k) => k + 1);
164
- setDialogOpen(true);
165
- };
166
- const handleEditUser = (user) => {
167
- setSelectedUser(user);
168
- setDialogOpen(true);
169
- };
170
- const handleClose = () => {
171
- setDialogOpen(false);
172
- setSelectedUser(void 0);
173
- };
174
- const handleDelete = async () => {
175
- if (!userToDelete || !deleteUser) return;
176
- setDeleteInProgress(true);
177
- try {
178
- await deleteUser(userToDelete);
179
- snackbarController.open({
180
- type: "success",
181
- message: t("user_deleted_successfully")
182
- });
183
- setDeleteConfirmOpen(false);
184
- setUserToDelete(void 0);
185
- refreshCurrentPage();
186
- } catch (error_1) {
187
- snackbarController.open({
188
- type: "error",
189
- message: error_1 instanceof Error ? error_1.message : t("error_deleting_user")
190
- });
191
- } finally {
192
- setDeleteInProgress(false);
193
- }
194
- };
195
- const handleResetPassword = async () => {
196
- if (!userToReset || !resetPassword) return;
197
- setResetInProgress(true);
198
- try {
199
- const result_0 = await resetPassword(userToReset);
200
- setResetConfirmOpen(false);
201
- setUserToReset(void 0);
202
- setCreationResult(result_0);
203
- snackbarController.open({
204
- type: "success",
205
- message: t("reset_password_success")
206
- });
207
- } catch (error_2) {
208
- snackbarController.open({
209
- type: "error",
210
- message: error_2 instanceof Error ? error_2.message : t("error_resetting_password")
211
- });
212
- } finally {
213
- setResetInProgress(false);
214
- }
215
- };
216
- return /* @__PURE__ */ jsxs(Container, { className: "w-full flex flex-col py-4 gap-4", maxWidth: "6xl", children: [
217
- /* @__PURE__ */ jsx(BootstrapAdminBanner, { className: "mb-4" }),
218
- /* @__PURE__ */ jsxs("div", { className: "flex items-center mt-12 mb-4 gap-4", children: [
219
- /* @__PURE__ */ jsx(Typography, { gutterBottom: true, variant: "h4", className: "grow mb-0", component: "h4", children: t("users") }),
220
- roles && roles.length > 0 && /* @__PURE__ */ jsxs(Select, { value: roleFilter || "__all__", onValueChange: (v) => handleRoleFilterChange(v === "__all__" ? "" : v), placeholder: t("all_roles") || "All Roles", size: "small", className: "w-48", children: [
221
- /* @__PURE__ */ jsx(SelectItem, { value: "__all__", children: t("all_roles") || "All Roles" }),
222
- roles.map((role) => /* @__PURE__ */ jsx(SelectItem, { value: role.id, children: role.name }, role.id))
223
- ] }),
224
- /* @__PURE__ */ jsx(SearchBar, { placeholder: t("search_users"), onTextSearch: (v_0) => handleSearch(v_0 || ""), size: "small", expandable: true }),
225
- /* @__PURE__ */ jsx(Button, { startIcon: /* @__PURE__ */ jsx(PlusIcon, {}), onClick: handleAddUser, disabled: !saveUser, children: t("add_user") })
226
- ] }),
227
- /* @__PURE__ */ jsx("div", { className: "overflow-auto", children: /* @__PURE__ */ jsxs(Table, { className: "w-full", children: [
228
- /* @__PURE__ */ jsxs(TableHeader, { children: [
229
- /* @__PURE__ */ jsx(TableCell, { header: true, className: "w-48", children: t("id") || "ID" }),
230
- /* @__PURE__ */ jsx(TableCell, { header: true, children: t("email") }),
231
- /* @__PURE__ */ jsx(TableCell, { header: true, children: t("name") }),
232
- /* @__PURE__ */ jsx(TableCell, { header: true, children: t("roles") }),
233
- /* @__PURE__ */ jsx(TableCell, { header: true, className: "whitespace-nowrap", children: t("created") }),
234
- /* @__PURE__ */ jsx(TableCell, { header: true, className: "w-24 text-right", children: t("actions") })
235
- ] }),
236
- /* @__PURE__ */ jsxs(TableBody, { children: [
237
- tableLoading || delegateLoading ? [{
238
- email: "w-48",
239
- name: "w-32",
240
- roles: ["w-16", "w-20"]
241
- }, {
242
- email: "w-32",
243
- name: "w-24",
244
- roles: ["w-24"]
245
- }, {
246
- email: "w-40",
247
- name: "w-36",
248
- roles: ["w-16", "w-16"]
249
- }].map((row, i) => /* @__PURE__ */ jsxs(TableRow, { children: [
250
- /* @__PURE__ */ jsx(TableCell, { className: "font-mono text-xs", children: /* @__PURE__ */ jsx(Skeleton, { className: "h-3 w-40" }) }),
251
- /* @__PURE__ */ jsx(TableCell, { children: /* @__PURE__ */ jsx(Skeleton, { className: `h-4 ${row.email}` }) }),
252
- /* @__PURE__ */ jsx(TableCell, { className: "font-medium", children: /* @__PURE__ */ jsx(Skeleton, { className: `h-4 ${row.name}` }) }),
253
- /* @__PURE__ */ jsx(TableCell, { children: /* @__PURE__ */ jsx("div", { className: "flex flex-wrap gap-2", children: row.roles.map((w, j) => /* @__PURE__ */ jsx(Skeleton, { className: `h-6 ${w} rounded-full` }, j)) }) }),
254
- /* @__PURE__ */ jsx(TableCell, { className: "whitespace-nowrap text-sm", children: /* @__PURE__ */ jsx(Skeleton, { className: "h-4 w-20" }) }),
255
- /* @__PURE__ */ jsx(TableCell, { className: "text-right whitespace-nowrap", children: /* @__PURE__ */ jsxs("div", { className: "flex justify-end items-center gap-1", children: [
256
- /* @__PURE__ */ jsx(Skeleton, { className: "h-7 w-7 rounded-md" }),
257
- /* @__PURE__ */ jsx(Skeleton, { className: "h-7 w-7 rounded-md" })
258
- ] }) })
259
- ] }, `skeleton-${i}`)) : displayUsers.map((user_0) => /* @__PURE__ */ jsxs(TableRow, { onClick: () => saveUser && handleEditUser(user_0), children: [
260
- /* @__PURE__ */ jsx(TableCell, { className: "font-mono text-xs", children: user_0.uid }),
261
- /* @__PURE__ */ jsx(TableCell, { children: user_0.email }),
262
- /* @__PURE__ */ jsx(TableCell, { className: "font-medium", children: user_0.displayName }),
263
- /* @__PURE__ */ jsx(TableCell, { children: /* @__PURE__ */ jsx("div", { className: "flex flex-wrap gap-2", children: user_0.roles?.map((roleId) => {
264
- const role_0 = roles?.find((r) => r.id === roleId);
265
- return role_0 ? /* @__PURE__ */ jsx(RoleChip, { role: role_0 }, roleId) : /* @__PURE__ */ jsx("span", { children: roleId }, roleId);
266
- }) }) }),
267
- /* @__PURE__ */ jsx(TableCell, { className: "whitespace-nowrap text-sm text-surface-accent-600 dark:text-surface-accent-400", children: user_0.createdAt ? new Date(user_0.createdAt).toLocaleDateString() : "-" }),
268
- /* @__PURE__ */ jsx(TableCell, { className: "text-right whitespace-nowrap", children: /* @__PURE__ */ jsxs("div", { className: "flex justify-end items-center gap-1", children: [
269
- resetPassword && /* @__PURE__ */ jsx(Tooltip, { asChild: true, title: t("reset_password"), children: /* @__PURE__ */ jsx(IconButton, { size: "small", onClick: (e) => {
270
- e.stopPropagation();
271
- setUserToReset(user_0);
272
- setResetConfirmOpen(true);
273
- }, children: /* @__PURE__ */ jsx(KeyRoundIcon, { size: iconSize.small }) }) }),
274
- deleteUser && /* @__PURE__ */ jsx(Tooltip, { asChild: true, title: t("delete_this_user"), children: /* @__PURE__ */ jsx(IconButton, { size: "small", onClick: (e_0) => {
275
- e_0.stopPropagation();
276
- setUserToDelete(user_0);
277
- setDeleteConfirmOpen(true);
278
- }, children: /* @__PURE__ */ jsx(Trash2Icon, { size: iconSize.small }) }) })
279
- ] }) })
280
- ] }, user_0.uid)),
281
- displayUsers.length === 0 && !tableLoading && !delegateLoading && /* @__PURE__ */ jsx(TableRow, { children: /* @__PURE__ */ jsx(TableCell, { colspan: 6, children: /* @__PURE__ */ jsxs(CenteredView, { className: "flex flex-col gap-4 my-8 items-center", children: [
282
- /* @__PURE__ */ jsx(Typography, { variant: "label", children: usersError ? t("no_permission_to_view_users") : searchQuery ? t("no_users_found") : t("no_users_yet") }),
283
- usersError && /* @__PURE__ */ jsx(Typography, { variant: "caption", color: "secondary", children: t("no_permission_description") })
284
- ] }) }) })
285
- ] })
286
- ] }) }),
287
- displayTotal > PAGE_SIZE && /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between px-2 py-3", children: [
288
- /* @__PURE__ */ jsx(Typography, { variant: "body2", className: "text-surface-accent-500 dark:text-surface-accent-400", children: `${page * PAGE_SIZE + 1}–${Math.min((page + 1) * PAGE_SIZE, displayTotal)} / ${displayTotal}` }),
289
- /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-1", children: [
290
- /* @__PURE__ */ jsx(IconButton, { size: "small", disabled: page === 0, onClick: () => handlePageChange(page - 1), children: /* @__PURE__ */ jsx(ChevronLeftIcon, { size: iconSize.smallest }) }),
291
- /* @__PURE__ */ jsxs(Typography, { variant: "body2", className: "px-3 text-surface-accent-600 dark:text-surface-accent-300", children: [
292
- page + 1,
293
- " / ",
294
- totalPages
295
- ] }),
296
- /* @__PURE__ */ jsx(IconButton, { size: "small", disabled: page >= totalPages - 1, onClick: () => handlePageChange(page + 1), children: /* @__PURE__ */ jsx(ChevronRightIcon, { size: iconSize.smallest }) })
297
- ] })
298
- ] }),
299
- saveUser && /* @__PURE__ */ jsx(UserDetailsForm, { open: dialogOpen, user: selectedUser, roles, saveUser, createUser, handleClose, onCreationResult: setCreationResult, onSaved: refreshCurrentPage }, selectedUser?.uid ?? `new-${formKey}`),
300
- creationResult && /* @__PURE__ */ jsx(CreationResultDialog, { result: creationResult, onClose: () => setCreationResult(null) }),
301
- /* @__PURE__ */ jsx(ConfirmationDialog, { open: deleteConfirmOpen, loading: deleteInProgress, onAccept: handleDelete, onCancel: () => {
302
- setDeleteConfirmOpen(false);
303
- setUserToDelete(void 0);
304
- }, title: /* @__PURE__ */ jsx(Fragment, { children: t("delete_confirmation_title") }), body: /* @__PURE__ */ jsx(Fragment, { children: t("delete_user_confirmation") }) }),
305
- /* @__PURE__ */ jsx(ConfirmationDialog, { open: resetConfirmOpen, loading: resetInProgress, onAccept: handleResetPassword, onCancel: () => {
306
- setResetConfirmOpen(false);
307
- setUserToReset(void 0);
308
- }, title: /* @__PURE__ */ jsx(Fragment, { children: t("reset_password") }), body: /* @__PURE__ */ jsx(Fragment, { children: t("reset_password_confirmation") }) })
309
- ] });
310
- }
311
- function UserDetailsForm({
312
- open,
313
- user: userProp,
314
- roles,
315
- saveUser,
316
- createUser,
317
- handleClose,
318
- onCreationResult,
319
- onSaved
320
- }) {
321
- const snackbarController = useSnackbarController();
322
- const {
323
- t
324
- } = useTranslation();
325
- const isNewUser = !userProp;
326
- const [displayName, setDisplayName] = useState(userProp?.displayName || "");
327
- const [email, setEmail] = useState(userProp?.email || "");
328
- const [selectedRoleIds, setSelectedRoleIds] = useState(userProp?.roles || []);
329
- const [isSubmitting, setIsSubmitting] = useState(false);
330
- const [errors, setErrors] = useState({});
331
- const [submitCount, setSubmitCount] = useState(0);
332
- const validate = () => {
333
- const newErrors = {};
334
- if (!displayName) newErrors.displayName = "Required";
335
- if (!email) newErrors.email = "Required";
336
- else if (!/\S+@\S+\.\S+/.test(email)) newErrors.email = "Invalid email";
337
- setErrors(newErrors);
338
- return Object.keys(newErrors).length === 0;
339
- };
340
- const handleSubmit = async (e) => {
341
- e.preventDefault();
342
- setSubmitCount((c) => c + 1);
343
- if (!validate()) return;
344
- setIsSubmitting(true);
345
- try {
346
- const userRoles = selectedRoleIds;
347
- const userToSave = {
348
- uid: userProp?.uid || crypto.randomUUID(),
349
- email,
350
- displayName: displayName || null,
351
- photoURL: userProp?.photoURL || null,
352
- providerId: userProp?.providerId || "custom",
353
- isAnonymous: userProp?.isAnonymous || false,
354
- roles: userRoles
355
- };
356
- if (isNewUser && createUser && onCreationResult) {
357
- const result = await createUser(userToSave);
358
- handleClose();
359
- onCreationResult(result);
360
- } else {
361
- await saveUser(userToSave);
362
- handleClose();
363
- }
364
- onSaved?.();
365
- } catch (error) {
366
- snackbarController.open({
367
- type: "error",
368
- message: error instanceof Error ? error.message : "Failed to save user"
369
- });
370
- } finally {
371
- setIsSubmitting(false);
372
- }
373
- };
374
- const dirty = isNewUser || displayName !== (userProp?.displayName || "") || email !== (userProp?.email || "") || (() => {
375
- const prev = userProp?.roles || [];
376
- if (selectedRoleIds.length !== prev.length) return true;
377
- const set = new Set(prev);
378
- return selectedRoleIds.some((id) => !set.has(id));
379
- })();
380
- return /* @__PURE__ */ jsx(Dialog, { open, onOpenChange: (open_0) => !open_0 ? handleClose() : void 0, maxWidth: "4xl", children: /* @__PURE__ */ jsxs("form", { onSubmit: handleSubmit, autoComplete: "off", noValidate: true, style: {
381
- display: "flex",
382
- flexDirection: "column",
383
- position: "relative",
384
- height: "100%"
385
- }, children: [
386
- /* @__PURE__ */ jsx(DialogTitle, { variant: "h4", gutterBottom: false, children: t("user") }),
387
- /* @__PURE__ */ jsx(DialogContent, { className: "h-full grow", children: /* @__PURE__ */ jsxs("div", { className: "grid grid-cols-12 gap-4", children: [
388
- !isNewUser && /* @__PURE__ */ jsx("div", { className: "col-span-12", children: /* @__PURE__ */ jsx(TextField, { name: "uid", value: userProp?.uid || "", label: t("id") || "ID", disabled: true }) }),
389
- /* @__PURE__ */ jsxs("div", { className: "col-span-12", children: [
390
- /* @__PURE__ */ jsx(TextField, { name: "displayName", required: true, error: submitCount > 0 && Boolean(errors.displayName), value: displayName, onChange: (e_0) => setDisplayName(e_0.target.value), label: t("name") }),
391
- submitCount > 0 && errors.displayName && /* @__PURE__ */ jsx(Typography, { variant: "caption", color: "error", children: errors.displayName })
392
- ] }),
393
- /* @__PURE__ */ jsxs("div", { className: "col-span-12", children: [
394
- /* @__PURE__ */ jsx(TextField, { required: true, error: submitCount > 0 && Boolean(errors.email), name: "email", value: email, onChange: (e_1) => setEmail(e_1.target.value), label: t("email"), disabled: !isNewUser }),
395
- submitCount > 0 && errors.email && /* @__PURE__ */ jsx(Typography, { variant: "caption", color: "error", children: errors.email })
396
- ] }),
397
- roles && roles.length > 0 && /* @__PURE__ */ jsx("div", { className: "col-span-12", children: /* @__PURE__ */ jsx(MultiSelect, { className: "w-full", label: t("roles"), value: selectedRoleIds, onValueChange: (value) => setSelectedRoleIds(value), children: roles.map((role) => /* @__PURE__ */ jsx(MultiSelectItem, { value: role.id, children: /* @__PURE__ */ jsx(RoleChip, { role }) }, role.id)) }) })
398
- ] }) }),
399
- /* @__PURE__ */ jsxs(DialogActions, { children: [
400
- /* @__PURE__ */ jsx(Button, { variant: "text", onClick: handleClose, children: t("cancel") }),
401
- /* @__PURE__ */ jsx(LoadingButton, { variant: "filled", type: "submit", disabled: !dirty, loading: isSubmitting, children: isNewUser ? t("create_user") : t("update") })
402
- ] })
403
- ] }) });
404
- }
405
- export {
406
- UsersView
407
- };
408
- //# sourceMappingURL=UsersView-Cex24r8H.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"UsersView-Cex24r8H.js","sources":["../src/components/admin/UsersView.tsx"],"sourcesContent":["\nimport React, { useState, useEffect, useCallback, useRef } from \"react\";\nimport { User } from \"@rebasepro/types\";\nimport { useSnackbarController, useAuthController, useTranslation } from \"@rebasepro/core\";\nimport { useBreadcrumbsController } from \"../../index\";\nimport {\n Alert,\n Button,\n CenteredView,\n CheckCircleIcon,\n ChevronLeftIcon,\n ChevronRightIcon,\n CircularProgress,\n Container,\n CopyIcon,\n Dialog,\n DialogActions,\n DialogContent,\n DialogTitle,\n IconButton,\n iconSize,\n KeyRoundIcon,\n LoadingButton,\n MailIcon,\n MultiSelect,\n MultiSelectItem,\n PlusIcon,\n SearchBar,\n Select,\n SelectItem,\n Skeleton,\n Table,\n TableBody,\n TableCell,\n TableHeader,\n TableRow,\n TextField,\n Tooltip,\n Trash2Icon,\n Typography\n} from \"@rebasepro/ui\";\nimport { RoleChip } from \"./RoleChip\";\nimport { UserManagementDelegate, Role, UserCreationResult } from \"@rebasepro/types\";\nimport { ConfirmationDialog, BootstrapAdminBanner } from \"@rebasepro/core\";\nimport { CreationResultDialog } from \"./CreationResultDialog\";\n\n\nconst PAGE_SIZE = 25;\n\n// ============================================\n// UsersView Component\n// ============================================\nexport function UsersView({ userManagement }: {\n userManagement: UserManagementDelegate;\n}) {\n const { roles, saveUser, createUser, deleteUser, resetPassword, loading: delegateLoading, bootstrapAdmin, usersError } = userManagement;\n const snackbarController = useSnackbarController();\n const { user: loggedInUser } = useAuthController();\n const { t } = useTranslation();\n const breadcrumbs = useBreadcrumbsController();\n\n React.useEffect(() => {\n breadcrumbs.set({\n breadcrumbs: [{ title: t(\"users\"),\nurl: \"/users\" }]\n });\n\n }, []);\n\n const [dialogOpen, setDialogOpen] = useState(false);\n const [selectedUser, setSelectedUser] = useState<User | undefined>();\n const [deleteConfirmOpen, setDeleteConfirmOpen] = useState(false);\n const [userToDelete, setUserToDelete] = useState<User | undefined>();\n const [deleteInProgress, setDeleteInProgress] = useState(false);\n const [formKey, setFormKey] = useState(0);\n const [bootstrapping, setBootstrapping] = useState(false);\n\n // Creation result state\n const [creationResult, setCreationResult] = useState<UserCreationResult | null>(null);\n\n // Reset password\n const [resetConfirmOpen, setResetConfirmOpen] = useState(false);\n const [userToReset, setUserToReset] = useState<User | undefined>();\n const [resetInProgress, setResetInProgress] = useState(false);\n\n // Check if server-side search is available\n const hasServerSearch = !!userManagement.searchUsers;\n\n // ---- Server-side pagination state ----\n const [searchQuery, setSearchQuery] = useState(\"\");\n const [roleFilter, setRoleFilter] = useState<string>(\"\");\n const [page, setPage] = useState(0);\n const [paginatedUsers, setPaginatedUsers] = useState<User[]>([]);\n const [totalUsers, setTotalUsers] = useState(0);\n const [tableLoading, setTableLoading] = useState(hasServerSearch);\n\n // Debounce timer ref for search\n const searchTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n // Fallback: use in-memory users if no searchUsers\n const allUsers = userManagement.users;\n\n /**\n * Fetch a page of users from the server.\n * Only shows the loading skeleton when we have no data yet (initial load).\n * Subsequent re-fetches (pagination, search) update in-place without flashing.\n */\n const fetchPage = useCallback(async (pageNum: number, search: string, filterRole: string, forceLoading = false) => {\n if (!userManagement.searchUsers) return;\n\n // Only show skeleton on initial load or explicit requests (search/filter/page change).\n // This avoids flashing skeletons when the effect re-fires from dep changes.\n if (forceLoading) {\n setTableLoading(true);\n }\n try {\n const result = await userManagement.searchUsers({\n search: search || undefined,\n roleId: filterRole || undefined,\n limit: PAGE_SIZE,\n offset: pageNum * PAGE_SIZE,\n orderBy: \"createdAt\",\n orderDir: \"desc\"\n });\n setPaginatedUsers(result.users);\n setTotalUsers(result.total);\n } catch (error: unknown) {\n console.error(\"Failed to fetch users:\", error);\n snackbarController.open({ type: \"error\",\nmessage: error instanceof Error ? error.message : \"Failed to load users\" });\n } finally {\n setTableLoading(false);\n }\n }, [userManagement.searchUsers, snackbarController]);\n\n // Stable ref for fetchPage so the initial-load effect doesn't re-fire\n // every time fetchPage's reference changes (which happens on parent re-renders).\n const fetchPageRef = useRef(fetchPage);\n fetchPageRef.current = fetchPage;\n const initialFetchDone = useRef(false);\n\n // Load initial page when delegate finishes loading — runs exactly once.\n useEffect(() => {\n if (!delegateLoading && !usersError && hasServerSearch && !initialFetchDone.current) {\n initialFetchDone.current = true;\n fetchPageRef.current(0, \"\", roleFilter, true);\n }\n }, [delegateLoading, usersError, hasServerSearch, roleFilter]);\n\n // Handle search changes (debounced)\n const handleSearch = useCallback((value: string) => {\n setSearchQuery(value);\n setPage(0);\n\n if (searchTimerRef.current) {\n clearTimeout(searchTimerRef.current);\n }\n\n if (hasServerSearch) {\n searchTimerRef.current = setTimeout(() => {\n fetchPage(0, value, roleFilter, true);\n }, 300);\n }\n }, [hasServerSearch, fetchPage, roleFilter]);\n\n const handleRoleFilterChange = useCallback((newRole: string) => {\n setRoleFilter(newRole);\n setPage(0);\n if (hasServerSearch) {\n fetchPage(0, searchQuery, newRole, true);\n }\n }, [hasServerSearch, fetchPage, searchQuery]);\n\n // Handle page change\n const handlePageChange = useCallback((newPage: number) => {\n setPage(newPage);\n if (hasServerSearch) {\n fetchPage(newPage, searchQuery, roleFilter, true);\n }\n }, [hasServerSearch, fetchPage, searchQuery, roleFilter]);\n\n // Refresh current page (after create/update/delete)\n const refreshCurrentPage = useCallback(() => {\n if (hasServerSearch) {\n fetchPage(page, searchQuery, roleFilter);\n }\n }, [hasServerSearch, fetchPage, page, searchQuery, roleFilter]);\n\n // Determine which users to show\n let displayUsers: User[];\n let displayTotal: number;\n\n if (hasServerSearch) {\n displayUsers = paginatedUsers;\n displayTotal = totalUsers;\n } else {\n // Fallback: local filtering for backward compat\n const filtered = allUsers.filter(u => {\n let matches = true;\n if (searchQuery) {\n const q = searchQuery.toLowerCase();\n matches = !!(u.email?.toLowerCase().includes(q) || u.displayName?.toLowerCase().includes(q));\n }\n if (matches && roleFilter) {\n matches = !!u.roles?.includes(roleFilter);\n }\n return matches;\n });\n displayTotal = filtered.length;\n displayUsers = filtered.slice(page * PAGE_SIZE, (page + 1) * PAGE_SIZE);\n }\n\n const totalPages = Math.max(1, Math.ceil(displayTotal / PAGE_SIZE));\n\n // Check if any admin exists\n const hasAdmin = allUsers.some(u => u.roles?.includes(\"admin\"));\n\n const handleBootstrap = async () => {\n if (!bootstrapAdmin) return;\n setBootstrapping(true);\n try {\n await bootstrapAdmin();\n snackbarController.open({ type: \"success\",\nmessage: t(\"bootstrap_admin_success\") });\n window.location.reload();\n } catch (error: unknown) {\n snackbarController.open({ type: \"error\",\nmessage: error instanceof Error ? error.message : t(\"failed_to_bootstrap_admin\") });\n } finally {\n setBootstrapping(false);\n }\n };\n\n const handleAddUser = () => {\n setSelectedUser(undefined);\n setFormKey(k => k + 1);\n setDialogOpen(true);\n };\n\n const handleEditUser = (user: User) => {\n setSelectedUser(user);\n setDialogOpen(true);\n };\n\n const handleClose = () => {\n setDialogOpen(false);\n setSelectedUser(undefined);\n };\n\n const handleDelete = async () => {\n if (!userToDelete || !deleteUser) return;\n setDeleteInProgress(true);\n try {\n await deleteUser(userToDelete);\n snackbarController.open({ type: \"success\",\nmessage: t(\"user_deleted_successfully\") });\n setDeleteConfirmOpen(false);\n setUserToDelete(undefined);\n refreshCurrentPage();\n } catch (error: unknown) {\n snackbarController.open({ type: \"error\",\nmessage: error instanceof Error ? error.message : t(\"error_deleting_user\") });\n } finally {\n setDeleteInProgress(false);\n }\n };\n\n const handleResetPassword = async () => {\n if (!userToReset || !resetPassword) return;\n setResetInProgress(true);\n try {\n const result = await resetPassword(userToReset);\n setResetConfirmOpen(false);\n setUserToReset(undefined);\n setCreationResult(result);\n snackbarController.open({ type: \"success\",\nmessage: t(\"reset_password_success\") });\n } catch (error: unknown) {\n snackbarController.open({ type: \"error\",\nmessage: error instanceof Error ? error.message : t(\"error_resetting_password\") });\n } finally {\n setResetInProgress(false);\n }\n };\n\n return (\n <Container className=\"w-full flex flex-col py-4 gap-4\" maxWidth={\"6xl\"}>\n <BootstrapAdminBanner className=\"mb-4\" />\n\n <div className=\"flex items-center mt-12 mb-4 gap-4\">\n <Typography gutterBottom variant=\"h4\" className=\"grow mb-0\" component=\"h4\">\n {t(\"users\")}\n </Typography>\n {roles && roles.length > 0 && (\n <Select\n value={roleFilter || \"__all__\"}\n onValueChange={(v) => handleRoleFilterChange(v === \"__all__\" ? \"\" : v)}\n placeholder={t(\"all_roles\") || \"All Roles\"}\n size=\"small\"\n className=\"w-48\"\n >\n <SelectItem value=\"__all__\">{t(\"all_roles\") || \"All Roles\"}</SelectItem>\n {roles.map(role => (\n <SelectItem key={role.id} value={role.id}>{role.name}</SelectItem>\n ))}\n </Select>\n )}\n <SearchBar\n placeholder={t(\"search_users\")}\n onTextSearch={(v) => handleSearch(v || \"\")}\n size=\"small\"\n expandable\n />\n <Button startIcon={<PlusIcon/>} onClick={handleAddUser} disabled={!saveUser}>\n {t(\"add_user\")}\n </Button>\n </div>\n\n <div className=\"overflow-auto\">\n <Table className=\"w-full\">\n <TableHeader>\n <TableCell header className=\"w-48\">{t(\"id\") || \"ID\"}</TableCell>\n <TableCell header>{t(\"email\")}</TableCell>\n <TableCell header>{t(\"name\")}</TableCell>\n <TableCell header>{t(\"roles\")}</TableCell>\n <TableCell header className=\"whitespace-nowrap\">{t(\"created\")}</TableCell>\n <TableCell header className=\"w-24 text-right\">{t(\"actions\")}</TableCell>\n </TableHeader>\n <TableBody>\n {(tableLoading || delegateLoading) ? (\n [\n { email: \"w-48\",\nname: \"w-32\",\nroles: [\"w-16\", \"w-20\"] },\n { email: \"w-32\",\nname: \"w-24\",\nroles: [\"w-24\"] },\n { email: \"w-40\",\nname: \"w-36\",\nroles: [\"w-16\", \"w-16\"] }\n ].map((row, i) => (\n <TableRow key={`skeleton-${i}`}>\n <TableCell className=\"font-mono text-xs\"><Skeleton className=\"h-3 w-40\"/></TableCell>\n <TableCell><Skeleton className={`h-4 ${row.email}`}/></TableCell>\n <TableCell className=\"font-medium\"><Skeleton className={`h-4 ${row.name}`}/></TableCell>\n <TableCell>\n <div className=\"flex flex-wrap gap-2\">\n {row.roles.map((w, j) => (\n <Skeleton key={j} className={`h-6 ${w} rounded-full`}/>\n ))}\n </div>\n </TableCell>\n <TableCell className=\"whitespace-nowrap text-sm\">\n <Skeleton className=\"h-4 w-20\"/>\n </TableCell>\n <TableCell className=\"text-right whitespace-nowrap\">\n <div className=\"flex justify-end items-center gap-1\">\n <Skeleton className=\"h-7 w-7 rounded-md\"/>\n <Skeleton className=\"h-7 w-7 rounded-md\"/>\n </div>\n </TableCell>\n </TableRow>\n ))\n ) : (\n displayUsers.map(user => (\n <TableRow key={user.uid} onClick={() => saveUser && handleEditUser(user)}>\n <TableCell className=\"font-mono text-xs\">{user.uid}</TableCell>\n <TableCell>{user.email}</TableCell>\n <TableCell className=\"font-medium\">{user.displayName}</TableCell>\n <TableCell>\n <div className=\"flex flex-wrap gap-2\">\n {user.roles?.map((roleId: string) => {\n const role = roles?.find(r => r.id === roleId);\n return role ? <RoleChip key={roleId} role={role}/> : <span key={roleId}>{roleId}</span>;\n })}\n </div>\n </TableCell>\n <TableCell className=\"whitespace-nowrap text-sm text-surface-accent-600 dark:text-surface-accent-400\">\n {user.createdAt ? new Date(user.createdAt).toLocaleDateString() : \"-\"}\n </TableCell>\n <TableCell className=\"text-right whitespace-nowrap\">\n <div className=\"flex justify-end items-center gap-1\">\n {resetPassword && (\n <Tooltip asChild title={t(\"reset_password\")}>\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n setUserToReset(user);\n setResetConfirmOpen(true);\n }}>\n <KeyRoundIcon size={iconSize.small}/>\n </IconButton>\n </Tooltip>\n )}\n {deleteUser && (\n <Tooltip asChild title={t(\"delete_this_user\")}>\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n setUserToDelete(user);\n setDeleteConfirmOpen(true);\n }}>\n <Trash2Icon size={iconSize.small}/>\n </IconButton>\n </Tooltip>\n )}\n </div>\n </TableCell>\n </TableRow>\n )))}\n\n {displayUsers.length === 0 && !tableLoading && !delegateLoading && (\n <TableRow>\n <TableCell colspan={6}>\n <CenteredView className=\"flex flex-col gap-4 my-8 items-center\">\n <Typography variant=\"label\">\n {usersError\n ? t(\"no_permission_to_view_users\")\n : searchQuery ? t(\"no_users_found\") : t(\"no_users_yet\")}\n </Typography>\n {usersError && (\n <Typography variant=\"caption\" color=\"secondary\">\n {t(\"no_permission_description\")}\n </Typography>\n )}\n </CenteredView>\n </TableCell>\n </TableRow>\n )}\n </TableBody>\n </Table>\n </div>\n\n {/* Pagination */}\n {displayTotal > PAGE_SIZE && (\n <div className=\"flex items-center justify-between px-2 py-3\">\n <Typography variant=\"body2\" className=\"text-surface-accent-500 dark:text-surface-accent-400\">\n {`${page * PAGE_SIZE + 1}–${Math.min((page + 1) * PAGE_SIZE, displayTotal)} / ${displayTotal}`}\n </Typography>\n <div className=\"flex items-center gap-1\">\n <IconButton\n size=\"small\"\n disabled={page === 0}\n onClick={() => handlePageChange(page - 1)}>\n <ChevronLeftIcon size={iconSize.smallest}/>\n </IconButton>\n <Typography variant=\"body2\" className=\"px-3 text-surface-accent-600 dark:text-surface-accent-300\">\n {page + 1} / {totalPages}\n </Typography>\n <IconButton\n size=\"small\"\n disabled={page >= totalPages - 1}\n onClick={() => handlePageChange(page + 1)}>\n <ChevronRightIcon size={iconSize.smallest}/>\n </IconButton>\n </div>\n </div>\n )}\n\n {/* User Edit Dialog */}\n {saveUser && (\n <UserDetailsForm\n key={selectedUser?.uid ?? `new-${formKey}`}\n open={dialogOpen}\n user={selectedUser}\n roles={roles}\n saveUser={saveUser}\n createUser={createUser}\n handleClose={handleClose}\n onCreationResult={setCreationResult}\n onSaved={refreshCurrentPage}\n />\n )}\n\n {/* Creation Result Dialog */}\n {creationResult && (\n <CreationResultDialog\n result={creationResult}\n onClose={() => setCreationResult(null)}\n />\n )}\n\n {/* Delete Confirmation */}\n <ConfirmationDialog\n open={deleteConfirmOpen}\n loading={deleteInProgress}\n onAccept={handleDelete}\n onCancel={() => { setDeleteConfirmOpen(false); setUserToDelete(undefined); }}\n title={<>{t(\"delete_confirmation_title\")}</>}\n body={<>{t(\"delete_user_confirmation\")}</>}\n />\n\n {/* Reset Password Confirmation */}\n <ConfirmationDialog\n open={resetConfirmOpen}\n loading={resetInProgress}\n onAccept={handleResetPassword}\n onCancel={() => { setResetConfirmOpen(false); setUserToReset(undefined); }}\n title={<>{t(\"reset_password\")}</>}\n body={<>{t(\"reset_password_confirmation\")}</>}\n />\n </Container>\n );\n}\n\n// ============================================\n// UserDetailsForm Component\n// ============================================\nfunction UserDetailsForm({\n open,\n user: userProp,\n roles,\n saveUser,\n createUser,\n handleClose,\n onCreationResult,\n onSaved\n}: {\n open: boolean;\n user?: User;\n roles?: Role[];\n saveUser: (user: User) => Promise<User>;\n createUser?: (user: User) => Promise<UserCreationResult>;\n handleClose: () => void;\n onCreationResult?: (result: UserCreationResult) => void;\n onSaved?: () => void;\n}) {\n const snackbarController = useSnackbarController();\n const { t } = useTranslation();\n const isNewUser = !userProp;\n\n const [displayName, setDisplayName] = useState(userProp?.displayName || \"\");\n const [email, setEmail] = useState(userProp?.email || \"\");\n const [selectedRoleIds, setSelectedRoleIds] = useState<string[]>(\n userProp?.roles || []\n );\n const [isSubmitting, setIsSubmitting] = useState(false);\n const [errors, setErrors] = useState<{ displayName?: string; email?: string; roles?: string }>({});\n const [submitCount, setSubmitCount] = useState(0);\n\n const validate = () => {\n const newErrors: typeof errors = {};\n if (!displayName) newErrors.displayName = \"Required\";\n if (!email) newErrors.email = \"Required\";\n else if (!/\\S+@\\S+\\.\\S+/.test(email)) newErrors.email = \"Invalid email\";\n setErrors(newErrors);\n return Object.keys(newErrors).length === 0;\n };\n\n const handleSubmit = async (e: React.FormEvent) => {\n e.preventDefault();\n setSubmitCount(c => c + 1);\n\n if (!validate()) return;\n\n setIsSubmitting(true);\n try {\n const userRoles = selectedRoleIds;\n const userToSave: User = {\n uid: userProp?.uid || crypto.randomUUID(),\n email,\n displayName: displayName || null,\n photoURL: userProp?.photoURL || null,\n providerId: userProp?.providerId || \"custom\",\n isAnonymous: userProp?.isAnonymous || false,\n roles: userRoles\n };\n\n if (isNewUser && createUser && onCreationResult) {\n // Use createUser for new users to get invitation/password info\n const result = await createUser(userToSave);\n handleClose();\n onCreationResult(result);\n } else {\n await saveUser(userToSave);\n handleClose();\n }\n onSaved?.();\n } catch (error: unknown) {\n snackbarController.open({ type: \"error\",\nmessage: error instanceof Error ? error.message : \"Failed to save user\" });\n } finally {\n setIsSubmitting(false);\n }\n };\n\n const dirty = isNewUser ||\n displayName !== (userProp?.displayName || \"\") ||\n email !== (userProp?.email || \"\") ||\n (() => {\n const prev = userProp?.roles || [];\n if (selectedRoleIds.length !== prev.length) return true;\n const set = new Set(prev);\n return selectedRoleIds.some(id => !set.has(id));\n })();\n\n return (\n <Dialog open={open} onOpenChange={(open) => !open ? handleClose() : undefined} maxWidth=\"4xl\">\n <form onSubmit={handleSubmit} autoComplete=\"off\" noValidate\n style={{ display: \"flex\",\nflexDirection: \"column\",\nposition: \"relative\",\nheight: \"100%\" }}>\n\n <DialogTitle variant=\"h4\" gutterBottom={false}>\n {t(\"user\")}\n </DialogTitle>\n\n <DialogContent className=\"h-full grow\">\n <div className=\"grid grid-cols-12 gap-4\">\n {!isNewUser && (\n <div className=\"col-span-12\">\n <TextField\n name=\"uid\"\n value={userProp?.uid || \"\"}\n label={t(\"id\") || \"ID\"}\n disabled\n />\n </div>\n )}\n <div className=\"col-span-12\">\n <TextField\n name=\"displayName\"\n required\n error={submitCount > 0 && Boolean(errors.displayName)}\n value={displayName}\n onChange={(e) => setDisplayName(e.target.value)}\n label={t(\"name\")}\n />\n {submitCount > 0 && errors.displayName && (\n <Typography variant=\"caption\" color=\"error\">{errors.displayName}</Typography>\n )}\n </div>\n\n <div className=\"col-span-12\">\n <TextField\n required\n error={submitCount > 0 && Boolean(errors.email)}\n name=\"email\"\n value={email}\n onChange={(e) => setEmail(e.target.value)}\n label={t(\"email\")}\n disabled={!isNewUser}\n />\n {submitCount > 0 && errors.email && (\n <Typography variant=\"caption\" color=\"error\">{errors.email}</Typography>\n )}\n </div>\n\n {roles && roles.length > 0 && (\n <div className=\"col-span-12\">\n <MultiSelect\n className=\"w-full\"\n label={t(\"roles\")}\n value={selectedRoleIds}\n onValueChange={(value: string[]) => setSelectedRoleIds(value)}\n >\n {roles.map(role => (\n <MultiSelectItem key={role.id} value={role.id}>\n <RoleChip role={role}/>\n </MultiSelectItem>\n ))}\n </MultiSelect>\n </div>\n )}\n </div>\n </DialogContent>\n\n <DialogActions>\n <Button variant=\"text\" onClick={handleClose}>\n {t(\"cancel\")}\n </Button>\n <LoadingButton\n variant=\"filled\"\n type=\"submit\"\n disabled={!dirty}\n loading={isSubmitting}\n >\n {isNewUser ? t(\"create_user\") : t(\"update\")}\n </LoadingButton>\n </DialogActions>\n </form>\n </Dialog>\n );\n}\n"],"names":["PAGE_SIZE","UsersView","userManagement","roles","saveUser","createUser","deleteUser","resetPassword","loading","delegateLoading","bootstrapAdmin","usersError","snackbarController","useSnackbarController","user","loggedInUser","useAuthController","t","useTranslation","breadcrumbs","useBreadcrumbsController","React","useEffect","set","title","url","dialogOpen","setDialogOpen","useState","selectedUser","setSelectedUser","deleteConfirmOpen","setDeleteConfirmOpen","userToDelete","setUserToDelete","deleteInProgress","setDeleteInProgress","formKey","setFormKey","bootstrapping","setBootstrapping","creationResult","setCreationResult","resetConfirmOpen","setResetConfirmOpen","userToReset","setUserToReset","resetInProgress","setResetInProgress","hasServerSearch","searchUsers","searchQuery","setSearchQuery","roleFilter","setRoleFilter","page","setPage","paginatedUsers","setPaginatedUsers","totalUsers","setTotalUsers","tableLoading","setTableLoading","searchTimerRef","useRef","allUsers","users","fetchPage","useCallback","pageNum","search","filterRole","forceLoading","result","undefined","roleId","limit","offset","orderBy","orderDir","total","error","console","open","type","message","Error","fetchPageRef","current","initialFetchDone","handleSearch","value","clearTimeout","setTimeout","handleRoleFilterChange","newRole","handlePageChange","newPage","refreshCurrentPage","displayUsers","displayTotal","filtered","filter","u","matches","q","toLowerCase","email","includes","displayName","length","slice","totalPages","Math","max","ceil","some","handleAddUser","k","handleEditUser","handleClose","handleDelete","handleResetPassword","v","map","role","id","name","row","i","w","j","uid","find","r","createdAt","Date","toLocaleDateString","e","stopPropagation","iconSize","small","min","smallest","UserDetailsForm","userProp","onCreationResult","onSaved","isNewUser","setDisplayName","setEmail","selectedRoleIds","setSelectedRoleIds","isSubmitting","setIsSubmitting","errors","setErrors","submitCount","setSubmitCount","validate","newErrors","test","Object","keys","handleSubmit","preventDefault","c","userRoles","userToSave","crypto","randomUUID","photoURL","providerId","isAnonymous","dirty","prev","Set","has","display","flexDirection","position","height","Boolean","target"],"mappings":"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"}
@@ -1,19 +0,0 @@
1
- import { EntityCollection } from "@rebasepro/types";
2
- export type CollectionEditorPermissionsBuilder<USER = any, EC extends EntityCollection = EntityCollection> = (params: {
3
- user: USER | null;
4
- collection?: EC;
5
- }) => CollectionEditorPermissions;
6
- export type CollectionEditorPermissions = {
7
- /**
8
- * Is the user allowed to create new collections.
9
- */
10
- createCollections: boolean;
11
- /**
12
- * Is the user allowed to modify this collection
13
- */
14
- editCollections: boolean;
15
- /**
16
- * Is the user allowed to delete this collection
17
- */
18
- deleteCollections: boolean;
19
- };
@@ -1,4 +0,0 @@
1
- import { Role } from "@rebasepro/types";
2
- export declare function RoleChip({ role }: {
3
- role: Role;
4
- }): import("react/jsx-runtime").JSX.Element;
@@ -1,2 +0,0 @@
1
- import { CollectionActionsProps } from "@rebasepro/types";
2
- export declare function RolesFilterSelect({ tableController }: CollectionActionsProps<any>): import("react/jsx-runtime").JSX.Element | null;
@@ -1,4 +0,0 @@
1
- import { UserManagementDelegate } from "@rebasepro/types";
2
- export declare function RolesView({ userManagement }: {
3
- userManagement: UserManagementDelegate;
4
- }): import("react/jsx-runtime").JSX.Element;
@@ -1,2 +0,0 @@
1
- import { FieldProps } from "../../types/fields";
2
- export declare function UserRolesSelectField({ propertyKey, value, setValue, disabled }: FieldProps): import("react/jsx-runtime").JSX.Element;
@@ -1,4 +0,0 @@
1
- import { UserManagementDelegate } from "@rebasepro/types";
2
- export declare function UsersView({ userManagement }: {
3
- userManagement: UserManagementDelegate;
4
- }): import("react/jsx-runtime").JSX.Element;
@@ -1 +0,0 @@
1
- {"version":3,"file":"index-DjduZG1T.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"index-MKPc70-v.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;"}