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.
- package/lib/components/ActionList.js +164 -37
- package/lib/components/AlertDialog.js +128 -12
- package/lib/components/AlertWithIcon.js +88 -29
- package/lib/components/ConnectedModal.js +35 -12
- package/lib/components/Dropdowns/GroupsDropdown.js +63 -45
- package/lib/components/Dropdowns/ManagedNavDropdown.js +92 -67
- package/lib/components/Dropdowns/UserDropdown.js +105 -24
- package/lib/components/Dropdowns/index.js +4 -10
- package/lib/components/EntityOwnerList.js +47 -21
- package/lib/components/Error.js +101 -12
- package/lib/components/ErrorBoundary.js +127 -38
- package/lib/components/ErrorMessage.js +39 -12
- package/lib/components/Forms/DateField.js +56 -45
- package/lib/components/Forms/TimeField.js +76 -45
- package/lib/components/Forms/index.js +3 -5
- package/lib/components/Groups/CreateEditCopySaveButtons.js +109 -14
- package/lib/components/Groups/ExternalGroups/Attach.js +206 -151
- package/lib/components/Groups/ExternalGroups/Table.js +176 -48
- package/lib/components/Groups/GroupCreateOrEditCommonProps.js +2 -2
- package/lib/components/Groups/RosterSyncInfo.js +142 -23
- package/lib/components/HOC/AccessibleAppComponent.js +88 -72
- package/lib/components/HOC/ActivityRequiredComponent.js +68 -33
- package/lib/components/HOC/AsyncComponent.js +49 -41
- package/lib/components/HOC/AuthenticatedComponent.js +55 -44
- package/lib/components/HOC/CollectionComponent.js +154 -104
- package/lib/components/HOC/CollectionFirstItemComponent.js +45 -40
- package/lib/components/HOC/CollectionItemComponent.js +152 -100
- package/lib/components/HOC/ConnectedModalComponent.js +87 -69
- package/lib/components/HOC/DataDependentComponent.js +26 -27
- package/lib/components/HOC/EntityComponent.js +57 -53
- package/lib/components/HOC/FullscreenModalComponent.js +139 -108
- package/lib/components/HOC/GroupActivityRequiredComponent.js +27 -20
- package/lib/components/HOC/GuidComponent.js +20 -20
- package/lib/components/HOC/ModelContextDependencyVerifyComponent.js +32 -29
- package/lib/components/HOC/ModelErrorRedirectComponent.js +37 -39
- package/lib/components/HOC/SearchPersistorComponent.js +237 -173
- package/lib/components/HOC/UnauthenticatedComponent.js +32 -30
- package/lib/components/HOC/UserComponent.js +6 -8
- package/lib/components/Icons/IconAlphaList.js +28 -8
- package/lib/components/Icons/IconExternalUser.js +28 -8
- package/lib/components/Icons/IconImpersonation.js +28 -8
- package/lib/components/Icons/IconStopImpersonating.js +28 -8
- package/lib/components/Icons/IconTable.js +29 -9
- package/lib/components/Icons/IconTableDeleteCol.js +28 -8
- package/lib/components/Icons/IconTableDeleteRow.js +28 -8
- package/lib/components/Icons/IconTableInsertCol.js +28 -8
- package/lib/components/Icons/IconTableInsertRow.js +28 -8
- package/lib/components/Impersonation/Button.js +71 -16
- package/lib/components/Impersonation/Link.js +72 -16
- package/lib/components/Impersonation/UserDetail.js +60 -11
- package/lib/components/Loading.js +23 -8
- package/lib/components/LockDownBrowser/Check.js +188 -51
- package/lib/components/LockDownBrowser/ExitButton.js +22 -13
- package/lib/components/LockDownBrowser/Launch.js +64 -64
- package/lib/components/Lti/Confirm.js +147 -14
- package/lib/components/Lti/CreateNonLtiGroupAlertDialog.js +165 -36
- package/lib/components/Lti/Launch.js +99 -25
- package/lib/components/Lti/LaunchGroup.js +81 -16
- package/lib/components/ManageTable.js +304 -90
- package/lib/components/ManageTableNoDataComponent.js +38 -7
- package/lib/components/NewVersionAlert.js +76 -49
- package/lib/components/NotFound.js +81 -11
- package/lib/components/Notifications.js +179 -129
- package/lib/components/PaginationNextButton.js +28 -9
- package/lib/components/PaginationPreviousButton.js +28 -9
- package/lib/components/Quill/CustomToolbar.js +427 -222
- package/lib/components/Quill/Formats/Image.js +67 -67
- package/lib/components/Quill/Formats/List.js +38 -47
- package/lib/components/Quill/Formats/Video.js +23 -26
- package/lib/components/Quill/ImageDropModule.js +136 -114
- package/lib/components/Quill/ImageWarning.js +41 -12
- package/lib/components/Quill/ImageWithAltTextModal.js +420 -89
- package/lib/components/Quill/Specs/CustomImageSpec.js +32 -31
- package/lib/components/Quill/Specs/CustomVideoSpec.js +22 -23
- package/lib/components/Quill/TableModule/Blots/BaseTableBlot.js +89 -97
- package/lib/components/Quill/TableModule/Blots/TableBlot.js +47 -50
- package/lib/components/Quill/TableModule/Blots/TableBodyBlot.js +48 -51
- package/lib/components/Quill/TableModule/Blots/TableCellBlot.js +219 -224
- package/lib/components/Quill/TableModule/Blots/TableContainer.js +75 -86
- package/lib/components/Quill/TableModule/Blots/TableRowBlot.js +70 -73
- package/lib/components/Quill/TableModule/constants.js +40 -42
- package/lib/components/Quill/TableModule/index.js +357 -305
- package/lib/components/Quill/TableModule/utils.js +39 -48
- package/lib/components/Quill/accessibilityFix.js +219 -223
- package/lib/components/Quill/index.js +30 -33
- package/lib/components/RefreshIndicator/Bordered.js +44 -10
- package/lib/components/RefreshIndicator/Inline.js +43 -12
- package/lib/components/RefreshIndicator/index.js +257 -62
- package/lib/components/SearchControls.js +211 -14
- package/lib/components/SentryRoute.js +5 -7
- package/lib/components/Tables/RoleFilter.js +66 -38
- package/lib/components/Tables/TextFilter.js +58 -18
- package/lib/components/UserRoles/Add.js +193 -99
- package/lib/components/UserRoles/Context.js +3 -6
- package/lib/components/UserRoles/RoleCell.js +176 -75
- package/lib/components/UserRoles/Select.js +151 -20
- package/lib/components/UserRoles/Table.js +215 -82
- package/lib/components/UserRoles/index.js +526 -386
- package/lib/config/eslint/index.js +26 -29
- package/lib/config/eslint/lib/order.js +21 -28
- package/lib/config/eslint/lib/prettier.js +15 -19
- package/lib/config/eslint/lib/typescript.js +87 -113
- package/lib/config/eslint/react.js +18 -15
- package/lib/constants/baseActivity.js +26 -28
- package/lib/constants/baseRole.js +10 -12
- package/lib/constants/configuration.js +43 -55
- package/lib/constants/externalProviderType.js +6 -8
- package/lib/constants/fetchErrorData.js +10 -12
- package/lib/constants/index.js +13 -15
- package/lib/constants/lockDownBrowser.js +23 -25
- package/lib/constants/mockData.js +370 -300
- package/lib/constants/modelStatus.js +11 -13
- package/lib/constants/notificationType.js +8 -10
- package/lib/constants/operatingSystem.js +8 -10
- package/lib/constants/shard.js +7 -9
- package/lib/constants/table.js +18 -22
- package/lib/constants/tier.js +8 -10
- package/lib/constants/userRole.js +11 -8
- package/lib/endpointMappings.js +191 -182
- package/lib/hooks/useCollection.js +79 -65
- package/lib/hooks/useCollectionConfiguration.js +220 -80
- package/lib/hooks/useCollectionItem.js +151 -57
- package/lib/hooks/useGuid.js +16 -9
- package/lib/hooks/usePrevious.js +14 -13
- package/lib/index.js +11 -26
- package/lib/redux/actionCreator.js +44 -35
- package/lib/redux/actions/AuthAction.js +45 -32
- package/lib/redux/actions/ModalAction.js +6 -8
- package/lib/redux/actions/ModelAction.js +95 -43
- package/lib/redux/actions/NotificationAction.js +6 -8
- package/lib/redux/actions/SearchAction.js +5 -7
- package/lib/redux/actions/index.js +6 -8
- package/lib/redux/configureReducers.js +48 -46
- package/lib/redux/configureStore.js +77 -91
- package/lib/redux/helpers.js +2 -5
- package/lib/redux/reducers/authReducer.js +44 -43
- package/lib/redux/reducers/index.js +7 -14
- package/lib/redux/reducers/modalsReducer.js +43 -31
- package/lib/redux/reducers/modelsReducer.js +131 -137
- package/lib/redux/reducers/notificationsReducer.js +20 -20
- package/lib/redux/reducers/searchReducer.js +13 -13
- package/lib/redux/sagas/appInsightsSaga.js +19 -21
- package/lib/redux/sagas/authSaga.js +248 -234
- package/lib/redux/sagas/caliperSaga.js +142 -131
- package/lib/redux/sagas/clockOffsetSaga.js +29 -32
- package/lib/redux/sagas/configurationSaga.js +8 -10
- package/lib/redux/sagas/downtimeApiErrorSaga.js +16 -19
- package/lib/redux/sagas/errorSaga.js +23 -24
- package/lib/redux/sagas/googleAnalyticsSaga.js +24 -27
- package/lib/redux/sagas/identityProviderSaga.js +19 -21
- package/lib/redux/sagas/initialDataLoadSaga.js +34 -31
- package/lib/redux/sagas/lockDownBrowserErrorSaga.js +25 -22
- package/lib/redux/sagas/modelFetchSaga.js +302 -286
- package/lib/redux/sagas/noStoreSaga.js +60 -61
- package/lib/redux/sagas/postLoginDataSaga.js +37 -32
- package/lib/redux/sagas/postLoginRedirectSaga.js +22 -27
- package/lib/redux/sagas/rootSaga.js +77 -60
- package/lib/redux/sagas/sentrySaga.js +25 -28
- package/lib/redux/sagas/userIdSaga.js +13 -15
- package/lib/services/codeProviderService.js +21 -21
- package/lib/services/dateService.js +6 -8
- package/lib/services/documentService.js +10 -11
- package/lib/services/fetchService.js +103 -95
- package/lib/services/persistenceService.js +27 -30
- package/lib/services/ticketProviderService.js +25 -25
- package/lib/services/tokenPersistenceService.js +8 -10
- package/lib/services/windowService.js +14 -16
- package/lib/startup.js +110 -101
- package/lib/types/AppConfiguration.js +2 -2
- package/lib/types/Artifact.js +7 -9
- package/lib/types/BaseReduxState.js +2 -2
- package/lib/types/Client.js +2 -2
- package/lib/types/Collection.js +2 -2
- package/lib/types/Configuration.js +2 -2
- package/lib/types/DeepLinkingResponseRequest.js +2 -2
- package/lib/types/DeletableModel.js +2 -2
- package/lib/types/Event.js +2 -2
- package/lib/types/ExternalGroup.js +2 -2
- package/lib/types/ExternalProvider.js +2 -2
- package/lib/types/ExternalTerm.js +2 -2
- package/lib/types/Group.js +2 -2
- package/lib/types/IdentityProvider.js +2 -2
- package/lib/types/LtiLaunch.js +2 -2
- package/lib/types/NameOnlyEntity.js +2 -2
- package/lib/types/Notification.js +2 -2
- package/lib/types/OptionalRecord.js +2 -2
- package/lib/types/OwnerSchedule.js +2 -2
- package/lib/types/PropertyOfType.js +2 -2
- package/lib/types/Quill.js +2 -2
- package/lib/types/RoleDescription.js +2 -2
- package/lib/types/Search.js +2 -2
- package/lib/types/SimpleLocation.js +2 -2
- package/lib/types/UniTime.js +2 -2
- package/lib/types/User.js +2 -2
- package/lib/types/UserRole.js +2 -2
- package/lib/types/auth/AuthState.js +2 -2
- package/lib/types/auth/CasV1LoginRequestBody.js +2 -2
- package/lib/types/auth/ClientCredentials.js +2 -2
- package/lib/types/auth/CodeProviderService.js +2 -2
- package/lib/types/auth/LocalLoginRequestBody.js +2 -2
- package/lib/types/auth/TicketProviderService.js +2 -2
- package/lib/types/auth/TokenPersistenceService.js +2 -2
- package/lib/types/auth/index.js +8 -10
- package/lib/types/externals.d.js +2 -0
- package/lib/types/index.js +29 -31
- package/lib/types/net/EndpointConfig.js +2 -2
- package/lib/types/net/EndpointMapping.js +2 -2
- package/lib/types/net/EndpointMappings.js +2 -2
- package/lib/types/net/ErrorHandler.js +2 -2
- package/lib/types/net/FetchConfig.js +2 -2
- package/lib/types/net/FetchErrorData.js +6 -8
- package/lib/types/net/FetchResult.js +2 -2
- package/lib/types/net/HTTPMethod.js +2 -2
- package/lib/types/net/HTTPStatusCode.js +12 -14
- package/lib/types/net/Metadata.js +2 -2
- package/lib/types/net/Model.js +2 -2
- package/lib/types/net/ModelCollection.js +2 -2
- package/lib/types/net/ModelsState.js +2 -2
- package/lib/types/net/OAuthToken.js +2 -2
- package/lib/types/net/OAuthTokenOrNull.js +2 -2
- package/lib/types/net/TokenAccessFunction.js +2 -2
- package/lib/types/net/index.js +17 -19
- package/lib/utils/baseActivity.js +83 -85
- package/lib/utils/baseRole.js +32 -36
- package/lib/utils/collection.js +403 -297
- package/lib/utils/cookies.js +19 -23
- package/lib/utils/date.js +188 -205
- package/lib/utils/dom.js +130 -131
- package/lib/utils/domainIdentifier.js +4 -8
- package/lib/utils/entityUserRole.js +2 -5
- package/lib/utils/error.js +14 -19
- package/lib/utils/events.js +32 -31
- package/lib/utils/externalGroup.js +20 -25
- package/lib/utils/externalProviders.js +4 -7
- package/lib/utils/externalTerms.js +6 -6
- package/lib/utils/fetch.js +168 -176
- package/lib/utils/group.js +14 -11
- package/lib/utils/groupDates.js +38 -46
- package/lib/utils/groupRoles.js +23 -32
- package/lib/utils/lockDownBrowser.js +12 -15
- package/lib/utils/logger.js +23 -28
- package/lib/utils/lti.js +4 -7
- package/lib/utils/model.js +28 -43
- package/lib/utils/number.js +9 -13
- package/lib/utils/promise.js +23 -26
- package/lib/utils/quill.js +55 -60
- package/lib/utils/route.js +52 -60
- package/lib/utils/search.js +72 -87
- package/lib/utils/shard.js +33 -42
- package/lib/utils/sort.js +47 -50
- package/lib/utils/string.js +10 -12
- package/lib/utils/table.js +29 -33
- package/lib/utils/timezone.js +7 -12
- package/lib/utils/url.js +130 -144
- package/lib/utils/user.js +54 -64
- package/lib/utils/userAgent.js +7 -14
- package/lib/utils/userRole.js +36 -39
- package/package.json +17 -3
|
@@ -1,396 +1,536 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
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
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
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
|
-
|
|
337
|
-
|
|
338
|
-
|
|
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
|
-
|
|
341
|
-
|
|
342
|
-
|
|
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
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
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
|
-
|
|
381
|
-
const
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
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
|
-
|
|
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":[]}
|