1
0
mirror of https://github.com/haiwen/seahub.git synced 2025-09-10 19:29:56 +00:00
Files
seahub/frontend/src/pages/org-admin/org-users-users.js

329 lines
10 KiB
JavaScript
Raw Normal View History

import React, { Component, Fragment } from 'react';
import PropTypes from 'prop-types';
import { navigate } from '@gatsbyjs/reach-router';
import Nav from './org-users-nav';
import OrgUsersList from './org-users-list';
import MainPanelTopbar from './main-panel-topbar';
import ModalPortal from '../../components/modal-portal';
import ImportOrgUsersDialog from '../../components/dialog/org-import-users-dialog';
import AddOrgUserDialog from '../../components/dialog/org-add-user-dialog';
import InviteUserDialog from '../../components/dialog/org-admin-invite-user-dialog';
2023-12-07 22:50:39 +08:00
import InviteUserViaWeiXinDialog from '../../components/dialog/org-admin-invite-user-via-weixin-dialog';
import toaster from '../../components/toast';
import { seafileAPI } from '../../utils/seafile-api';
import OrgUserInfo from '../../models/org-user';
2023-12-07 22:50:39 +08:00
import { gettext, invitationLink, orgID, siteRoot, orgEnableAdminInviteUser} from '../../utils/constants';
import { Utils } from '../../utils/utils';
2021-07-22 11:54:10 +08:00
class Search extends React.Component {
constructor(props) {
super(props);
this.state = {
value: ''
};
}
handleInputChange = (e) => {
this.setState({
value: e.target.value
});
};
2021-07-22 11:54:10 +08:00
handleKeyDown = (e) => {
2021-07-22 11:54:10 +08:00
if (e.key == 'Enter') {
e.preventDefault();
this.handleSubmit();
}
};
2021-07-22 11:54:10 +08:00
handleSubmit = () => {
const value = this.state.value.trim();
if (!value) {
return false;
}
this.props.submit(value);
};
2021-07-22 11:54:10 +08:00
render() {
return (
<div className="input-icon">
<i className="d-flex input-icon-addon fas fa-search"></i>
<input
type="text"
className="form-control search-input h-6 mr-1"
style={{width: '15rem'}}
placeholder={this.props.placeholder}
value={this.state.value}
onChange={this.handleInputChange}
onKeyDown={this.handleKeyDown}
2021-07-22 11:54:10 +08:00
autoComplete="off"
/>
</div>
);
}
}
Search.propTypes = {
placeholder: PropTypes.string.isRequired,
submit: PropTypes.func.isRequired,
};
class OrgUsers extends Component {
constructor(props) {
super(props);
this.state = {
orgUsers: [],
page: 1,
pageNext: false,
sortBy: '',
sortOrder: 'asc',
isShowAddOrgUserDialog: false,
isImportOrgUsersDialogOpen: false,
2023-12-07 22:50:39 +08:00
isInviteUserDialogOpen: false,
isInviteUserViaWeiXinDialogOpen: false
};
}
componentDidMount() {
let urlParams = (new URL(window.location)).searchParams;
const { page, sortBy, sortOrder } = this.state;
this.setState({
/*
perPage: parseInt(urlParams.get('per_page') || perPage),
currentPage: parseInt(urlParams.get('page') || currentPage),
*/
page: parseInt(urlParams.get('page') || page),
sortBy: urlParams.get('order_by') || sortBy,
sortOrder: urlParams.get('direction') || sortOrder
}, () => {
this.initOrgUsersData(this.state.page);
});
}
sortByQuotaUsage = () => {
this.setState({
sortBy: 'quota_usage',
sortOrder: this.state.sortOrder == 'asc' ? 'desc' : 'asc',
page: 1
}, () => {
let url = new URL(location.href);
let searchParams = new URLSearchParams(url.search);
const { page, sortBy, sortOrder } = this.state;
searchParams.set('page', page);
searchParams.set('order_by', sortBy);
searchParams.set('direction', sortOrder);
url.search = searchParams.toString();
navigate(url.toString());
this.initOrgUsersData(page);
});
};
toggleImportOrgUsersDialog = () => {
this.setState({isImportOrgUsersDialogOpen: !this.state.isImportOrgUsersDialogOpen});
};
toggleAddOrgUser = () => {
this.setState({isShowAddOrgUserDialog: !this.state.isShowAddOrgUserDialog});
};
toggleInviteUserDialog = () => {
this.setState({isInviteUserDialogOpen: !this.state.isInviteUserDialogOpen});
};
2023-12-07 22:50:39 +08:00
toggleInviteUserViaWeiXinDialog = () => {
this.setState({isInviteUserViaWeiXinDialogOpen: !this.state.isInviteUserViaWeiXinDialogOpen});
}
initOrgUsersData = (page) => {
const { sortBy, sortOrder } = this.state;
seafileAPI.orgAdminListOrgUsers(orgID, '', page, sortBy, sortOrder).then(res => {
let userList = res.data.user_list.map(item => {
return new OrgUserInfo(item);
});
this.setState({
orgUsers: userList,
pageNext: res.data.page_next,
page: res.data.page,
});
}).catch(error => {
let errMessage = Utils.getErrorMsg(error);
toaster.danger(errMessage);
});
};
importOrgUsers = (file) => {
toaster.notify(gettext('It may take some time, please wait.'));
seafileAPI.orgAdminImportUsersViaFile(orgID, file).then((res) => {
if (res.data.success.length) {
const users = res.data.success.map(item => {
if (item.institution == undefined) {
item.institution = '';
}
return new OrgUserInfo(item);
});
this.setState({
orgUsers: users.concat(this.state.orgUsers)
});
}
res.data.failed.forEach(item => {
const msg = `${item.email}: ${item.error_msg}`;
toaster.danger(msg);
});
}).catch((error) => {
let errMsg = Utils.getErrorMsg(error);
toaster.danger(errMsg);
});
};
addOrgUser = (email, name, password) => {
seafileAPI.orgAdminAddOrgUser(orgID, email, name, password).then(res => {
let userInfo = new OrgUserInfo(res.data);
this.state.orgUsers.unshift(userInfo);
this.setState({
orgUsers: this.state.orgUsers
});
this.toggleAddOrgUser();
let msg = gettext('successfully added user %s.');
msg = msg.replace('%s', email);
toaster.success(msg);
}).catch(error => {
let errMessage = Utils.getErrorMsg(error);
toaster.danger(errMessage);
this.toggleAddOrgUser();
});
};
toggleOrgUsersDelete = (email, username) => {
seafileAPI.orgAdminDeleteOrgUser(orgID, email).then(res => {
let users = this.state.orgUsers.filter(item => item.email != email);
this.setState({orgUsers: users});
let msg = gettext('Deleted user %s');
msg = msg.replace('%s', username);
toaster.success(msg);
}).catch(error => {
let errMessage = Utils.getErrorMsg(error);
toaster.danger(errMessage);
});
};
2023-12-07 22:50:39 +08:00
inviteOrgUser = (emails) => {
seafileAPI.orgAdminInviteOrgUsers(orgID, emails.split(',')).then(res => {
this.toggleInviteUserDialog();
let users = res.data.success.map(user => {
return new OrgUserInfo(user);
});
this.setState({
orgUsers: users.concat(this.state.orgUsers)
});
res.data.success.map(item => {
let msg = gettext('successfully sent email to %s.');
msg = msg.replace('%s', item.email);
toaster.success(msg);
});
res.data.failed.map(item => {
const msg = `${item.email}: ${item.error_msg}`;
toaster.danger(msg);
});
}).catch(error => {
this.toggleInviteUserDialog();
let errMessage = Utils.getErrorMsg(error);
toaster.danger(errMessage);
});
};
changeStatus= (email, isActive) => {
seafileAPI.orgAdminChangeOrgUserStatus(orgID, email, isActive).then(res => {
let users = this.state.orgUsers.map(item => {
if (item.email == email) {
item['is_active']= res.data['is_active'];
}
return item;
});
this.setState({orgUsers: users});
toaster.success(gettext('Edit succeeded.'));
}).catch(error => {
let errMessage = Utils.getErrorMsg(error);
toaster.danger(errMessage);
});
};
2021-07-22 11:54:10 +08:00
searchItems = (keyword) => {
navigate(`${siteRoot}org/useradmin/search-users/?query=${encodeURIComponent(keyword)}`);
};
2021-07-22 11:54:10 +08:00
getSearch = () => {
return <Search
placeholder={gettext('Search users')}
submit={this.searchItems}
/>;
};
2021-07-22 11:54:10 +08:00
render() {
const topBtn = 'btn btn-secondary operation-item';
let topbarChildren;
topbarChildren = (
<Fragment>
2023-12-07 22:50:39 +08:00
<button className="btn btn-secondary operation-item" onClick={this.toggleImportOrgUsersDialog}>{gettext('Import users')}</button>
<button className={topBtn} title={gettext('Add user')} onClick={this.toggleAddOrgUser}>
<i className="fas fa-plus-square text-secondary mr-1"></i>{gettext('Add user')}</button>
{orgEnableAdminInviteUser &&
<button className={topBtn} title={gettext('Invite users')} onClick={this.toggleInviteUserDialog}>
<i className="fas fa-plus-square text-secondary mr-1"></i>{gettext('Invite users')}</button>
}
{invitationLink &&
2023-12-07 22:50:39 +08:00
<button className={topBtn} title={'通过微信邀请用户'} onClick={this.toggleInviteUserViaWeiXinDialog}>
<i className="fas fa-plus-square text-secondary mr-1"></i>{''}</button>
}
{this.state.isImportOrgUsersDialogOpen &&
<ModalPortal>
<ImportOrgUsersDialog importUsersInBatch={this.importOrgUsers} toggle={this.toggleImportOrgUsersDialog}/>
</ModalPortal>
}
{this.state.isShowAddOrgUserDialog &&
<ModalPortal>
<AddOrgUserDialog handleSubmit={this.addOrgUser} toggle={this.toggleAddOrgUser}/>
</ModalPortal>
}
{this.state.isInviteUserDialogOpen &&
<ModalPortal>
2023-12-07 22:50:39 +08:00
<InviteUserDialog handleSubmit={this.inviteOrgUser} toggle={this.toggleInviteUserDialog}/>
</ModalPortal>
}
{this.state.isInviteUserViaWeiXinDialogOpen &&
<ModalPortal>
<InviteUserViaWeiXinDialog invitationLink={invitationLink} toggle={this.toggleInviteUserViaWeiXinDialog}/>
</ModalPortal>
}
</Fragment>
);
return (
<Fragment>
2021-07-22 11:54:10 +08:00
<MainPanelTopbar children={topbarChildren} search={this.getSearch()}/>
<div className="main-panel-center flex-row">
<div className="cur-view-container">
<Nav currentItem="all" />
<OrgUsersList
initOrgUsersData={this.initOrgUsersData}
toggleDelete={this.toggleOrgUsersDelete}
changeStatus={this.changeStatus}
orgUsers={this.state.orgUsers}
page={this.state.page}
pageNext={this.state.pageNext}
sortBy={this.state.sortBy}
sortOrder={this.state.sortOrder}
sortByQuotaUsage={this.sortByQuotaUsage}
/>
</div>
</div>
</Fragment>
);
}
}
export default OrgUsers;