|
|
@@ -0,0 +1,231 @@
|
|
|
+import React, { Fragment } from 'react';
|
|
|
+
|
|
|
+import { UserPicture } from '@growi/ui';
|
|
|
+import dateFnsFormat from 'date-fns/format';
|
|
|
+import { useTranslation } from 'next-i18next';
|
|
|
+import PropTypes from 'prop-types';
|
|
|
+
|
|
|
+import AdminUsersContainer from '~/client/services/AdminUsersContainer';
|
|
|
+
|
|
|
+import { withUnstatedContainers } from '../../UnstatedUtils';
|
|
|
+
|
|
|
+import SortIcons from './SortIcons';
|
|
|
+import UserMenu from './UserMenu';
|
|
|
+
|
|
|
+
|
|
|
+class UserTable extends React.Component {
|
|
|
+
|
|
|
+ constructor(props) {
|
|
|
+ super(props);
|
|
|
+
|
|
|
+ this.state = {
|
|
|
+
|
|
|
+ };
|
|
|
+
|
|
|
+ this.getUserStatusLabel = this.getUserStatusLabel.bind(this);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * return status label element by `userStatus`
|
|
|
+ * @param {string} userStatus
|
|
|
+ * @return status label element
|
|
|
+ */
|
|
|
+ getUserStatusLabel(userStatus) {
|
|
|
+ let additionalClassName;
|
|
|
+ let text;
|
|
|
+
|
|
|
+ switch (userStatus) {
|
|
|
+ case 1:
|
|
|
+ additionalClassName = 'badge-info';
|
|
|
+ text = 'Approval Pending';
|
|
|
+ break;
|
|
|
+ case 2:
|
|
|
+ additionalClassName = 'badge-success';
|
|
|
+ text = 'Active';
|
|
|
+ break;
|
|
|
+ case 3:
|
|
|
+ additionalClassName = 'badge-warning';
|
|
|
+ text = 'Suspended';
|
|
|
+ break;
|
|
|
+ case 4:
|
|
|
+ additionalClassName = 'badge-danger';
|
|
|
+ text = 'Deleted';
|
|
|
+ break;
|
|
|
+ case 5:
|
|
|
+ additionalClassName = 'badge-pink';
|
|
|
+ text = 'Invited';
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ return (
|
|
|
+ <span className={`badge badge-pill ${additionalClassName}`}>
|
|
|
+ {text}
|
|
|
+ </span>
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * return admin label element by `isAdmin`
|
|
|
+ * @param {string} isAdmin
|
|
|
+ * @return admin label element
|
|
|
+ */
|
|
|
+ getUserAdminLabel(isAdmin) {
|
|
|
+ const { t } = this.props;
|
|
|
+
|
|
|
+ if (isAdmin) {
|
|
|
+ return <span className="badge badge-indigo badge-pill ml-2">{t('admin:user_management.user_table.administrator')}</span>;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ sortIconsClickedHandler(sort, sortOrder) {
|
|
|
+ const isAsc = sortOrder === 'asc';
|
|
|
+
|
|
|
+ const { adminUsersContainer } = this.props;
|
|
|
+ adminUsersContainer.sort(sort, isAsc);
|
|
|
+ }
|
|
|
+
|
|
|
+ render() {
|
|
|
+ const { t, adminUsersContainer } = this.props;
|
|
|
+
|
|
|
+ const isCurrentSortOrderAsc = adminUsersContainer.state.sortOrder === 'asc';
|
|
|
+
|
|
|
+ return (
|
|
|
+ <Fragment>
|
|
|
+ <div className="table-responsive text-nowrap h-100">
|
|
|
+ <table className="table table-default table-bordered table-user-list">
|
|
|
+ <thead>
|
|
|
+ <tr>
|
|
|
+ <th width="100px">#</th>
|
|
|
+ <th>
|
|
|
+ <div className="d-flex align-items-center">
|
|
|
+ <div className="mr-3">
|
|
|
+ {t('user_management.status')}
|
|
|
+ </div>
|
|
|
+ <SortIcons
|
|
|
+ isSelected={adminUsersContainer.state.sort === 'status'}
|
|
|
+ isAsc={isCurrentSortOrderAsc}
|
|
|
+ onClick={(sortOrder) => {
|
|
|
+ this.sortIconsClickedHandler('status', sortOrder);
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </div>
|
|
|
+ </th>
|
|
|
+ <th>
|
|
|
+ <div className="d-flex align-items-center">
|
|
|
+ <div className="mr-3">
|
|
|
+ <code>username</code>
|
|
|
+ </div>
|
|
|
+ <SortIcons
|
|
|
+ isSelected={adminUsersContainer.state.sort === 'username'}
|
|
|
+ isAsc={isCurrentSortOrderAsc}
|
|
|
+ onClick={(sortOrder) => {
|
|
|
+ this.sortIconsClickedHandler('username', sortOrder);
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </div>
|
|
|
+ </th>
|
|
|
+ <th>
|
|
|
+ <div className="d-flex align-items-center">
|
|
|
+ <div className="mr-3">
|
|
|
+ {t('Name')}
|
|
|
+ </div>
|
|
|
+ <SortIcons
|
|
|
+ isSelected={adminUsersContainer.state.sort === 'name'}
|
|
|
+ isAsc={isCurrentSortOrderAsc}
|
|
|
+ onClick={(sortOrder) => {
|
|
|
+ this.sortIconsClickedHandler('name', sortOrder);
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </div>
|
|
|
+ </th>
|
|
|
+ <th>
|
|
|
+ <div className="d-flex align-items-center">
|
|
|
+ <div className="mr-3">
|
|
|
+ {t('Email')}
|
|
|
+ </div>
|
|
|
+ <SortIcons
|
|
|
+ isSelected={adminUsersContainer.state.sort === 'email'}
|
|
|
+ isAsc={isCurrentSortOrderAsc}
|
|
|
+ onClick={(sortOrder) => {
|
|
|
+ this.sortIconsClickedHandler('email', sortOrder);
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </div>
|
|
|
+ </th>
|
|
|
+ <th width="100px">
|
|
|
+ <div className="d-flex align-items-center">
|
|
|
+ <div className="mr-3">
|
|
|
+ {t('Created')}
|
|
|
+ </div>
|
|
|
+ <SortIcons
|
|
|
+ isSelected={adminUsersContainer.state.sort === 'createdAt'}
|
|
|
+ isAsc={isCurrentSortOrderAsc}
|
|
|
+ onClick={(sortOrder) => {
|
|
|
+ this.sortIconsClickedHandler('createdAt', sortOrder);
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </div>
|
|
|
+ </th>
|
|
|
+ <th width="150px">
|
|
|
+ <div className="d-flex align-items-center">
|
|
|
+ <div className="mr-3">
|
|
|
+ {t('last_login')}
|
|
|
+ </div>
|
|
|
+ <SortIcons
|
|
|
+ isSelected={adminUsersContainer.state.sort === 'lastLoginAt'}
|
|
|
+ isAsc={isCurrentSortOrderAsc}
|
|
|
+ onClick={(sortOrder) => {
|
|
|
+ this.sortIconsClickedHandler('lastLoginAt', sortOrder);
|
|
|
+ }}
|
|
|
+ />
|
|
|
+ </div>
|
|
|
+ </th>
|
|
|
+ <th width="70px"></th>
|
|
|
+ </tr>
|
|
|
+ </thead>
|
|
|
+ <tbody>
|
|
|
+ {adminUsersContainer.state.users.map((user) => {
|
|
|
+ return (
|
|
|
+ <tr data-testid="user-table-tr" key={user._id}>
|
|
|
+ <td>
|
|
|
+ <UserPicture user={user} />
|
|
|
+ </td>
|
|
|
+ <td>{this.getUserStatusLabel(user.status)} {this.getUserAdminLabel(user.admin)}</td>
|
|
|
+ <td>
|
|
|
+ <strong>{user.username}</strong>
|
|
|
+ </td>
|
|
|
+ <td>{user.name}</td>
|
|
|
+ <td>{user.email}</td>
|
|
|
+ <td>{dateFnsFormat(new Date(user.createdAt), 'yyyy-MM-dd')}</td>
|
|
|
+ <td>
|
|
|
+ {user.lastLoginAt && <span>{dateFnsFormat(new Date(user.lastLoginAt), 'yyyy-MM-dd HH:mm')}</span>}
|
|
|
+ </td>
|
|
|
+ <td>
|
|
|
+ <UserMenu user={user} />
|
|
|
+ </td>
|
|
|
+ </tr>
|
|
|
+ );
|
|
|
+ })}
|
|
|
+ </tbody>
|
|
|
+ </table>
|
|
|
+ </div>
|
|
|
+ </Fragment>
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+UserTable.propTypes = {
|
|
|
+ t: PropTypes.func.isRequired, // i18next
|
|
|
+ adminUsersContainer: PropTypes.instanceOf(AdminUsersContainer).isRequired,
|
|
|
+};
|
|
|
+
|
|
|
+const UserTableWrapperFC = (props) => {
|
|
|
+ const { t } = useTranslation('admin');
|
|
|
+ return <UserTable t={t} {...props} />;
|
|
|
+};
|
|
|
+
|
|
|
+const UserTableWrapper = withUnstatedContainers(UserTableWrapperFC, [AdminUsersContainer]);
|
|
|
+
|
|
|
+export default UserTableWrapper;
|