UserGroupPage.tsx 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. import React, { Fragment } from 'react';
  2. import UserGroupTable from './UserGroupTable';
  3. import UserGroupCreateForm from './UserGroupCreateForm';
  4. import UserGroupDeleteModal from './UserGroupDeleteModal';
  5. import { withUnstatedContainers } from '../../UnstatedUtils';
  6. import AppContainer from '~/client/services/AppContainer';
  7. import { toastSuccess, toastError } from '~/client/util/apiNotification';
  8. import { IUserGroup, IUserGroupRelation } from '~/interfaces/user';
  9. import Xss from '~/services/xss';
  10. import { CustomWindow } from '~/interfaces/global';
  11. type Props = {
  12. appContainer: AppContainer,
  13. };
  14. type State = {
  15. userGroups: IUserGroup[],
  16. userGroupRelations: IUserGroupRelation[],
  17. selectedUserGroup: IUserGroup | undefined,
  18. isDeleteModalShown: boolean,
  19. };
  20. class UserGroupPage extends React.Component<Props, State> {
  21. xss: Xss;
  22. state: State;
  23. constructor(props) {
  24. super(props);
  25. this.state = {
  26. userGroups: [],
  27. userGroupRelations: [],
  28. selectedUserGroup: undefined, // not null but undefined (to use defaultProps in UserGroupDeleteModal)
  29. isDeleteModalShown: false,
  30. };
  31. this.xss = (window as CustomWindow).xss;
  32. this.showDeleteModal = this.showDeleteModal.bind(this);
  33. this.hideDeleteModal = this.hideDeleteModal.bind(this);
  34. this.addUserGroup = this.addUserGroup.bind(this);
  35. this.deleteUserGroupById = this.deleteUserGroupById.bind(this);
  36. }
  37. async componentDidMount() {
  38. await this.syncUserGroupAndRelations();
  39. }
  40. async showDeleteModal(group: IUserGroup) {
  41. try {
  42. await this.syncUserGroupAndRelations();
  43. this.setState({
  44. selectedUserGroup: group,
  45. isDeleteModalShown: true,
  46. });
  47. }
  48. catch (err) {
  49. toastError(err);
  50. }
  51. }
  52. hideDeleteModal() {
  53. this.setState({
  54. selectedUserGroup: undefined,
  55. isDeleteModalShown: false,
  56. });
  57. }
  58. addUserGroup(userGroup, users) {
  59. this.setState((prevState) => {
  60. const userGroupRelations = Object.assign(prevState.userGroupRelations, {
  61. [userGroup._id]: users,
  62. });
  63. return {
  64. userGroups: [...prevState.userGroups, userGroup],
  65. userGroupRelations,
  66. };
  67. });
  68. }
  69. async deleteUserGroupById(deleteGroupId: string, actionName: string, transferToUserGroupId: string) {
  70. try {
  71. const res = await this.props.appContainer.apiv3.delete(`/user-groups/${deleteGroupId}`, {
  72. actionName,
  73. transferToUserGroupId,
  74. });
  75. this.setState((prevState) => {
  76. const userGroups = prevState.userGroups.filter((userGroup) => {
  77. return userGroup._id !== deleteGroupId;
  78. });
  79. delete prevState.userGroupRelations[deleteGroupId];
  80. return {
  81. userGroups,
  82. userGroupRelations: prevState.userGroupRelations,
  83. selectedUserGroup: undefined,
  84. isDeleteModalShown: false,
  85. };
  86. });
  87. toastSuccess(`Deleted a group "${this.xss.process(res.data.userGroup.name)}"`);
  88. }
  89. catch (err) {
  90. toastError(new Error('Unable to delete the group'));
  91. }
  92. }
  93. async syncUserGroupAndRelations() {
  94. try {
  95. const userGroupsRes = await this.props.appContainer.apiv3.get('/user-groups', { pagination: false });
  96. const userGroupRelationsRes = await this.props.appContainer.apiv3.get('/user-group-relations');
  97. this.setState({
  98. userGroups: userGroupsRes.data.userGroups,
  99. userGroupRelations: userGroupRelationsRes.data.userGroupRelations,
  100. });
  101. }
  102. catch (err) {
  103. toastError(err);
  104. }
  105. }
  106. render() {
  107. const { isAclEnabled } = this.props.appContainer.config;
  108. return (
  109. <Fragment>
  110. <UserGroupCreateForm
  111. isAclEnabled={isAclEnabled}
  112. onCreate={this.addUserGroup}
  113. />
  114. <UserGroupTable
  115. userGroups={this.state.userGroups}
  116. isAclEnabled={isAclEnabled}
  117. onDelete={this.showDeleteModal}
  118. userGroupRelations={this.state.userGroupRelations}
  119. />
  120. <UserGroupDeleteModal
  121. appContainer={this.props.appContainer}
  122. userGroups={this.state.userGroups}
  123. deleteUserGroup={this.state.selectedUserGroup}
  124. onDelete={this.deleteUserGroupById}
  125. isShow={this.state.isDeleteModalShown}
  126. onShow={this.showDeleteModal}
  127. onHide={this.hideDeleteModal}
  128. />
  129. </Fragment>
  130. );
  131. }
  132. }
  133. /**
  134. * Wrapper component for using unstated
  135. */
  136. const UserGroupPageWrapper = withUnstatedContainers(UserGroupPage, [AppContainer]);
  137. export default UserGroupPageWrapper;