studiokit-scaffolding-js 7.0.12-next.1.3 → 7.0.12-next.2.1

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 (258) hide show
  1. package/lib/components/ActionList.js +164 -37
  2. package/lib/components/AlertDialog.js +128 -12
  3. package/lib/components/AlertWithIcon.js +88 -29
  4. package/lib/components/ConnectedModal.js +35 -12
  5. package/lib/components/Dropdowns/GroupsDropdown.js +63 -45
  6. package/lib/components/Dropdowns/ManagedNavDropdown.js +92 -67
  7. package/lib/components/Dropdowns/UserDropdown.js +105 -24
  8. package/lib/components/Dropdowns/index.js +4 -10
  9. package/lib/components/EntityOwnerList.js +47 -21
  10. package/lib/components/Error.js +101 -12
  11. package/lib/components/ErrorBoundary.js +127 -38
  12. package/lib/components/ErrorMessage.js +39 -12
  13. package/lib/components/Forms/DateField.js +56 -45
  14. package/lib/components/Forms/TimeField.js +76 -45
  15. package/lib/components/Forms/index.js +3 -5
  16. package/lib/components/Groups/CreateEditCopySaveButtons.js +109 -14
  17. package/lib/components/Groups/ExternalGroups/Attach.js +206 -151
  18. package/lib/components/Groups/ExternalGroups/Table.js +176 -48
  19. package/lib/components/Groups/GroupCreateOrEditCommonProps.js +2 -2
  20. package/lib/components/Groups/RosterSyncInfo.js +142 -23
  21. package/lib/components/HOC/AccessibleAppComponent.js +88 -72
  22. package/lib/components/HOC/ActivityRequiredComponent.js +68 -33
  23. package/lib/components/HOC/AsyncComponent.js +49 -41
  24. package/lib/components/HOC/AuthenticatedComponent.js +55 -44
  25. package/lib/components/HOC/CollectionComponent.js +154 -104
  26. package/lib/components/HOC/CollectionFirstItemComponent.js +45 -40
  27. package/lib/components/HOC/CollectionItemComponent.js +152 -100
  28. package/lib/components/HOC/ConnectedModalComponent.js +87 -69
  29. package/lib/components/HOC/DataDependentComponent.js +26 -27
  30. package/lib/components/HOC/EntityComponent.js +57 -53
  31. package/lib/components/HOC/FullscreenModalComponent.js +139 -108
  32. package/lib/components/HOC/GroupActivityRequiredComponent.js +27 -20
  33. package/lib/components/HOC/GuidComponent.js +20 -20
  34. package/lib/components/HOC/ModelContextDependencyVerifyComponent.js +32 -29
  35. package/lib/components/HOC/ModelErrorRedirectComponent.js +37 -39
  36. package/lib/components/HOC/SearchPersistorComponent.js +237 -173
  37. package/lib/components/HOC/UnauthenticatedComponent.js +32 -30
  38. package/lib/components/HOC/UserComponent.js +6 -8
  39. package/lib/components/Icons/IconAlphaList.js +28 -8
  40. package/lib/components/Icons/IconExternalUser.js +28 -8
  41. package/lib/components/Icons/IconImpersonation.js +28 -8
  42. package/lib/components/Icons/IconStopImpersonating.js +28 -8
  43. package/lib/components/Icons/IconTable.js +29 -9
  44. package/lib/components/Icons/IconTableDeleteCol.js +28 -8
  45. package/lib/components/Icons/IconTableDeleteRow.js +28 -8
  46. package/lib/components/Icons/IconTableInsertCol.js +28 -8
  47. package/lib/components/Icons/IconTableInsertRow.js +28 -8
  48. package/lib/components/Impersonation/Button.js +71 -16
  49. package/lib/components/Impersonation/Link.js +72 -16
  50. package/lib/components/Impersonation/UserDetail.js +60 -11
  51. package/lib/components/Loading.js +23 -8
  52. package/lib/components/LockDownBrowser/Check.js +188 -51
  53. package/lib/components/LockDownBrowser/ExitButton.js +22 -13
  54. package/lib/components/LockDownBrowser/Launch.js +64 -64
  55. package/lib/components/Lti/Confirm.js +147 -14
  56. package/lib/components/Lti/CreateNonLtiGroupAlertDialog.js +165 -36
  57. package/lib/components/Lti/Launch.js +99 -25
  58. package/lib/components/Lti/LaunchGroup.js +81 -16
  59. package/lib/components/ManageTable.js +304 -90
  60. package/lib/components/ManageTableNoDataComponent.js +38 -7
  61. package/lib/components/NewVersionAlert.js +76 -49
  62. package/lib/components/NotFound.js +81 -11
  63. package/lib/components/Notifications.js +179 -129
  64. package/lib/components/PaginationNextButton.js +28 -9
  65. package/lib/components/PaginationPreviousButton.js +28 -9
  66. package/lib/components/Quill/CustomToolbar.js +427 -222
  67. package/lib/components/Quill/Formats/Image.js +67 -67
  68. package/lib/components/Quill/Formats/List.js +38 -47
  69. package/lib/components/Quill/Formats/Video.js +23 -26
  70. package/lib/components/Quill/ImageDropModule.js +136 -114
  71. package/lib/components/Quill/ImageWarning.js +41 -12
  72. package/lib/components/Quill/ImageWithAltTextModal.js +420 -89
  73. package/lib/components/Quill/Specs/CustomImageSpec.js +32 -31
  74. package/lib/components/Quill/Specs/CustomVideoSpec.js +22 -23
  75. package/lib/components/Quill/TableModule/Blots/BaseTableBlot.js +89 -97
  76. package/lib/components/Quill/TableModule/Blots/TableBlot.js +47 -50
  77. package/lib/components/Quill/TableModule/Blots/TableBodyBlot.js +48 -51
  78. package/lib/components/Quill/TableModule/Blots/TableCellBlot.js +219 -224
  79. package/lib/components/Quill/TableModule/Blots/TableContainer.js +75 -86
  80. package/lib/components/Quill/TableModule/Blots/TableRowBlot.js +70 -73
  81. package/lib/components/Quill/TableModule/constants.js +40 -42
  82. package/lib/components/Quill/TableModule/index.js +357 -305
  83. package/lib/components/Quill/TableModule/utils.js +39 -48
  84. package/lib/components/Quill/accessibilityFix.js +219 -223
  85. package/lib/components/Quill/index.js +30 -33
  86. package/lib/components/RefreshIndicator/Bordered.js +44 -10
  87. package/lib/components/RefreshIndicator/Inline.js +43 -12
  88. package/lib/components/RefreshIndicator/index.js +257 -62
  89. package/lib/components/SearchControls.js +211 -14
  90. package/lib/components/SentryRoute.js +5 -7
  91. package/lib/components/Tables/RoleFilter.js +66 -38
  92. package/lib/components/Tables/TextFilter.js +58 -18
  93. package/lib/components/UserRoles/Add.js +193 -99
  94. package/lib/components/UserRoles/Context.js +3 -6
  95. package/lib/components/UserRoles/RoleCell.js +176 -75
  96. package/lib/components/UserRoles/Select.js +151 -20
  97. package/lib/components/UserRoles/Table.js +215 -82
  98. package/lib/components/UserRoles/index.js +526 -386
  99. package/lib/config/eslint/index.js +26 -29
  100. package/lib/config/eslint/lib/order.js +21 -28
  101. package/lib/config/eslint/lib/prettier.js +15 -19
  102. package/lib/config/eslint/lib/typescript.js +87 -113
  103. package/lib/config/eslint/react.js +18 -15
  104. package/lib/constants/baseActivity.js +26 -28
  105. package/lib/constants/baseRole.js +10 -12
  106. package/lib/constants/configuration.js +43 -55
  107. package/lib/constants/externalProviderType.js +6 -8
  108. package/lib/constants/fetchErrorData.js +10 -12
  109. package/lib/constants/index.js +13 -15
  110. package/lib/constants/lockDownBrowser.js +23 -25
  111. package/lib/constants/mockData.js +370 -300
  112. package/lib/constants/modelStatus.js +11 -13
  113. package/lib/constants/notificationType.js +8 -10
  114. package/lib/constants/operatingSystem.js +8 -10
  115. package/lib/constants/shard.js +7 -9
  116. package/lib/constants/table.js +18 -22
  117. package/lib/constants/tier.js +8 -10
  118. package/lib/constants/userRole.js +11 -8
  119. package/lib/endpointMappings.js +191 -182
  120. package/lib/hooks/useCollection.js +79 -65
  121. package/lib/hooks/useCollectionConfiguration.js +220 -80
  122. package/lib/hooks/useCollectionItem.js +151 -57
  123. package/lib/hooks/useGuid.js +16 -9
  124. package/lib/hooks/usePrevious.js +14 -13
  125. package/lib/index.js +11 -26
  126. package/lib/redux/actionCreator.js +44 -35
  127. package/lib/redux/actions/AuthAction.js +45 -32
  128. package/lib/redux/actions/ModalAction.js +6 -8
  129. package/lib/redux/actions/ModelAction.js +95 -43
  130. package/lib/redux/actions/NotificationAction.js +6 -8
  131. package/lib/redux/actions/SearchAction.js +5 -7
  132. package/lib/redux/actions/index.js +6 -8
  133. package/lib/redux/configureReducers.js +48 -46
  134. package/lib/redux/configureStore.js +77 -91
  135. package/lib/redux/helpers.js +2 -5
  136. package/lib/redux/reducers/authReducer.js +44 -43
  137. package/lib/redux/reducers/index.js +7 -14
  138. package/lib/redux/reducers/modalsReducer.js +43 -31
  139. package/lib/redux/reducers/modelsReducer.js +131 -137
  140. package/lib/redux/reducers/notificationsReducer.js +20 -20
  141. package/lib/redux/reducers/searchReducer.js +13 -13
  142. package/lib/redux/sagas/appInsightsSaga.js +19 -21
  143. package/lib/redux/sagas/authSaga.js +248 -234
  144. package/lib/redux/sagas/caliperSaga.js +142 -131
  145. package/lib/redux/sagas/clockOffsetSaga.js +29 -32
  146. package/lib/redux/sagas/configurationSaga.js +8 -10
  147. package/lib/redux/sagas/downtimeApiErrorSaga.js +16 -19
  148. package/lib/redux/sagas/errorSaga.js +23 -24
  149. package/lib/redux/sagas/googleAnalyticsSaga.js +24 -27
  150. package/lib/redux/sagas/identityProviderSaga.js +19 -21
  151. package/lib/redux/sagas/initialDataLoadSaga.js +34 -31
  152. package/lib/redux/sagas/lockDownBrowserErrorSaga.js +25 -22
  153. package/lib/redux/sagas/modelFetchSaga.js +302 -286
  154. package/lib/redux/sagas/noStoreSaga.js +60 -61
  155. package/lib/redux/sagas/postLoginDataSaga.js +37 -32
  156. package/lib/redux/sagas/postLoginRedirectSaga.js +22 -27
  157. package/lib/redux/sagas/rootSaga.js +77 -60
  158. package/lib/redux/sagas/sentrySaga.js +25 -28
  159. package/lib/redux/sagas/userIdSaga.js +13 -15
  160. package/lib/services/codeProviderService.js +21 -21
  161. package/lib/services/dateService.js +6 -8
  162. package/lib/services/documentService.js +10 -11
  163. package/lib/services/fetchService.js +103 -95
  164. package/lib/services/persistenceService.js +27 -30
  165. package/lib/services/ticketProviderService.js +25 -25
  166. package/lib/services/tokenPersistenceService.js +8 -10
  167. package/lib/services/windowService.js +14 -16
  168. package/lib/startup.js +110 -101
  169. package/lib/types/AppConfiguration.js +2 -2
  170. package/lib/types/Artifact.js +7 -9
  171. package/lib/types/BaseReduxState.js +2 -2
  172. package/lib/types/Client.js +2 -2
  173. package/lib/types/Collection.js +2 -2
  174. package/lib/types/Configuration.js +2 -2
  175. package/lib/types/DeepLinkingResponseRequest.js +2 -2
  176. package/lib/types/DeletableModel.js +2 -2
  177. package/lib/types/Event.js +2 -2
  178. package/lib/types/ExternalGroup.js +2 -2
  179. package/lib/types/ExternalProvider.js +2 -2
  180. package/lib/types/ExternalTerm.js +2 -2
  181. package/lib/types/Group.js +2 -2
  182. package/lib/types/IdentityProvider.js +2 -2
  183. package/lib/types/LtiLaunch.js +2 -2
  184. package/lib/types/NameOnlyEntity.js +2 -2
  185. package/lib/types/Notification.js +2 -2
  186. package/lib/types/OptionalRecord.js +2 -2
  187. package/lib/types/OwnerSchedule.js +2 -2
  188. package/lib/types/PropertyOfType.js +2 -2
  189. package/lib/types/Quill.js +2 -2
  190. package/lib/types/RoleDescription.js +2 -2
  191. package/lib/types/Search.js +2 -2
  192. package/lib/types/SimpleLocation.js +2 -2
  193. package/lib/types/UniTime.js +2 -2
  194. package/lib/types/User.js +2 -2
  195. package/lib/types/UserRole.js +2 -2
  196. package/lib/types/auth/AuthState.js +2 -2
  197. package/lib/types/auth/CasV1LoginRequestBody.js +2 -2
  198. package/lib/types/auth/ClientCredentials.js +2 -2
  199. package/lib/types/auth/CodeProviderService.js +2 -2
  200. package/lib/types/auth/LocalLoginRequestBody.js +2 -2
  201. package/lib/types/auth/TicketProviderService.js +2 -2
  202. package/lib/types/auth/TokenPersistenceService.js +2 -2
  203. package/lib/types/auth/index.js +8 -10
  204. package/lib/types/externals.d.js +2 -0
  205. package/lib/types/index.js +29 -31
  206. package/lib/types/net/EndpointConfig.js +2 -2
  207. package/lib/types/net/EndpointMapping.js +2 -2
  208. package/lib/types/net/EndpointMappings.js +2 -2
  209. package/lib/types/net/ErrorHandler.js +2 -2
  210. package/lib/types/net/FetchConfig.js +2 -2
  211. package/lib/types/net/FetchErrorData.js +6 -8
  212. package/lib/types/net/FetchResult.js +2 -2
  213. package/lib/types/net/HTTPMethod.js +2 -2
  214. package/lib/types/net/HTTPStatusCode.js +12 -14
  215. package/lib/types/net/Metadata.js +2 -2
  216. package/lib/types/net/Model.js +2 -2
  217. package/lib/types/net/ModelCollection.js +2 -2
  218. package/lib/types/net/ModelsState.js +2 -2
  219. package/lib/types/net/OAuthToken.js +2 -2
  220. package/lib/types/net/OAuthTokenOrNull.js +2 -2
  221. package/lib/types/net/TokenAccessFunction.js +2 -2
  222. package/lib/types/net/index.js +17 -19
  223. package/lib/utils/baseActivity.js +83 -85
  224. package/lib/utils/baseRole.js +32 -36
  225. package/lib/utils/collection.js +403 -297
  226. package/lib/utils/cookies.js +19 -23
  227. package/lib/utils/date.js +188 -205
  228. package/lib/utils/dom.js +130 -131
  229. package/lib/utils/domainIdentifier.js +4 -8
  230. package/lib/utils/entityUserRole.js +2 -5
  231. package/lib/utils/error.js +14 -19
  232. package/lib/utils/events.js +32 -31
  233. package/lib/utils/externalGroup.js +20 -25
  234. package/lib/utils/externalProviders.js +4 -7
  235. package/lib/utils/externalTerms.js +6 -6
  236. package/lib/utils/fetch.js +168 -176
  237. package/lib/utils/group.js +14 -11
  238. package/lib/utils/groupDates.js +38 -46
  239. package/lib/utils/groupRoles.js +23 -32
  240. package/lib/utils/lockDownBrowser.js +12 -15
  241. package/lib/utils/logger.js +23 -28
  242. package/lib/utils/lti.js +4 -7
  243. package/lib/utils/model.js +28 -43
  244. package/lib/utils/number.js +9 -13
  245. package/lib/utils/promise.js +23 -26
  246. package/lib/utils/quill.js +55 -60
  247. package/lib/utils/route.js +52 -60
  248. package/lib/utils/search.js +72 -87
  249. package/lib/utils/shard.js +33 -42
  250. package/lib/utils/sort.js +47 -50
  251. package/lib/utils/string.js +10 -12
  252. package/lib/utils/table.js +29 -33
  253. package/lib/utils/timezone.js +7 -12
  254. package/lib/utils/url.js +130 -144
  255. package/lib/utils/user.js +54 -64
  256. package/lib/utils/userAgent.js +7 -14
  257. package/lib/utils/userRole.js +36 -39
  258. package/package.json +17 -3
@@ -1,396 +1,536 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.mapStateToProps = exports.UserRoles = void 0;
4
- const jsx_runtime_1 = require("react/jsx-runtime");
5
- const lodash_1 = require("lodash");
6
- const react_1 = require("react");
7
- const react_bootstrap_1 = require("react-bootstrap");
8
- const react_redux_1 = require("react-redux");
9
- const uuid_1 = require("uuid");
10
- const AlertDialog_1 = require("../../components/AlertDialog");
11
- const Loading_1 = require("../../components/Loading");
12
- const Add_1 = require("../../components/UserRoles/Add");
13
- const Table_1 = require("../../components/UserRoles/Table");
14
- const constants_1 = require("../../constants");
15
- const baseRole_1 = require("../../constants/baseRole");
16
- const modelStatus_1 = require("../../constants/modelStatus");
17
- const actionCreator_1 = require("../../redux/actionCreator");
18
- const actions_1 = require("../../redux/actions");
19
- const noStoreSaga_1 = require("../../redux/sagas/noStoreSaga");
20
- const baseActivity_1 = require("../../utils/baseActivity");
21
- const baseRole_2 = require("../../utils/baseRole");
22
- const domainIdentifier_1 = require("../../utils/domainIdentifier");
23
- const entityUserRole_1 = require("../../utils/entityUserRole");
24
- const fetch_1 = require("../../utils/fetch");
25
- const sort_1 = require("../../utils/sort");
26
- const user_1 = require("../../utils/user");
27
- const userRole_1 = require("../../utils/userRole");
28
- const AlertWithIcon_1 = require("../AlertWithIcon");
29
- const Context_1 = require("./Context");
1
+ import _defineProperty from "@babel/runtime/helpers/defineProperty";
2
+ import { cloneDeep, isEqual, pickBy } from 'lodash';
3
+ import React, { Component } from 'react';
4
+ import { Col, Row } from 'react-bootstrap';
5
+ import { connect } from 'react-redux';
6
+ import { v4 as uuidv4 } from 'uuid';
7
+ import { AlertDialog } from '../../components/AlertDialog';
8
+ import { Loading } from '../../components/Loading';
9
+ import { UserRolesAdd } from '../../components/UserRoles/Add';
10
+ import { UserRolesTable } from '../../components/UserRoles/Table';
11
+ import { getEndpointMappings } from '../../constants';
12
+ import { BASE_ROLE } from '../../constants/baseRole';
13
+ import { MODEL_STATUS } from '../../constants/modelStatus';
14
+ import { dispatchAction, dispatchModelFetchRequest } from '../../redux/actionCreator';
15
+ import { MODEL_FETCH_REQUEST_ACTION_TYPE, MODEL_FETCH_RESULT_ACTION_TYPE } from '../../redux/actions';
16
+ import { noStoreHooks } from '../../redux/sagas/noStoreSaga';
17
+ import { canPerformActivityGlobally, canPerformActivityOnEntity, defaultOptions } from '../../utils/baseActivity';
18
+ import { singularArticleForBaseRole, textForBaseRole } from '../../utils/baseRole';
19
+ import { getDomainIdentifierTypePluralString } from '../../utils/domainIdentifier';
20
+ import { getRootModelName } from '../../utils/entityUserRole';
21
+ import { isFetchErrorData, prepareFetch } from '../../utils/fetch';
22
+ import { sortByNames } from '../../utils/sort';
23
+ import { displayName } from '../../utils/user';
24
+ import { convertToUserWithRoles, getUserId } from '../../utils/userRole';
25
+ import { AlertWithIcon } from '../AlertWithIcon';
26
+ import { UserRolesContext } from './Context';
27
+ import { jsxs as _jsxs, jsx as _jsx, Fragment as _Fragment } from "react/jsx-runtime";
30
28
  /**
31
29
  * Component used to manage either UserRoles (global roles) or EntityUserRoles.
32
30
  */
33
- class UserRoles extends react_1.Component {
34
- constructor() {
35
- super(...arguments);
36
- this.state = {
37
- sortedUsers: [],
38
- // add
39
- addUsersHookId: (0, uuid_1.v4)(),
40
- isAdding: false,
41
- isUpdating: false,
42
- isRemoving: false,
43
- shouldResetAddForm: false,
44
- shouldShowRemoveDialog: false
45
- };
46
- this.setStateFromProps = (props) => {
47
- const { modelArray: userRoles, filterUsers } = props;
48
- const usersWithRoles = Object.values(userRoles.reduce((result, userRole) => {
49
- const userId = (0, userRole_1.getUserId)(userRole);
50
- let userWithRoles = result[userId];
51
- if (!userWithRoles) {
52
- userWithRoles = (0, userRole_1.convertToUserWithRoles)(userRole);
53
- }
54
- else {
55
- userWithRoles.roles.push(userRole);
56
- }
57
- result[userId] = userWithRoles;
58
- return result;
59
- }, {}));
60
- const filteredUsers = filterUsers ? filterUsers(usersWithRoles) : usersWithRoles;
61
- const sortedUsers = filteredUsers.sort(sort_1.sortByNames);
62
- this.setState({
63
- sortedUsers
64
- });
65
- };
66
- this.textForRole = (role, lowercase = false) => {
67
- if (lowercase && this.props.lowercaseTextForRole) {
68
- return this.props.lowercaseTextForRole(role);
69
- }
70
- const roleString = this.props.textForRole ? this.props.textForRole(role) : (0, baseRole_2.textForBaseRole)(role);
71
- return lowercase ? roleString.toLowerCase() : roleString;
72
- };
73
- this.singularArticleForRole = (role) => {
74
- return this.props.singularArticleForRole
75
- ? this.props.singularArticleForRole(role)
76
- : (0, baseRole_2.singularArticleForBaseRole)(role);
77
- };
78
- //#region Add Users
79
- this.addUserRoles = (identifiers, role) => {
80
- const { entity } = this.props;
81
- this.setState({
82
- identifiersToAdd: identifiers,
83
- roleForAdd: role,
84
- successMessage: undefined,
85
- failMessage: undefined,
86
- existingMessage: undefined,
87
- blockedMessage: undefined,
88
- shouldResetAddForm: false,
89
- isAdding: true
90
- });
91
- const { addUsersHookId } = this.state;
92
- noStoreSaga_1.noStoreHooks.registerNoStoreActionHook(addUsersHookId, (data) => {
93
- noStoreSaga_1.noStoreHooks.unregisterNoStoreActionHook(addUsersHookId);
94
- this.didAdd(data);
95
- });
96
- const rootModelName = (0, entityUserRole_1.getRootModelName)(this.props);
97
- (0, actionCreator_1.dispatchModelFetchRequest)({
98
- modelName: `${rootModelName ? `${rootModelName}.` : ''}addUserRoles`,
99
- pathParams: entity ? [entity.id] : undefined,
100
- body: {
101
- entityId: entity ? entity.id : undefined,
102
- identifiers: identifiers,
103
- roleName: role
104
- },
105
- noStore: true,
106
- guid: addUsersHookId
107
- });
108
- };
109
- this.didAdd = (data) => {
110
- const { entityName, onAdd, model, modelName, pathParams } = this.props;
111
- const { roleForAdd, identifiersToAdd } = this.state;
112
- if (!roleForAdd || !identifiersToAdd || identifiersToAdd.length === 0) {
113
- throw new Error('didAdd was called in the incorrect state');
114
- }
115
- if (!data || (0, fetch_1.isFetchErrorData)(data)) {
116
- const failMessage = `The following identifiers were not added${entityName ? ` to your ${entityName}` : ''}:\r\n${identifiersToAdd.join('\r\n')}`;
117
- this.setState({
118
- shouldResetAddForm: true,
119
- isAdding: false,
120
- identifiersToAdd: undefined,
121
- roleForAdd: undefined,
122
- failMessage
123
- });
124
- return;
125
- }
126
- const { addedUserRoles, existingUserRoles, blockedUserRoles, invalidIdentifiers, allowedDomains, invalidDomainIdentifiers } = data;
127
- const roleString = this.textForRole(roleForAdd, true);
128
- const singularArticleString = this.singularArticleForRole(roleForAdd);
129
- const addedNames = addedUserRoles.length > 0
130
- ? addedUserRoles
131
- .map((au) => {
132
- return `${(0, user_1.displayName)(au)}${au.uid ? ` (${au.uid})` : ''}`;
133
- })
134
- .join('\r\n')
135
- : undefined;
136
- const successMessage = addedNames
137
- ? `The following ${roleString}${addedUserRoles.length > 1 ? 's were' : ' was'} successfully added${entityName ? ` to your ${entityName}` : ''}:\r\n${addedNames}`
138
- : undefined;
139
- const existingNames = existingUserRoles.length > 0
140
- ? existingUserRoles.map((eu) => `${(0, user_1.displayName)(eu)}${eu.uid ? ` (${eu.uid})` : ''}`).join('\r\n')
141
- : undefined;
142
- const hasMultipleExisting = existingUserRoles.length > 1;
143
- const existingMessage = existingNames
144
- ? `The following ${hasMultipleExisting ? 'people were' : 'person was'} already ${entityName
145
- ? `in your ${entityName} as ${hasMultipleExisting ? '' : `${singularArticleString} `}`
146
- : hasMultipleExisting
147
- ? ''
148
- : `${singularArticleString} `}${roleString}${hasMultipleExisting ? 's' : ''}:\r\n${existingNames}`
149
- : undefined;
150
- const blockedNames = !!blockedUserRoles && blockedUserRoles.length > 0
151
- ? blockedUserRoles.map((eu) => `${(0, user_1.displayName)(eu)}${eu.uid ? ` (${eu.uid})` : ''}`).join('\r\n')
152
- : undefined;
153
- const hasMultipleBlocked = !!blockedUserRoles && blockedUserRoles.length > 1;
154
- const blockedMessage = blockedNames
155
- ? `The following ${hasMultipleBlocked ? 'people' : 'person'} could not be added ${entityName ? `to your ${entityName}` : ''} as ${hasMultipleBlocked ? '' : `${singularArticleString} `}${roleString}${hasMultipleBlocked ? 's' : ''} because they already have a role:\r\n${blockedNames}`
156
- : undefined;
157
- let failMessage = invalidIdentifiers.length > 0
158
- ? `The following ${(0, domainIdentifier_1.getDomainIdentifierTypePluralString)()} are invalid:\r\n${invalidIdentifiers.join('\r\n')}`
159
- : undefined;
160
- if (invalidDomainIdentifiers.length > 0) {
161
- failMessage = `${failMessage ? `${failMessage}\r\n` : ''}The following ${(0, domainIdentifier_1.getDomainIdentifierTypePluralString)()} do not match the allowed domains of ${allowedDomains.replace(',', ', ')}:\r\n${invalidDomainIdentifiers.join('\r\n')}`;
162
- }
163
- this.setState({
164
- shouldResetAddForm: true,
165
- isAdding: false,
166
- identifiersToAdd: undefined,
167
- roleForAdd: undefined,
168
- successMessage,
169
- existingMessage,
170
- blockedMessage,
171
- failMessage
172
- });
173
- // only reload if we had some successful adds
174
- if (addedNames) {
175
- // update redux with added userRoles
176
- const updatedModel = (0, lodash_1.cloneDeep)(model);
177
- addedUserRoles.forEach(userRole => (updatedModel[userRole.id] = (0, lodash_1.cloneDeep)(userRole)));
178
- // create a dummy action and call `prepareFetch` to get the final `modelPath`
179
- const { modelPath } = (0, fetch_1.prepareFetch)({
180
- type: actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
181
- modelName,
182
- pathParams
183
- }, (0, constants_1.getEndpointMappings)());
184
- (0, actionCreator_1.dispatchAction)({
185
- type: actions_1.MODEL_FETCH_RESULT_ACTION_TYPE.FETCH_RESULT_RECEIVED,
186
- modelPath,
187
- data: updatedModel
188
- });
189
- if (onAdd) {
190
- onAdd(addedUserRoles);
191
- }
192
- }
193
- };
194
- //#endregion Add Users
195
- //#region Update
196
- this.updateUserRole = (userRoleToUpdate, newRole) => {
197
- this.setState({
198
- isUpdating: true,
199
- userRoleToUpdate,
200
- roleForUpdate: newRole,
201
- successMessage: undefined,
202
- failMessage: undefined
203
- });
204
- this.props.update({
205
- id: userRoleToUpdate.id,
206
- body: {},
207
- queryParams: {
208
- roleName: newRole
209
- }
210
- });
211
- };
212
- this.didUpdate = (isSuccess) => {
213
- const { entityName, onUpdate } = this.props;
214
- const { userRoleToUpdate, roleForUpdate } = this.state;
215
- if (userRoleToUpdate === undefined) {
216
- throw new Error('`didUpdate` was called without setting `userRoleToUpdate` in state');
217
- }
218
- if (roleForUpdate === undefined) {
219
- throw new Error('`didUpdate` was called without setting `roleForUpdate` in state');
220
- }
221
- const name = (0, user_1.displayName)(userRoleToUpdate);
222
- const roleString = this.textForRole(roleForUpdate, true);
223
- const singularArticleString = this.singularArticleForRole(roleForUpdate);
224
- if (!isSuccess) {
225
- const failMessage = `Oops! There was an error updating ${name} to ${singularArticleString} ${roleString}${entityName ? ` in your ${entityName}` : ''}. Please try again.`;
226
- this.setState({
227
- isUpdating: false,
228
- userRoleToUpdate: undefined,
229
- roleForUpdate: undefined,
230
- failMessage
231
- });
232
- return;
233
- }
234
- const successMessage = `${name} was successfully updated to ${singularArticleString} ${roleString}${entityName ? ` in your ${entityName}` : ''}.`;
235
- if (onUpdate) {
236
- onUpdate(userRoleToUpdate, roleForUpdate);
237
- }
238
- this.setState({
239
- isUpdating: false,
240
- userRoleToUpdate: undefined,
241
- roleForUpdate: undefined,
242
- successMessage
243
- });
244
- };
245
- //#endregion Update
246
- //#region Remove User
247
- this.alertRemoveUserRole = (userRoleToRemove) => {
248
- this.setState({
249
- userRoleToRemove,
250
- shouldShowRemoveDialog: true
251
- });
252
- };
253
- this.removeUserRoleTitle = (roleForRemove) => {
254
- return `Remove ${this.textForRole(roleForRemove)}`;
255
- };
256
- this.renderRemoveUserRoleDescription = (userRoleToRemove, warning) => {
257
- const { entity, externalProviders } = this.props;
258
- const group = entity && 'externalGroups' in entity ? entity : undefined;
259
- const roleString = this.textForRole(userRoleToRemove.role, true);
260
- const singularArticleString = this.singularArticleForRole(userRoleToRemove.role);
261
- const defaultWarning = warning !== null && warning !== void 0 ? warning : ((0, jsx_runtime_1.jsxs)("p", { className: "ma0", children: ["Are you sure you want to ", (0, jsx_runtime_1.jsxs)("strong", { children: ["remove ", (0, user_1.displayName)(userRoleToRemove)] }), " as", ' ', singularArticleString, " ", roleString, "?"] }));
262
- const externalGroupsToRemove = group === null || group === void 0 ? void 0 : group.externalGroups.filter(eg => eg.userId === userRoleToRemove.id);
263
- if (!externalGroupsToRemove ||
264
- externalGroupsToRemove.length === 0 ||
265
- userRoleToRemove.role !== baseRole_1.BASE_ROLE.GROUP_OWNER) {
266
- return defaultWarning;
267
- }
268
- const externalGroupsToRemoveWithRosterSync = externalGroupsToRemove.filter(eg => { var _a; return (_a = externalProviders === null || externalProviders === void 0 ? void 0 : externalProviders[eg.externalProviderId]) === null || _a === void 0 ? void 0 : _a.rosterSyncEnabled; });
269
- return ((0, jsx_runtime_1.jsxs)("div", { children: [(0, jsx_runtime_1.jsxs)("p", { children: ["The following course sections will be ", (0, jsx_runtime_1.jsx)("strong", { children: "disconnected" }), ".", ' ', externalGroupsToRemoveWithRosterSync.length > 0 && ((0, jsx_runtime_1.jsxs)("span", { children: ["Any automatically added students will be ", (0, jsx_runtime_1.jsx)("strong", { children: "removed" }), "."] }))] }), (0, jsx_runtime_1.jsx)("ul", { children: externalGroupsToRemove.map(eg => ((0, jsx_runtime_1.jsx)("li", { children: eg.description }, eg.id))) }), defaultWarning] }));
270
- };
271
- this.removeUserRole = (shouldRemove) => {
272
- const { userRoleToRemove } = this.state;
273
- if (userRoleToRemove === undefined) {
274
- throw new Error('`removeUser` was called without setting `userRoleToRemove` in state');
275
- }
276
- this.setState({
277
- isRemoving: shouldRemove,
278
- shouldShowRemoveDialog: false,
279
- successMessage: undefined,
280
- failMessage: undefined,
281
- // clear if cancelled
282
- userRoleToRemove: shouldRemove ? userRoleToRemove : undefined
283
- });
284
- if (!shouldRemove) {
285
- return;
286
- }
287
- this.props.delete({
288
- id: userRoleToRemove.id
289
- });
290
- };
291
- this.didRemove = (isSuccess) => {
292
- const { entityName, onRemove } = this.props;
293
- const { userRoleToRemove } = this.state;
294
- if (userRoleToRemove === undefined) {
295
- throw new Error('`didRemove` was called without setting `userRoleToRemove` in state');
296
- }
297
- const name = (0, user_1.displayName)(userRoleToRemove);
298
- if (!isSuccess) {
299
- const failMessage = `Oops! There was an error removing ${name}${entityName ? ` from your ${entityName}` : ''}. Please try again.`;
300
- this.setState({
301
- isRemoving: false,
302
- userRoleToRemove: undefined,
303
- failMessage
304
- });
305
- return;
306
- }
307
- const successMessage = `${name} was successfully removed${entityName ? ` from your ${entityName}` : ''}.`;
308
- if (onRemove) {
309
- onRemove(userRoleToRemove);
310
- }
311
- this.setState({
312
- isRemoving: false,
313
- userRoleToRemove: undefined,
314
- successMessage
315
- });
316
- };
317
- }
318
- componentDidMount() {
319
- this.setStateFromProps(this.props);
320
- const { addUsersHookId } = this.state;
321
- noStoreSaga_1.noStoreHooks.unregisterNoStoreActionHook(addUsersHookId);
322
- }
323
- componentDidUpdate(prevProps) {
324
- const { modelArray: prevModelArray, modelStatus: prevModelStatus } = prevProps;
325
- const { modelArray, modelStatus, entityName } = this.props;
326
- if (!(0, lodash_1.isEqual)(prevModelArray, modelArray)) {
327
- this.setStateFromProps(this.props);
328
- }
329
- // loading error
330
- if ((prevModelStatus === modelStatus_1.MODEL_STATUS.UNINITIALIZED || prevModelStatus === modelStatus_1.MODEL_STATUS.LOADING) &&
331
- modelStatus === modelStatus_1.MODEL_STATUS.ERROR) {
332
- this.setState({
333
- failMessage: `Oops! There was an error loading${entityName ? ` the people for your ${entityName}` : ''}.\r\nPlease try again.`
334
- });
31
+ export class UserRoles extends Component {
32
+ constructor() {
33
+ var _this;
34
+ super(...arguments);
35
+ _this = this;
36
+ _defineProperty(this, "state", {
37
+ sortedUsers: [],
38
+ // add
39
+ addUsersHookId: uuidv4(),
40
+ isAdding: false,
41
+ isUpdating: false,
42
+ isRemoving: false,
43
+ shouldResetAddForm: false,
44
+ shouldShowRemoveDialog: false
45
+ });
46
+ _defineProperty(this, "setStateFromProps", props => {
47
+ const {
48
+ modelArray: userRoles,
49
+ filterUsers
50
+ } = props;
51
+ const usersWithRoles = Object.values(userRoles.reduce((result, userRole) => {
52
+ const userId = getUserId(userRole);
53
+ let userWithRoles = result[userId];
54
+ if (!userWithRoles) {
55
+ userWithRoles = convertToUserWithRoles(userRole);
56
+ } else {
57
+ userWithRoles.roles.push(userRole);
335
58
  }
336
- // updating
337
- if (prevModelStatus === modelStatus_1.MODEL_STATUS.UPDATING && modelStatus !== modelStatus_1.MODEL_STATUS.UPDATING) {
338
- this.didUpdate(modelStatus === modelStatus_1.MODEL_STATUS.READY);
59
+ result[userId] = userWithRoles;
60
+ return result;
61
+ }, {}));
62
+ const filteredUsers = filterUsers ? filterUsers(usersWithRoles) : usersWithRoles;
63
+ const sortedUsers = filteredUsers.sort(sortByNames);
64
+ this.setState({
65
+ sortedUsers
66
+ });
67
+ });
68
+ _defineProperty(this, "textForRole", function (role) {
69
+ let lowercase = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false;
70
+ if (lowercase && _this.props.lowercaseTextForRole) {
71
+ return _this.props.lowercaseTextForRole(role);
72
+ }
73
+ const roleString = _this.props.textForRole ? _this.props.textForRole(role) : textForBaseRole(role);
74
+ return lowercase ? roleString.toLowerCase() : roleString;
75
+ });
76
+ _defineProperty(this, "singularArticleForRole", role => {
77
+ return this.props.singularArticleForRole ? this.props.singularArticleForRole(role) : singularArticleForBaseRole(role);
78
+ });
79
+ //#region Add Users
80
+ _defineProperty(this, "addUserRoles", (identifiers, role) => {
81
+ const {
82
+ entity
83
+ } = this.props;
84
+ this.setState({
85
+ identifiersToAdd: identifiers,
86
+ roleForAdd: role,
87
+ successMessage: undefined,
88
+ failMessage: undefined,
89
+ existingMessage: undefined,
90
+ blockedMessage: undefined,
91
+ shouldResetAddForm: false,
92
+ isAdding: true
93
+ });
94
+ const {
95
+ addUsersHookId
96
+ } = this.state;
97
+ noStoreHooks.registerNoStoreActionHook(addUsersHookId, data => {
98
+ noStoreHooks.unregisterNoStoreActionHook(addUsersHookId);
99
+ this.didAdd(data);
100
+ });
101
+ const rootModelName = getRootModelName(this.props);
102
+ dispatchModelFetchRequest({
103
+ modelName: `${rootModelName ? `${rootModelName}.` : ''}addUserRoles`,
104
+ pathParams: entity ? [entity.id] : undefined,
105
+ body: {
106
+ entityId: entity ? entity.id : undefined,
107
+ identifiers: identifiers,
108
+ roleName: role
109
+ },
110
+ noStore: true,
111
+ guid: addUsersHookId
112
+ });
113
+ });
114
+ _defineProperty(this, "didAdd", data => {
115
+ const {
116
+ entityName,
117
+ onAdd,
118
+ model,
119
+ modelName,
120
+ pathParams
121
+ } = this.props;
122
+ const {
123
+ roleForAdd,
124
+ identifiersToAdd
125
+ } = this.state;
126
+ if (!roleForAdd || !identifiersToAdd || identifiersToAdd.length === 0) {
127
+ throw new Error('didAdd was called in the incorrect state');
128
+ }
129
+ if (!data || isFetchErrorData(data)) {
130
+ const failMessage = `The following identifiers were not added${entityName ? ` to your ${entityName}` : ''}:\r\n${identifiersToAdd.join('\r\n')}`;
131
+ this.setState({
132
+ shouldResetAddForm: true,
133
+ isAdding: false,
134
+ identifiersToAdd: undefined,
135
+ roleForAdd: undefined,
136
+ failMessage
137
+ });
138
+ return;
139
+ }
140
+ const {
141
+ addedUserRoles,
142
+ existingUserRoles,
143
+ blockedUserRoles,
144
+ invalidIdentifiers,
145
+ allowedDomains,
146
+ invalidDomainIdentifiers
147
+ } = data;
148
+ const roleString = this.textForRole(roleForAdd, true);
149
+ const singularArticleString = this.singularArticleForRole(roleForAdd);
150
+ const addedNames = addedUserRoles.length > 0 ? addedUserRoles.map(au => {
151
+ return `${displayName(au)}${au.uid ? ` (${au.uid})` : ''}`;
152
+ }).join('\r\n') : undefined;
153
+ const successMessage = addedNames ? `The following ${roleString}${addedUserRoles.length > 1 ? 's were' : ' was'} successfully added${entityName ? ` to your ${entityName}` : ''}:\r\n${addedNames}` : undefined;
154
+ const existingNames = existingUserRoles.length > 0 ? existingUserRoles.map(eu => `${displayName(eu)}${eu.uid ? ` (${eu.uid})` : ''}`).join('\r\n') : undefined;
155
+ const hasMultipleExisting = existingUserRoles.length > 1;
156
+ const existingMessage = existingNames ? `The following ${hasMultipleExisting ? 'people were' : 'person was'} already ${entityName ? `in your ${entityName} as ${hasMultipleExisting ? '' : `${singularArticleString} `}` : hasMultipleExisting ? '' : `${singularArticleString} `}${roleString}${hasMultipleExisting ? 's' : ''}:\r\n${existingNames}` : undefined;
157
+ const blockedNames = !!blockedUserRoles && blockedUserRoles.length > 0 ? blockedUserRoles.map(eu => `${displayName(eu)}${eu.uid ? ` (${eu.uid})` : ''}`).join('\r\n') : undefined;
158
+ const hasMultipleBlocked = !!blockedUserRoles && blockedUserRoles.length > 1;
159
+ const blockedMessage = blockedNames ? `The following ${hasMultipleBlocked ? 'people' : 'person'} could not be added ${entityName ? `to your ${entityName}` : ''} as ${hasMultipleBlocked ? '' : `${singularArticleString} `}${roleString}${hasMultipleBlocked ? 's' : ''} because they already have a role:\r\n${blockedNames}` : undefined;
160
+ let failMessage = invalidIdentifiers.length > 0 ? `The following ${getDomainIdentifierTypePluralString()} are invalid:\r\n${invalidIdentifiers.join('\r\n')}` : undefined;
161
+ if (invalidDomainIdentifiers.length > 0) {
162
+ failMessage = `${failMessage ? `${failMessage}\r\n` : ''}The following ${getDomainIdentifierTypePluralString()} do not match the allowed domains of ${allowedDomains.replace(',', ', ')}:\r\n${invalidDomainIdentifiers.join('\r\n')}`;
163
+ }
164
+ this.setState({
165
+ shouldResetAddForm: true,
166
+ isAdding: false,
167
+ identifiersToAdd: undefined,
168
+ roleForAdd: undefined,
169
+ successMessage,
170
+ existingMessage,
171
+ blockedMessage,
172
+ failMessage
173
+ });
174
+
175
+ // only reload if we had some successful adds
176
+ if (addedNames) {
177
+ // update redux with added userRoles
178
+ const updatedModel = cloneDeep(model);
179
+ addedUserRoles.forEach(userRole => updatedModel[userRole.id] = cloneDeep(userRole));
180
+
181
+ // create a dummy action and call `prepareFetch` to get the final `modelPath`
182
+ const {
183
+ modelPath
184
+ } = prepareFetch({
185
+ type: MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
186
+ modelName,
187
+ pathParams
188
+ }, getEndpointMappings());
189
+ dispatchAction({
190
+ type: MODEL_FETCH_RESULT_ACTION_TYPE.FETCH_RESULT_RECEIVED,
191
+ modelPath,
192
+ data: updatedModel
193
+ });
194
+ if (onAdd) {
195
+ onAdd(addedUserRoles);
339
196
  }
340
- // removing
341
- if (prevModelStatus === modelStatus_1.MODEL_STATUS.DELETING && modelStatus !== modelStatus_1.MODEL_STATUS.DELETING) {
342
- this.didRemove(modelStatus === modelStatus_1.MODEL_STATUS.READY);
197
+ }
198
+ });
199
+ //#endregion Add Users
200
+ //#region Update
201
+ _defineProperty(this, "updateUserRole", (userRoleToUpdate, newRole) => {
202
+ this.setState({
203
+ isUpdating: true,
204
+ userRoleToUpdate,
205
+ roleForUpdate: newRole,
206
+ successMessage: undefined,
207
+ failMessage: undefined
208
+ });
209
+ this.props.update({
210
+ id: userRoleToUpdate.id,
211
+ body: {},
212
+ queryParams: {
213
+ roleName: newRole
343
214
  }
215
+ });
216
+ });
217
+ _defineProperty(this, "didUpdate", isSuccess => {
218
+ const {
219
+ entityName,
220
+ onUpdate
221
+ } = this.props;
222
+ const {
223
+ userRoleToUpdate,
224
+ roleForUpdate
225
+ } = this.state;
226
+ if (userRoleToUpdate === undefined) {
227
+ throw new Error('`didUpdate` was called without setting `userRoleToUpdate` in state');
228
+ }
229
+ if (roleForUpdate === undefined) {
230
+ throw new Error('`didUpdate` was called without setting `roleForUpdate` in state');
231
+ }
232
+ const name = displayName(userRoleToUpdate);
233
+ const roleString = this.textForRole(roleForUpdate, true);
234
+ const singularArticleString = this.singularArticleForRole(roleForUpdate);
235
+ if (!isSuccess) {
236
+ const failMessage = `Oops! There was an error updating ${name} to ${singularArticleString} ${roleString}${entityName ? ` in your ${entityName}` : ''}. Please try again.`;
237
+ this.setState({
238
+ isUpdating: false,
239
+ userRoleToUpdate: undefined,
240
+ roleForUpdate: undefined,
241
+ failMessage
242
+ });
243
+ return;
244
+ }
245
+ const successMessage = `${name} was successfully updated to ${singularArticleString} ${roleString}${entityName ? ` in your ${entityName}` : ''}.`;
246
+ if (onUpdate) {
247
+ onUpdate(userRoleToUpdate, roleForUpdate);
248
+ }
249
+ this.setState({
250
+ isUpdating: false,
251
+ userRoleToUpdate: undefined,
252
+ roleForUpdate: undefined,
253
+ successMessage
254
+ });
255
+ });
256
+ //#endregion Update
257
+ //#region Remove User
258
+ _defineProperty(this, "alertRemoveUserRole", userRoleToRemove => {
259
+ this.setState({
260
+ userRoleToRemove,
261
+ shouldShowRemoveDialog: true
262
+ });
263
+ });
264
+ _defineProperty(this, "removeUserRoleTitle", roleForRemove => {
265
+ return `Remove ${this.textForRole(roleForRemove)}`;
266
+ });
267
+ _defineProperty(this, "renderRemoveUserRoleDescription", (userRoleToRemove, warning) => {
268
+ const {
269
+ entity,
270
+ externalProviders
271
+ } = this.props;
272
+ const group = entity && 'externalGroups' in entity ? entity : undefined;
273
+ const roleString = this.textForRole(userRoleToRemove.role, true);
274
+ const singularArticleString = this.singularArticleForRole(userRoleToRemove.role);
275
+ const defaultWarning = warning ?? /*#__PURE__*/_jsxs("p", {
276
+ className: "ma0",
277
+ children: ["Are you sure you want to ", /*#__PURE__*/_jsxs("strong", {
278
+ children: ["remove ", displayName(userRoleToRemove)]
279
+ }), " as", ' ', singularArticleString, " ", roleString, "?"]
280
+ });
281
+ const externalGroupsToRemove = group?.externalGroups.filter(eg => eg.userId === userRoleToRemove.id);
282
+ if (!externalGroupsToRemove || externalGroupsToRemove.length === 0 || userRoleToRemove.role !== BASE_ROLE.GROUP_OWNER) {
283
+ return defaultWarning;
284
+ }
285
+ const externalGroupsToRemoveWithRosterSync = externalGroupsToRemove.filter(eg => externalProviders?.[eg.externalProviderId]?.rosterSyncEnabled);
286
+ return /*#__PURE__*/_jsxs("div", {
287
+ children: [/*#__PURE__*/_jsxs("p", {
288
+ children: ["The following course sections will be ", /*#__PURE__*/_jsx("strong", {
289
+ children: "disconnected"
290
+ }), ".", ' ', externalGroupsToRemoveWithRosterSync.length > 0 && /*#__PURE__*/_jsxs("span", {
291
+ children: ["Any automatically added students will be ", /*#__PURE__*/_jsx("strong", {
292
+ children: "removed"
293
+ }), "."]
294
+ })]
295
+ }), /*#__PURE__*/_jsx("ul", {
296
+ children: externalGroupsToRemove.map(eg => /*#__PURE__*/_jsx("li", {
297
+ children: eg.description
298
+ }, eg.id))
299
+ }), defaultWarning]
300
+ });
301
+ });
302
+ _defineProperty(this, "removeUserRole", shouldRemove => {
303
+ const {
304
+ userRoleToRemove
305
+ } = this.state;
306
+ if (userRoleToRemove === undefined) {
307
+ throw new Error('`removeUser` was called without setting `userRoleToRemove` in state');
308
+ }
309
+ this.setState({
310
+ isRemoving: shouldRemove,
311
+ shouldShowRemoveDialog: false,
312
+ successMessage: undefined,
313
+ failMessage: undefined,
314
+ // clear if cancelled
315
+ userRoleToRemove: shouldRemove ? userRoleToRemove : undefined
316
+ });
317
+ if (!shouldRemove) {
318
+ return;
319
+ }
320
+ this.props.delete({
321
+ id: userRoleToRemove.id
322
+ });
323
+ });
324
+ _defineProperty(this, "didRemove", isSuccess => {
325
+ const {
326
+ entityName,
327
+ onRemove
328
+ } = this.props;
329
+ const {
330
+ userRoleToRemove
331
+ } = this.state;
332
+ if (userRoleToRemove === undefined) {
333
+ throw new Error('`didRemove` was called without setting `userRoleToRemove` in state');
334
+ }
335
+ const name = displayName(userRoleToRemove);
336
+ if (!isSuccess) {
337
+ const failMessage = `Oops! There was an error removing ${name}${entityName ? ` from your ${entityName}` : ''}. Please try again.`;
338
+ this.setState({
339
+ isRemoving: false,
340
+ userRoleToRemove: undefined,
341
+ failMessage
342
+ });
343
+ return;
344
+ }
345
+ const successMessage = `${name} was successfully removed${entityName ? ` from your ${entityName}` : ''}.`;
346
+ if (onRemove) {
347
+ onRemove(userRoleToRemove);
348
+ }
349
+ this.setState({
350
+ isRemoving: false,
351
+ userRoleToRemove: undefined,
352
+ successMessage
353
+ });
354
+ });
355
+ }
356
+ componentDidMount() {
357
+ this.setStateFromProps(this.props);
358
+ const {
359
+ addUsersHookId
360
+ } = this.state;
361
+ noStoreHooks.unregisterNoStoreActionHook(addUsersHookId);
362
+ }
363
+ componentDidUpdate(prevProps) {
364
+ const {
365
+ modelArray: prevModelArray,
366
+ modelStatus: prevModelStatus
367
+ } = prevProps;
368
+ const {
369
+ modelArray,
370
+ modelStatus,
371
+ entityName
372
+ } = this.props;
373
+ if (!isEqual(prevModelArray, modelArray)) {
374
+ this.setStateFromProps(this.props);
375
+ }
376
+
377
+ // loading error
378
+ if ((prevModelStatus === MODEL_STATUS.UNINITIALIZED || prevModelStatus === MODEL_STATUS.LOADING) && modelStatus === MODEL_STATUS.ERROR) {
379
+ this.setState({
380
+ failMessage: `Oops! There was an error loading${entityName ? ` the people for your ${entityName}` : ''}.\r\nPlease try again.`
381
+ });
382
+ }
383
+
384
+ // updating
385
+ if (prevModelStatus === MODEL_STATUS.UPDATING && modelStatus !== MODEL_STATUS.UPDATING) {
386
+ this.didUpdate(modelStatus === MODEL_STATUS.READY);
344
387
  }
345
- //#endregion Remove User
346
- render() {
347
- const { modelStatus, canModify, readOnly, canDeleteSelf, allowMultipleRoles, isAddDisabled, isUpdateDisabled, isDeleteDisabled, defaultRole, requiredRole, roleDescriptions, addRoleExcludeList, entityName, renderTableDescription, renderAddDescription } = this.props;
348
- const { sortedUsers, isAdding, isUpdating, isRemoving, shouldResetAddForm, shouldShowRemoveDialog, userRoleToUpdate, userRoleToRemove, successMessage, existingMessage, blockedMessage, failMessage } = this.state;
349
- const addableRoleDescriptions = addRoleExcludeList
350
- ? (0, lodash_1.pickBy)(roleDescriptions, (_, key) => !addRoleExcludeList.includes(key))
351
- : roleDescriptions;
352
- const roles = Object.keys(roleDescriptions);
353
- const shouldRenderAsMutable = !readOnly && canModify;
354
- return ((0, jsx_runtime_1.jsxs)(jsx_runtime_1.Fragment, { children: [!!successMessage && ((0, jsx_runtime_1.jsx)(AlertWithIcon_1.AlertWithIcon, { id: "successMessageAlert", variant: "success", dismissible: true, onClose: () => this.setState({
355
- successMessage: undefined
356
- }), children: (0, jsx_runtime_1.jsx)("p", { className: "pre-wrap", children: successMessage }) })), !!existingMessage && ((0, jsx_runtime_1.jsx)(AlertWithIcon_1.AlertWithIcon, { id: "existingMessageAlert", variant: "info", dismissible: true, onClose: () => this.setState({
357
- existingMessage: undefined
358
- }), children: (0, jsx_runtime_1.jsx)("p", { className: "pre-wrap", children: existingMessage }) })), !!blockedMessage && ((0, jsx_runtime_1.jsx)(AlertWithIcon_1.AlertWithIcon, { id: "blockedMessageAlert", variant: "warning", dismissible: true, onClose: () => this.setState({
359
- blockedMessage: undefined
360
- }), children: (0, jsx_runtime_1.jsx)("p", { className: "pre-wrap", children: blockedMessage }) })), !!failMessage && ((0, jsx_runtime_1.jsx)(AlertWithIcon_1.AlertWithIcon, { id: "failMessageAlert", variant: "warning", dismissible: true, onClose: () => this.setState({
361
- failMessage: undefined
362
- }), children: (0, jsx_runtime_1.jsx)("p", { className: "pre-wrap", children: failMessage }) })), shouldRenderAsMutable && ((0, jsx_runtime_1.jsx)(Add_1.UserRolesAdd, { id: "entityUserRolesAdd", disabled: isAddDisabled, entityName: entityName, defaultRole: defaultRole, roleDescriptions: addableRoleDescriptions, renderAddDescription: renderAddDescription, textForRole: this.textForRole, isAddingUsersToRole: isAdding, shouldReset: shouldResetAddForm, addUserRoles: this.addUserRoles })), !!renderTableDescription && renderTableDescription(shouldRenderAsMutable), (0, jsx_runtime_1.jsx)(react_bootstrap_1.Row, { className: "mt3", children: (0, jsx_runtime_1.jsx)(react_bootstrap_1.Col, { xs: 12, children: modelStatus === modelStatus_1.MODEL_STATUS.LOADING ? ((0, jsx_runtime_1.jsx)(Loading_1.Loading, {})) : sortedUsers.length > 0 ? ((0, jsx_runtime_1.jsx)(Context_1.UserRolesContext.Provider, { value: {
363
- isUpdateDisabled: isUpdateDisabled || readOnly,
364
- isDeleteDisabled: isDeleteDisabled || readOnly,
365
- canModify: shouldRenderAsMutable,
366
- canDeleteSelf,
367
- allowMultipleRoles,
368
- roles,
369
- requiredRole,
370
- textForRole: this.textForRole,
371
- userRoleToUpdate,
372
- isUpdating,
373
- updateUserRole: this.updateUserRole,
374
- userRoleToRemove,
375
- isRemoving,
376
- removeUserRole: this.alertRemoveUserRole
377
- }, children: (0, jsx_runtime_1.jsx)(Table_1.UserRolesTable, { id: "entityUserRolesTable", users: sortedUsers }) })) : null }) }), !!userRoleToRemove && ((0, jsx_runtime_1.jsx)(AlertDialog_1.AlertDialog, { id: "removeUserAlert", isOpen: shouldShowRemoveDialog, title: this.removeUserRoleTitle(userRoleToRemove.role), description: this.renderRemoveUserRoleDescription(userRoleToRemove), onDestroy: () => this.removeUserRole(true), destroyText: "Yes, remove the user", onCancel: () => this.removeUserRole(false), cancelText: "No, I changed my mind" }))] }));
388
+
389
+ // removing
390
+ if (prevModelStatus === MODEL_STATUS.DELETING && modelStatus !== MODEL_STATUS.DELETING) {
391
+ this.didRemove(modelStatus === MODEL_STATUS.READY);
378
392
  }
393
+ }
394
+ //#endregion Remove User
395
+
396
+ render() {
397
+ const {
398
+ modelStatus,
399
+ canModify,
400
+ readOnly,
401
+ canDeleteSelf,
402
+ allowMultipleRoles,
403
+ isAddDisabled,
404
+ isUpdateDisabled,
405
+ isDeleteDisabled,
406
+ defaultRole,
407
+ requiredRole,
408
+ roleDescriptions,
409
+ addRoleExcludeList,
410
+ entityName,
411
+ renderTableDescription,
412
+ renderAddDescription
413
+ } = this.props;
414
+ const {
415
+ sortedUsers,
416
+ isAdding,
417
+ isUpdating,
418
+ isRemoving,
419
+ shouldResetAddForm,
420
+ shouldShowRemoveDialog,
421
+ userRoleToUpdate,
422
+ userRoleToRemove,
423
+ successMessage,
424
+ existingMessage,
425
+ blockedMessage,
426
+ failMessage
427
+ } = this.state;
428
+ const addableRoleDescriptions = addRoleExcludeList ? pickBy(roleDescriptions, (_, key) => !addRoleExcludeList.includes(key)) : roleDescriptions;
429
+ const roles = Object.keys(roleDescriptions);
430
+ const shouldRenderAsMutable = !readOnly && canModify;
431
+ return /*#__PURE__*/_jsxs(_Fragment, {
432
+ children: [!!successMessage && /*#__PURE__*/_jsx(AlertWithIcon, {
433
+ id: "successMessageAlert",
434
+ variant: "success",
435
+ dismissible: true,
436
+ onClose: () => this.setState({
437
+ successMessage: undefined
438
+ }),
439
+ children: /*#__PURE__*/_jsx("p", {
440
+ className: "pre-wrap",
441
+ children: successMessage
442
+ })
443
+ }), !!existingMessage && /*#__PURE__*/_jsx(AlertWithIcon, {
444
+ id: "existingMessageAlert",
445
+ variant: "info",
446
+ dismissible: true,
447
+ onClose: () => this.setState({
448
+ existingMessage: undefined
449
+ }),
450
+ children: /*#__PURE__*/_jsx("p", {
451
+ className: "pre-wrap",
452
+ children: existingMessage
453
+ })
454
+ }), !!blockedMessage && /*#__PURE__*/_jsx(AlertWithIcon, {
455
+ id: "blockedMessageAlert",
456
+ variant: "warning",
457
+ dismissible: true,
458
+ onClose: () => this.setState({
459
+ blockedMessage: undefined
460
+ }),
461
+ children: /*#__PURE__*/_jsx("p", {
462
+ className: "pre-wrap",
463
+ children: blockedMessage
464
+ })
465
+ }), !!failMessage && /*#__PURE__*/_jsx(AlertWithIcon, {
466
+ id: "failMessageAlert",
467
+ variant: "warning",
468
+ dismissible: true,
469
+ onClose: () => this.setState({
470
+ failMessage: undefined
471
+ }),
472
+ children: /*#__PURE__*/_jsx("p", {
473
+ className: "pre-wrap",
474
+ children: failMessage
475
+ })
476
+ }), shouldRenderAsMutable && /*#__PURE__*/_jsx(UserRolesAdd, {
477
+ id: "entityUserRolesAdd",
478
+ disabled: isAddDisabled,
479
+ entityName: entityName,
480
+ defaultRole: defaultRole,
481
+ roleDescriptions: addableRoleDescriptions,
482
+ renderAddDescription: renderAddDescription,
483
+ textForRole: this.textForRole,
484
+ isAddingUsersToRole: isAdding,
485
+ shouldReset: shouldResetAddForm,
486
+ addUserRoles: this.addUserRoles
487
+ }), !!renderTableDescription && renderTableDescription(shouldRenderAsMutable), /*#__PURE__*/_jsx(Row, {
488
+ className: "mt3",
489
+ children: /*#__PURE__*/_jsx(Col, {
490
+ xs: 12,
491
+ children: modelStatus === MODEL_STATUS.LOADING ? /*#__PURE__*/_jsx(Loading, {}) : sortedUsers.length > 0 ? /*#__PURE__*/_jsx(UserRolesContext.Provider, {
492
+ value: {
493
+ isUpdateDisabled: isUpdateDisabled || readOnly,
494
+ isDeleteDisabled: isDeleteDisabled || readOnly,
495
+ canModify: shouldRenderAsMutable,
496
+ canDeleteSelf,
497
+ allowMultipleRoles,
498
+ roles,
499
+ requiredRole,
500
+ textForRole: this.textForRole,
501
+ userRoleToUpdate,
502
+ isUpdating,
503
+ updateUserRole: this.updateUserRole,
504
+ userRoleToRemove,
505
+ isRemoving,
506
+ removeUserRole: this.alertRemoveUserRole
507
+ },
508
+ children: /*#__PURE__*/_jsx(UserRolesTable, {
509
+ id: "entityUserRolesTable",
510
+ users: sortedUsers
511
+ })
512
+ }) : null
513
+ })
514
+ }), !!userRoleToRemove && /*#__PURE__*/_jsx(AlertDialog, {
515
+ id: "removeUserAlert",
516
+ isOpen: shouldShowRemoveDialog,
517
+ title: this.removeUserRoleTitle(userRoleToRemove.role),
518
+ description: this.renderRemoveUserRoleDescription(userRoleToRemove),
519
+ onDestroy: () => this.removeUserRole(true),
520
+ destroyText: "Yes, remove the user",
521
+ onCancel: () => this.removeUserRole(false),
522
+ cancelText: "No, I changed my mind"
523
+ })]
524
+ });
525
+ }
379
526
  }
380
- exports.UserRoles = UserRoles;
381
- const mapStateToProps = (state, ownProps) => {
382
- const canModifyGlobally = (0, baseActivity_1.canPerformActivityGlobally)(ownProps.modifyUserRoleActivityName, (0, baseActivity_1.defaultOptions)(state));
383
- const canModify = canModifyGlobally ||
384
- (ownProps.entity
385
- ? (0, baseActivity_1.canPerformActivityOnEntity)(ownProps.modifyUserRoleActivityName, (0, baseActivity_1.defaultOptions)(state, ownProps, 'entity'))
386
- : false);
387
- return {
388
- canModify,
389
- canDeleteSelf: canModify ||
390
- (ownProps.entity && ownProps.deleteOwnUserRoleActivityName
391
- ? (0, baseActivity_1.canPerformActivityOnEntity)(ownProps.deleteOwnUserRoleActivityName, (0, baseActivity_1.defaultOptions)(state, ownProps, 'entity'))
392
- : false)
393
- };
527
+ export const mapStateToProps = (state, ownProps) => {
528
+ const canModifyGlobally = canPerformActivityGlobally(ownProps.modifyUserRoleActivityName, defaultOptions(state));
529
+ const canModify = canModifyGlobally || (ownProps.entity ? canPerformActivityOnEntity(ownProps.modifyUserRoleActivityName, defaultOptions(state, ownProps, 'entity')) : false);
530
+ return {
531
+ canModify,
532
+ canDeleteSelf: canModify || (ownProps.entity && ownProps.deleteOwnUserRoleActivityName ? canPerformActivityOnEntity(ownProps.deleteOwnUserRoleActivityName, defaultOptions(state, ownProps, 'entity')) : false)
533
+ };
394
534
  };
395
- exports.mapStateToProps = mapStateToProps;
396
- exports.default = (0, react_redux_1.connect)(exports.mapStateToProps)(UserRoles);
535
+ export default connect(mapStateToProps)(UserRoles);
536
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["cloneDeep","isEqual","pickBy","React","Component","Col","Row","connect","v4","uuidv4","AlertDialog","Loading","UserRolesAdd","UserRolesTable","getEndpointMappings","BASE_ROLE","MODEL_STATUS","dispatchAction","dispatchModelFetchRequest","MODEL_FETCH_REQUEST_ACTION_TYPE","MODEL_FETCH_RESULT_ACTION_TYPE","noStoreHooks","canPerformActivityGlobally","canPerformActivityOnEntity","defaultOptions","singularArticleForBaseRole","textForBaseRole","getDomainIdentifierTypePluralString","getRootModelName","isFetchErrorData","prepareFetch","sortByNames","displayName","convertToUserWithRoles","getUserId","AlertWithIcon","UserRolesContext","jsxs","_jsxs","jsx","_jsx","Fragment","_Fragment","UserRoles","constructor","_this","arguments","this","_defineProperty","sortedUsers","addUsersHookId","isAdding","isUpdating","isRemoving","shouldResetAddForm","shouldShowRemoveDialog","props","modelArray","userRoles","filterUsers","usersWithRoles","Object","values","reduce","result","userRole","userId","userWithRoles","roles","push","filteredUsers","sort","setState","role","lowercase","length","undefined","lowercaseTextForRole","roleString","textForRole","toLowerCase","singularArticleForRole","identifiers","entity","identifiersToAdd","roleForAdd","successMessage","failMessage","existingMessage","blockedMessage","state","registerNoStoreActionHook","data","unregisterNoStoreActionHook","didAdd","rootModelName","modelName","pathParams","id","body","entityId","roleName","noStore","guid","entityName","onAdd","model","Error","join","addedUserRoles","existingUserRoles","blockedUserRoles","invalidIdentifiers","allowedDomains","invalidDomainIdentifiers","singularArticleString","addedNames","map","au","uid","existingNames","eu","hasMultipleExisting","blockedNames","hasMultipleBlocked","replace","updatedModel","forEach","modelPath","type","FETCH_REQUEST","FETCH_RESULT_RECEIVED","userRoleToUpdate","newRole","roleForUpdate","update","queryParams","isSuccess","onUpdate","name","userRoleToRemove","roleForRemove","warning","externalProviders","group","defaultWarning","className","children","externalGroupsToRemove","externalGroups","filter","eg","GROUP_OWNER","externalGroupsToRemoveWithRosterSync","externalProviderId","rosterSyncEnabled","description","shouldRemove","delete","onRemove","componentDidMount","setStateFromProps","componentDidUpdate","prevProps","prevModelArray","modelStatus","prevModelStatus","UNINITIALIZED","LOADING","ERROR","UPDATING","didUpdate","READY","DELETING","didRemove","render","canModify","readOnly","canDeleteSelf","allowMultipleRoles","isAddDisabled","isUpdateDisabled","isDeleteDisabled","defaultRole","requiredRole","roleDescriptions","addRoleExcludeList","renderTableDescription","renderAddDescription","addableRoleDescriptions","_","key","includes","keys","shouldRenderAsMutable","variant","dismissible","onClose","disabled","isAddingUsersToRole","shouldReset","addUserRoles","xs","Provider","value","updateUserRole","removeUserRole","alertRemoveUserRole","users","isOpen","title","removeUserRoleTitle","renderRemoveUserRoleDescription","onDestroy","destroyText","onCancel","cancelText","mapStateToProps","ownProps","canModifyGlobally","modifyUserRoleActivityName","deleteOwnUserRoleActivityName"],"sources":["../../../src/components/UserRoles/index.tsx"],"sourcesContent":["import { cloneDeep, Dictionary, isEqual, pickBy } from 'lodash'\nimport React, { Component } from 'react'\nimport { Col, Row } from 'react-bootstrap'\nimport { connect } from 'react-redux'\nimport { v4 as uuidv4 } from 'uuid'\nimport { AlertDialog } from '../../components/AlertDialog'\nimport { Loading } from '../../components/Loading'\nimport { UserRolesAdd } from '../../components/UserRoles/Add'\nimport { UserRolesTable } from '../../components/UserRoles/Table'\nimport { getEndpointMappings } from '../../constants'\nimport { BASE_ROLE } from '../../constants/baseRole'\nimport { MODEL_STATUS } from '../../constants/modelStatus'\nimport { dispatchAction, dispatchModelFetchRequest } from '../../redux/actionCreator'\nimport {\n\tMODEL_FETCH_REQUEST_ACTION_TYPE,\n\tMODEL_FETCH_RESULT_ACTION_TYPE,\n\tModelFetchResultAction\n} from '../../redux/actions'\nimport { noStoreHooks } from '../../redux/sagas/noStoreSaga'\nimport {\n\tBaseReduxState,\n\tExternalProvider,\n\tFetchErrorData,\n\tGroup,\n\tModel,\n\tModelCollection,\n\tRoleDescriptions,\n\tUser,\n\tUserRole,\n\tUserWithRoles\n} from '../../types'\nimport { canPerformActivityGlobally, canPerformActivityOnEntity, defaultOptions } from '../../utils/baseActivity'\nimport { singularArticleForBaseRole, textForBaseRole } from '../../utils/baseRole'\nimport { getDomainIdentifierTypePluralString } from '../../utils/domainIdentifier'\nimport { getRootModelName } from '../../utils/entityUserRole'\nimport { isFetchErrorData, prepareFetch } from '../../utils/fetch'\nimport { sortByNames } from '../../utils/sort'\nimport { displayName } from '../../utils/user'\nimport { convertToUserWithRoles, getUserId } from '../../utils/userRole'\nimport { AlertWithIcon } from '../AlertWithIcon'\nimport { CollectionComponentWrappedProps } from '../HOC/CollectionComponent'\nimport { UserRolesContext } from './Context'\n\nexport interface UserRolesReduxProps {\n\t/** Is the current user allowed to add, update, and delete user roles? */\n\tcanModify?: boolean\n\t/** Is the current user allowed to delete their own user role? */\n\tcanDeleteSelf?: boolean\n}\n\nexport interface UserRolesOwnProps extends CollectionComponentWrappedProps<UserRole> {\n\t/** Should the component prevent modifications, independent of permissions? */\n\treadOnly?: boolean\n\t/** Is the current user allowed to delete their own user role? Overrides the value from redux. */\n\tcanDeleteSelf?: boolean\n\n\t/** If multiple roles allowed, each role is displayed per user, otherwise a select dropdown is displayed. */\n\tallowMultipleRoles?: boolean\n\t/** disable adding new user roles */\n\tisAddDisabled?: boolean\n\t/** disable all user role select dropdowns, if any */\n\tisUpdateDisabled?: boolean\n\t/** disable all user role delete buttons */\n\tisDeleteDisabled?: boolean\n\n\t/** The activity the current user must have in order to add, update, or delete user roles. */\n\tmodifyUserRoleActivityName: string\n\t/** (Optional) The activity the current user must have in order to delete their own user roles, if they cannot modify all. */\n\tdeleteOwnUserRoleActivityName?: string\n\t/** The role that will be shown first in the list of available roles. */\n\tdefaultRole: string\n\t/** (Optional) If provided, the last user role with the required role will be prevented from being removed. */\n\trequiredRole?: string\n\t/** A dictionary of all possible roles and their descriptions. */\n\troleDescriptions: RoleDescriptions\n\t/** (Optional) list of roles that should be excluded from the add options. */\n\taddRoleExcludeList?: string[]\n\n\t/** (Optional) The entity that owns the user roles, when targeting EntityUserRoles */\n\tentity?: Model\n\t/** (Optional) A user-friendly name for the entity type, when targeting EntityUserRoles  */\n\tentityName?: string\n\t/** (Optional) set if `renderRemoveUserRoleDescription` should check for roster sync */\n\texternalProviders?: ModelCollection<ExternalProvider>\n\n\t/** (Optional) Allow users to be excluded from being displayed in the table */\n\tfilterUsers?: (users: UserWithRoles[]) => UserWithRoles[]\n\t/** (Optional) Callback that is called after user roles are added */\n\tonAdd?: (userRoles: UserRole[]) => void\n\t/** (Optional) Callback that is called after a user role is updated */\n\tonUpdate?: (userRole: UserRole, role: string) => void\n\t/** (Optional) Callback that is called after a user role is removed */\n\tonRemove?: (userRole: UserRole) => void\n\t/** (Optional) Render custom components between the Add and Table components */\n\trenderTableDescription?: (canModify?: boolean) => React.JSX.Element\n\t/** (Optional) Render custom components at the start of the Add component */\n\trenderAddDescription?: (entityName?: string) => React.JSX.Element\n\t/** (Optional) Provide custom user-friendly names for displayed roles */\n\ttextForRole?: (role: string) => string\n\t/** (Optional) Provide custom user-friendly names for displayed roles that require custom casing */\n\tlowercaseTextForRole?: (role: string) => string\n\t/** (Optional) Provide custom user-friendly articles (\"a\" vs. \"an\") for displayed roles */\n\tsingularArticleForRole?: (role: string) => string\n}\n\nexport interface UserRolesProps extends UserRolesReduxProps, UserRolesOwnProps {}\n\ninterface UserRolesState {\n\tsortedUsers: UserWithRoles[]\n\t// add\n\taddUsersHookId: string\n\tisAdding: boolean\n\tidentifiersToAdd?: string[]\n\troleForAdd?: string\n\tshouldResetAddForm: boolean\n\t// update\n\tuserRoleToUpdate?: UserRole\n\troleForUpdate?: string\n\tisUpdating: boolean\n\t// remove\n\tshouldShowRemoveDialog: boolean\n\tuserRoleToRemove?: UserRole\n\tisRemoving: boolean\n\t// messages\n\tsuccessMessage?: string\n\texistingMessage?: string\n\tblockedMessage?: string\n\tfailMessage?: string\n}\n\nexport interface AddBusinessModel {\n\taddedUserRoles: UserRole[]\n\texistingUserRoles: UserRole[]\n\tblockedUserRoles?: UserRole[]\n\tinvalidIdentifiers: string[]\n\tallowedDomains: string\n\tinvalidDomainIdentifiers: string[]\n}\n\n/**\n * Component used to manage either UserRoles (global roles) or EntityUserRoles.\n */\nexport class UserRoles extends Component<UserRolesProps, UserRolesState> {\n\tstate: UserRolesState = {\n\t\tsortedUsers: [],\n\t\t// add\n\t\taddUsersHookId: uuidv4(),\n\t\tisAdding: false,\n\t\tisUpdating: false,\n\t\tisRemoving: false,\n\t\tshouldResetAddForm: false,\n\t\tshouldShowRemoveDialog: false\n\t}\n\n\tcomponentDidMount() {\n\t\tthis.setStateFromProps(this.props)\n\t\tconst { addUsersHookId } = this.state\n\t\tnoStoreHooks.unregisterNoStoreActionHook(addUsersHookId)\n\t}\n\n\tcomponentDidUpdate(prevProps: UserRolesProps) {\n\t\tconst { modelArray: prevModelArray, modelStatus: prevModelStatus } = prevProps\n\t\tconst { modelArray, modelStatus, entityName } = this.props\n\n\t\tif (!isEqual(prevModelArray, modelArray)) {\n\t\t\tthis.setStateFromProps(this.props)\n\t\t}\n\n\t\t// loading error\n\t\tif (\n\t\t\t(prevModelStatus === MODEL_STATUS.UNINITIALIZED || prevModelStatus === MODEL_STATUS.LOADING) &&\n\t\t\tmodelStatus === MODEL_STATUS.ERROR\n\t\t) {\n\t\t\tthis.setState({\n\t\t\t\tfailMessage: `Oops! There was an error loading${\n\t\t\t\t\tentityName ? ` the people for your ${entityName}` : ''\n\t\t\t\t}.\\r\\nPlease try again.`\n\t\t\t})\n\t\t}\n\n\t\t// updating\n\t\tif (prevModelStatus === MODEL_STATUS.UPDATING && modelStatus !== MODEL_STATUS.UPDATING) {\n\t\t\tthis.didUpdate(modelStatus === MODEL_STATUS.READY)\n\t\t}\n\n\t\t// removing\n\t\tif (prevModelStatus === MODEL_STATUS.DELETING && modelStatus !== MODEL_STATUS.DELETING) {\n\t\t\tthis.didRemove(modelStatus === MODEL_STATUS.READY)\n\t\t}\n\t}\n\n\tsetStateFromProps = (props: UserRolesProps) => {\n\t\tconst { modelArray: userRoles, filterUsers } = props\n\t\tconst usersWithRoles = Object.values(\n\t\t\tuserRoles.reduce((result: Dictionary<UserWithRoles>, userRole: UserRole) => {\n\t\t\t\tconst userId = getUserId(userRole)\n\t\t\t\tlet userWithRoles = result[userId] as UserWithRoles | undefined\n\t\t\t\tif (!userWithRoles) {\n\t\t\t\t\tuserWithRoles = convertToUserWithRoles(userRole)\n\t\t\t\t} else {\n\t\t\t\t\tuserWithRoles.roles.push(userRole)\n\t\t\t\t}\n\t\t\t\tresult[userId] = userWithRoles\n\t\t\t\treturn result\n\t\t\t}, {})\n\t\t)\n\t\tconst filteredUsers = filterUsers ? filterUsers(usersWithRoles) : usersWithRoles\n\t\tconst sortedUsers = filteredUsers.sort(sortByNames)\n\t\tthis.setState({\n\t\t\tsortedUsers\n\t\t})\n\t}\n\n\ttextForRole = (role: string, lowercase = false) => {\n\t\tif (lowercase && this.props.lowercaseTextForRole) {\n\t\t\treturn this.props.lowercaseTextForRole(role)\n\t\t}\n\t\tconst roleString = this.props.textForRole ? this.props.textForRole(role) : textForBaseRole(role)\n\t\treturn lowercase ? roleString.toLowerCase() : roleString\n\t}\n\n\tsingularArticleForRole = (role: string) => {\n\t\treturn this.props.singularArticleForRole\n\t\t\t? this.props.singularArticleForRole(role)\n\t\t\t: singularArticleForBaseRole(role)\n\t}\n\n\t//#region Add Users\n\n\taddUserRoles = (identifiers: string[], role: string) => {\n\t\tconst { entity } = this.props\n\t\tthis.setState({\n\t\t\tidentifiersToAdd: identifiers,\n\t\t\troleForAdd: role,\n\t\t\tsuccessMessage: undefined,\n\t\t\tfailMessage: undefined,\n\t\t\texistingMessage: undefined,\n\t\t\tblockedMessage: undefined,\n\t\t\tshouldResetAddForm: false,\n\t\t\tisAdding: true\n\t\t})\n\t\tconst { addUsersHookId } = this.state\n\t\tnoStoreHooks.registerNoStoreActionHook(addUsersHookId, (data: AddBusinessModel | FetchErrorData | null) => {\n\t\t\tnoStoreHooks.unregisterNoStoreActionHook(addUsersHookId)\n\t\t\tthis.didAdd(data)\n\t\t})\n\t\tconst rootModelName = getRootModelName(this.props)\n\t\tdispatchModelFetchRequest({\n\t\t\tmodelName: `${rootModelName ? `${rootModelName}.` : ''}addUserRoles`,\n\t\t\tpathParams: entity ? [entity.id] : undefined,\n\t\t\tbody: {\n\t\t\t\tentityId: entity ? entity.id : undefined,\n\t\t\t\tidentifiers: identifiers,\n\t\t\t\troleName: role\n\t\t\t},\n\t\t\tnoStore: true,\n\t\t\tguid: addUsersHookId\n\t\t})\n\t}\n\n\tdidAdd = (data: AddBusinessModel | FetchErrorData | null) => {\n\t\tconst { entityName, onAdd, model, modelName, pathParams } = this.props\n\t\tconst { roleForAdd, identifiersToAdd } = this.state\n\t\tif (!roleForAdd || !identifiersToAdd || identifiersToAdd.length === 0) {\n\t\t\tthrow new Error('didAdd was called in the incorrect state')\n\t\t}\n\n\t\tif (!data || isFetchErrorData(data)) {\n\t\t\tconst failMessage = `The following identifiers were not added${\n\t\t\t\tentityName ? ` to your ${entityName}` : ''\n\t\t\t}:\\r\\n${identifiersToAdd.join('\\r\\n')}`\n\n\t\t\tthis.setState({\n\t\t\t\tshouldResetAddForm: true,\n\t\t\t\tisAdding: false,\n\t\t\t\tidentifiersToAdd: undefined,\n\t\t\t\troleForAdd: undefined,\n\t\t\t\tfailMessage\n\t\t\t})\n\t\t\treturn\n\t\t}\n\n\t\tconst {\n\t\t\taddedUserRoles,\n\t\t\texistingUserRoles,\n\t\t\tblockedUserRoles,\n\t\t\tinvalidIdentifiers,\n\t\t\tallowedDomains,\n\t\t\tinvalidDomainIdentifiers\n\t\t} = data\n\t\tconst roleString = this.textForRole(roleForAdd, true)\n\t\tconst singularArticleString = this.singularArticleForRole(roleForAdd)\n\n\t\tconst addedNames =\n\t\t\taddedUserRoles.length > 0\n\t\t\t\t? addedUserRoles\n\t\t\t\t\t\t.map((au: User) => {\n\t\t\t\t\t\t\treturn `${displayName(au)}${au.uid ? ` (${au.uid})` : ''}`\n\t\t\t\t\t\t})\n\t\t\t\t\t\t.join('\\r\\n')\n\t\t\t\t: undefined\n\n\t\tconst successMessage = addedNames\n\t\t\t? `The following ${roleString}${addedUserRoles.length > 1 ? 's were' : ' was'} successfully added${\n\t\t\t\t\tentityName ? ` to your ${entityName}` : ''\n\t\t\t\t}:\\r\\n${addedNames}`\n\t\t\t: undefined\n\n\t\tconst existingNames =\n\t\t\texistingUserRoles.length > 0\n\t\t\t\t? existingUserRoles.map((eu: User) => `${displayName(eu)}${eu.uid ? ` (${eu.uid})` : ''}`).join('\\r\\n')\n\t\t\t\t: undefined\n\n\t\tconst hasMultipleExisting = existingUserRoles.length > 1\n\t\tconst existingMessage = existingNames\n\t\t\t? `The following ${hasMultipleExisting ? 'people were' : 'person was'} already ${\n\t\t\t\t\tentityName\n\t\t\t\t\t\t? `in your ${entityName} as ${hasMultipleExisting ? '' : `${singularArticleString} `}`\n\t\t\t\t\t\t: hasMultipleExisting\n\t\t\t\t\t\t\t? ''\n\t\t\t\t\t\t\t: `${singularArticleString} `\n\t\t\t\t}${roleString}${hasMultipleExisting ? 's' : ''}:\\r\\n${existingNames}`\n\t\t\t: undefined\n\n\t\tconst blockedNames =\n\t\t\t!!blockedUserRoles && blockedUserRoles.length > 0\n\t\t\t\t? blockedUserRoles.map((eu: User) => `${displayName(eu)}${eu.uid ? ` (${eu.uid})` : ''}`).join('\\r\\n')\n\t\t\t\t: undefined\n\t\tconst hasMultipleBlocked = !!blockedUserRoles && blockedUserRoles.length > 1\n\t\tconst blockedMessage = blockedNames\n\t\t\t? `The following ${hasMultipleBlocked ? 'people' : 'person'} could not be added ${\n\t\t\t\t\tentityName ? `to your ${entityName}` : ''\n\t\t\t\t} as ${hasMultipleBlocked ? '' : `${singularArticleString} `}${roleString}${\n\t\t\t\t\thasMultipleBlocked ? 's' : ''\n\t\t\t\t} because they already have a role:\\r\\n${blockedNames}`\n\t\t\t: undefined\n\n\t\tlet failMessage =\n\t\t\tinvalidIdentifiers.length > 0\n\t\t\t\t? `The following ${getDomainIdentifierTypePluralString()} are invalid:\\r\\n${invalidIdentifiers.join(\n\t\t\t\t\t\t'\\r\\n'\n\t\t\t\t\t)}`\n\t\t\t\t: undefined\n\n\t\tif (invalidDomainIdentifiers.length > 0) {\n\t\t\tfailMessage = `${\n\t\t\t\tfailMessage ? `${failMessage}\\r\\n` : ''\n\t\t\t}The following ${getDomainIdentifierTypePluralString()} do not match the allowed domains of ${allowedDomains.replace(\n\t\t\t\t',',\n\t\t\t\t', '\n\t\t\t)}:\\r\\n${invalidDomainIdentifiers.join('\\r\\n')}`\n\t\t}\n\n\t\tthis.setState({\n\t\t\tshouldResetAddForm: true,\n\t\t\tisAdding: false,\n\t\t\tidentifiersToAdd: undefined,\n\t\t\troleForAdd: undefined,\n\t\t\tsuccessMessage,\n\t\t\texistingMessage,\n\t\t\tblockedMessage,\n\t\t\tfailMessage\n\t\t})\n\n\t\t// only reload if we had some successful adds\n\t\tif (addedNames) {\n\t\t\t// update redux with added userRoles\n\t\t\tconst updatedModel = cloneDeep(model)\n\t\t\taddedUserRoles.forEach(userRole => (updatedModel[userRole.id] = cloneDeep(userRole)))\n\n\t\t\t// create a dummy action and call `prepareFetch` to get the final `modelPath`\n\t\t\tconst { modelPath } = prepareFetch(\n\t\t\t\t{\n\t\t\t\t\ttype: MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,\n\t\t\t\t\tmodelName,\n\t\t\t\t\tpathParams\n\t\t\t\t},\n\t\t\t\tgetEndpointMappings()\n\t\t\t)\n\t\t\tdispatchAction<ModelFetchResultAction>({\n\t\t\t\ttype: MODEL_FETCH_RESULT_ACTION_TYPE.FETCH_RESULT_RECEIVED,\n\t\t\t\tmodelPath,\n\t\t\t\tdata: updatedModel\n\t\t\t})\n\n\t\t\tif (onAdd) {\n\t\t\t\tonAdd(addedUserRoles)\n\t\t\t}\n\t\t}\n\t}\n\n\t//#endregion Add Users\n\n\t//#region Update\n\n\tupdateUserRole = (userRoleToUpdate: UserRole, newRole: string) => {\n\t\tthis.setState({\n\t\t\tisUpdating: true,\n\t\t\tuserRoleToUpdate,\n\t\t\troleForUpdate: newRole,\n\t\t\tsuccessMessage: undefined,\n\t\t\tfailMessage: undefined\n\t\t})\n\t\tthis.props.update({\n\t\t\tid: userRoleToUpdate.id,\n\t\t\tbody: {},\n\t\t\tqueryParams: {\n\t\t\t\troleName: newRole\n\t\t\t}\n\t\t})\n\t}\n\n\tdidUpdate = (isSuccess: boolean) => {\n\t\tconst { entityName, onUpdate } = this.props\n\t\tconst { userRoleToUpdate, roleForUpdate } = this.state\n\t\tif (userRoleToUpdate === undefined) {\n\t\t\tthrow new Error('`didUpdate` was called without setting `userRoleToUpdate` in state')\n\t\t}\n\t\tif (roleForUpdate === undefined) {\n\t\t\tthrow new Error('`didUpdate` was called without setting `roleForUpdate` in state')\n\t\t}\n\n\t\tconst name = displayName(userRoleToUpdate)\n\t\tconst roleString = this.textForRole(roleForUpdate, true)\n\t\tconst singularArticleString = this.singularArticleForRole(roleForUpdate)\n\n\t\tif (!isSuccess) {\n\t\t\tconst failMessage = `Oops! There was an error updating ${name} to ${singularArticleString} ${roleString}${\n\t\t\t\tentityName ? ` in your ${entityName}` : ''\n\t\t\t}. Please try again.`\n\t\t\tthis.setState({\n\t\t\t\tisUpdating: false,\n\t\t\t\tuserRoleToUpdate: undefined,\n\t\t\t\troleForUpdate: undefined,\n\t\t\t\tfailMessage\n\t\t\t})\n\t\t\treturn\n\t\t}\n\n\t\tconst successMessage = `${name} was successfully updated to ${singularArticleString} ${roleString}${\n\t\t\tentityName ? ` in your ${entityName}` : ''\n\t\t}.`\n\n\t\tif (onUpdate) {\n\t\t\tonUpdate(userRoleToUpdate, roleForUpdate)\n\t\t}\n\n\t\tthis.setState({\n\t\t\tisUpdating: false,\n\t\t\tuserRoleToUpdate: undefined,\n\t\t\troleForUpdate: undefined,\n\t\t\tsuccessMessage\n\t\t})\n\t}\n\n\t//#endregion Update\n\n\t//#region Remove User\n\n\talertRemoveUserRole = (userRoleToRemove: UserRole) => {\n\t\tthis.setState({\n\t\t\tuserRoleToRemove,\n\t\t\tshouldShowRemoveDialog: true\n\t\t})\n\t}\n\n\tremoveUserRoleTitle = (roleForRemove: string) => {\n\t\treturn `Remove ${this.textForRole(roleForRemove)}`\n\t}\n\n\trenderRemoveUserRoleDescription = (userRoleToRemove: UserRole, warning?: React.JSX.Element) => {\n\t\tconst { entity, externalProviders } = this.props\n\t\tconst group = entity && 'externalGroups' in entity ? (entity as Group) : undefined\n\n\t\tconst roleString = this.textForRole(userRoleToRemove.role, true)\n\t\tconst singularArticleString = this.singularArticleForRole(userRoleToRemove.role)\n\t\tconst defaultWarning = warning ?? (\n\t\t\t<p className=\"ma0\">\n\t\t\t\tAre you sure you want to <strong>remove {displayName(userRoleToRemove)}</strong> as{' '}\n\t\t\t\t{singularArticleString} {roleString}?\n\t\t\t</p>\n\t\t)\n\n\t\tconst externalGroupsToRemove = group?.externalGroups.filter(eg => eg.userId === userRoleToRemove.id)\n\n\t\tif (\n\t\t\t!externalGroupsToRemove ||\n\t\t\texternalGroupsToRemove.length === 0 ||\n\t\t\tuserRoleToRemove.role !== BASE_ROLE.GROUP_OWNER\n\t\t) {\n\t\t\treturn defaultWarning\n\t\t}\n\n\t\tconst externalGroupsToRemoveWithRosterSync = externalGroupsToRemove.filter(\n\t\t\teg => externalProviders?.[eg.externalProviderId]?.rosterSyncEnabled\n\t\t)\n\n\t\treturn (\n\t\t\t<div>\n\t\t\t\t<p>\n\t\t\t\t\tThe following course sections will be <strong>disconnected</strong>.{' '}\n\t\t\t\t\t{externalGroupsToRemoveWithRosterSync.length > 0 && (\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tAny automatically added students will be <strong>removed</strong>.\n\t\t\t\t\t\t</span>\n\t\t\t\t\t)}\n\t\t\t\t</p>\n\t\t\t\t<ul>\n\t\t\t\t\t{externalGroupsToRemove.map(eg => (\n\t\t\t\t\t\t<li key={eg.id}>{eg.description}</li>\n\t\t\t\t\t))}\n\t\t\t\t</ul>\n\t\t\t\t{defaultWarning}\n\t\t\t</div>\n\t\t)\n\t}\n\n\tremoveUserRole = (shouldRemove: boolean) => {\n\t\tconst { userRoleToRemove } = this.state\n\t\tif (userRoleToRemove === undefined) {\n\t\t\tthrow new Error('`removeUser` was called without setting `userRoleToRemove` in state')\n\t\t}\n\t\tthis.setState({\n\t\t\tisRemoving: shouldRemove,\n\t\t\tshouldShowRemoveDialog: false,\n\t\t\tsuccessMessage: undefined,\n\t\t\tfailMessage: undefined,\n\t\t\t// clear if cancelled\n\t\t\tuserRoleToRemove: shouldRemove ? userRoleToRemove : undefined\n\t\t})\n\t\tif (!shouldRemove) {\n\t\t\treturn\n\t\t}\n\t\tthis.props.delete({\n\t\t\tid: userRoleToRemove.id\n\t\t})\n\t}\n\n\tdidRemove = (isSuccess: boolean) => {\n\t\tconst { entityName, onRemove } = this.props\n\t\tconst { userRoleToRemove } = this.state\n\t\tif (userRoleToRemove === undefined) {\n\t\t\tthrow new Error('`didRemove` was called without setting `userRoleToRemove` in state')\n\t\t}\n\n\t\tconst name = displayName(userRoleToRemove)\n\t\tif (!isSuccess) {\n\t\t\tconst failMessage = `Oops! There was an error removing ${name}${\n\t\t\t\tentityName ? ` from your ${entityName}` : ''\n\t\t\t}. Please try again.`\n\t\t\tthis.setState({\n\t\t\t\tisRemoving: false,\n\t\t\t\tuserRoleToRemove: undefined,\n\t\t\t\tfailMessage\n\t\t\t})\n\t\t\treturn\n\t\t}\n\n\t\tconst successMessage = `${name} was successfully removed${entityName ? ` from your ${entityName}` : ''}.`\n\n\t\tif (onRemove) {\n\t\t\tonRemove(userRoleToRemove)\n\t\t}\n\n\t\tthis.setState({\n\t\t\tisRemoving: false,\n\t\t\tuserRoleToRemove: undefined,\n\t\t\tsuccessMessage\n\t\t})\n\t}\n\n\t//#endregion Remove User\n\n\trender() {\n\t\tconst {\n\t\t\tmodelStatus,\n\t\t\tcanModify,\n\t\t\treadOnly,\n\t\t\tcanDeleteSelf,\n\t\t\tallowMultipleRoles,\n\t\t\tisAddDisabled,\n\t\t\tisUpdateDisabled,\n\t\t\tisDeleteDisabled,\n\t\t\tdefaultRole,\n\t\t\trequiredRole,\n\t\t\troleDescriptions,\n\t\t\taddRoleExcludeList,\n\t\t\tentityName,\n\t\t\trenderTableDescription,\n\t\t\trenderAddDescription\n\t\t} = this.props\n\t\tconst {\n\t\t\tsortedUsers,\n\t\t\tisAdding,\n\t\t\tisUpdating,\n\t\t\tisRemoving,\n\t\t\tshouldResetAddForm,\n\t\t\tshouldShowRemoveDialog,\n\t\t\tuserRoleToUpdate,\n\t\t\tuserRoleToRemove,\n\t\t\tsuccessMessage,\n\t\t\texistingMessage,\n\t\t\tblockedMessage,\n\t\t\tfailMessage\n\t\t} = this.state\n\n\t\tconst addableRoleDescriptions = addRoleExcludeList\n\t\t\t? pickBy(roleDescriptions, (_, key) => !addRoleExcludeList.includes(key))\n\t\t\t: roleDescriptions\n\t\tconst roles = Object.keys(roleDescriptions)\n\t\tconst shouldRenderAsMutable = !readOnly && canModify\n\n\t\treturn (\n\t\t\t<>\n\t\t\t\t{!!successMessage && (\n\t\t\t\t\t<AlertWithIcon\n\t\t\t\t\t\tid=\"successMessageAlert\"\n\t\t\t\t\t\tvariant=\"success\"\n\t\t\t\t\t\tdismissible\n\t\t\t\t\t\tonClose={() =>\n\t\t\t\t\t\t\tthis.setState({\n\t\t\t\t\t\t\t\tsuccessMessage: undefined\n\t\t\t\t\t\t\t})\n\t\t\t\t\t\t}>\n\t\t\t\t\t\t<p className=\"pre-wrap\">{successMessage}</p>\n\t\t\t\t\t</AlertWithIcon>\n\t\t\t\t)}\n\t\t\t\t{!!existingMessage && (\n\t\t\t\t\t<AlertWithIcon\n\t\t\t\t\t\tid=\"existingMessageAlert\"\n\t\t\t\t\t\tvariant=\"info\"\n\t\t\t\t\t\tdismissible\n\t\t\t\t\t\tonClose={() =>\n\t\t\t\t\t\t\tthis.setState({\n\t\t\t\t\t\t\t\texistingMessage: undefined\n\t\t\t\t\t\t\t})\n\t\t\t\t\t\t}>\n\t\t\t\t\t\t<p className=\"pre-wrap\">{existingMessage}</p>\n\t\t\t\t\t</AlertWithIcon>\n\t\t\t\t)}\n\t\t\t\t{!!blockedMessage && (\n\t\t\t\t\t<AlertWithIcon\n\t\t\t\t\t\tid=\"blockedMessageAlert\"\n\t\t\t\t\t\tvariant=\"warning\"\n\t\t\t\t\t\tdismissible\n\t\t\t\t\t\tonClose={() =>\n\t\t\t\t\t\t\tthis.setState({\n\t\t\t\t\t\t\t\tblockedMessage: undefined\n\t\t\t\t\t\t\t})\n\t\t\t\t\t\t}>\n\t\t\t\t\t\t<p className=\"pre-wrap\">{blockedMessage}</p>\n\t\t\t\t\t</AlertWithIcon>\n\t\t\t\t)}\n\t\t\t\t{!!failMessage && (\n\t\t\t\t\t<AlertWithIcon\n\t\t\t\t\t\tid=\"failMessageAlert\"\n\t\t\t\t\t\tvariant=\"warning\"\n\t\t\t\t\t\tdismissible\n\t\t\t\t\t\tonClose={() =>\n\t\t\t\t\t\t\tthis.setState({\n\t\t\t\t\t\t\t\tfailMessage: undefined\n\t\t\t\t\t\t\t})\n\t\t\t\t\t\t}>\n\t\t\t\t\t\t<p className=\"pre-wrap\">{failMessage}</p>\n\t\t\t\t\t</AlertWithIcon>\n\t\t\t\t)}\n\t\t\t\t{shouldRenderAsMutable && (\n\t\t\t\t\t<UserRolesAdd\n\t\t\t\t\t\tid=\"entityUserRolesAdd\"\n\t\t\t\t\t\tdisabled={isAddDisabled}\n\t\t\t\t\t\tentityName={entityName}\n\t\t\t\t\t\tdefaultRole={defaultRole}\n\t\t\t\t\t\troleDescriptions={addableRoleDescriptions}\n\t\t\t\t\t\trenderAddDescription={renderAddDescription}\n\t\t\t\t\t\ttextForRole={this.textForRole}\n\t\t\t\t\t\tisAddingUsersToRole={isAdding}\n\t\t\t\t\t\tshouldReset={shouldResetAddForm}\n\t\t\t\t\t\taddUserRoles={this.addUserRoles}\n\t\t\t\t\t/>\n\t\t\t\t)}\n\t\t\t\t{!!renderTableDescription && renderTableDescription(shouldRenderAsMutable)}\n\t\t\t\t<Row className=\"mt3\">\n\t\t\t\t\t<Col xs={12}>\n\t\t\t\t\t\t{modelStatus === MODEL_STATUS.LOADING ? (\n\t\t\t\t\t\t\t<Loading />\n\t\t\t\t\t\t) : sortedUsers.length > 0 ? (\n\t\t\t\t\t\t\t<UserRolesContext.Provider\n\t\t\t\t\t\t\t\tvalue={{\n\t\t\t\t\t\t\t\t\tisUpdateDisabled: isUpdateDisabled || readOnly,\n\t\t\t\t\t\t\t\t\tisDeleteDisabled: isDeleteDisabled || readOnly,\n\t\t\t\t\t\t\t\t\tcanModify: shouldRenderAsMutable,\n\t\t\t\t\t\t\t\t\tcanDeleteSelf,\n\t\t\t\t\t\t\t\t\tallowMultipleRoles,\n\t\t\t\t\t\t\t\t\troles,\n\t\t\t\t\t\t\t\t\trequiredRole,\n\t\t\t\t\t\t\t\t\ttextForRole: this.textForRole,\n\t\t\t\t\t\t\t\t\tuserRoleToUpdate,\n\t\t\t\t\t\t\t\t\tisUpdating,\n\t\t\t\t\t\t\t\t\tupdateUserRole: this.updateUserRole,\n\t\t\t\t\t\t\t\t\tuserRoleToRemove,\n\t\t\t\t\t\t\t\t\tisRemoving,\n\t\t\t\t\t\t\t\t\tremoveUserRole: this.alertRemoveUserRole\n\t\t\t\t\t\t\t\t}}>\n\t\t\t\t\t\t\t\t<UserRolesTable id=\"entityUserRolesTable\" users={sortedUsers} />\n\t\t\t\t\t\t\t</UserRolesContext.Provider>\n\t\t\t\t\t\t) : null}\n\t\t\t\t\t</Col>\n\t\t\t\t</Row>\n\t\t\t\t{!!userRoleToRemove && (\n\t\t\t\t\t<AlertDialog\n\t\t\t\t\t\tid=\"removeUserAlert\"\n\t\t\t\t\t\tisOpen={shouldShowRemoveDialog}\n\t\t\t\t\t\ttitle={this.removeUserRoleTitle(userRoleToRemove.role)}\n\t\t\t\t\t\tdescription={this.renderRemoveUserRoleDescription(userRoleToRemove)}\n\t\t\t\t\t\tonDestroy={() => this.removeUserRole(true)}\n\t\t\t\t\t\tdestroyText=\"Yes, remove the user\"\n\t\t\t\t\t\tonCancel={() => this.removeUserRole(false)}\n\t\t\t\t\t\tcancelText=\"No, I changed my mind\"\n\t\t\t\t\t/>\n\t\t\t\t)}\n\t\t\t</>\n\t\t)\n\t}\n}\n\nexport const mapStateToProps = (state: BaseReduxState, ownProps: UserRolesOwnProps): UserRolesReduxProps => {\n\tconst canModifyGlobally = canPerformActivityGlobally(ownProps.modifyUserRoleActivityName, defaultOptions(state))\n\n\tconst canModify =\n\t\tcanModifyGlobally ||\n\t\t(ownProps.entity\n\t\t\t? canPerformActivityOnEntity(ownProps.modifyUserRoleActivityName, defaultOptions(state, ownProps, 'entity'))\n\t\t\t: false)\n\n\treturn {\n\t\tcanModify,\n\t\tcanDeleteSelf:\n\t\t\tcanModify ||\n\t\t\t(ownProps.entity && ownProps.deleteOwnUserRoleActivityName\n\t\t\t\t? canPerformActivityOnEntity(\n\t\t\t\t\t\townProps.deleteOwnUserRoleActivityName,\n\t\t\t\t\t\tdefaultOptions(state, ownProps, 'entity')\n\t\t\t\t\t)\n\t\t\t\t: false)\n\t}\n}\n\nexport default connect(mapStateToProps)(UserRoles)\n"],"mappings":"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","ignoreList":[]}