grafana/public/app/features/users/UsersListPage.tsx

154 lines
4.3 KiB
TypeScript
Raw Normal View History

2018-10-03 02:43:10 -05:00
import React, { PureComponent } from 'react';
import { hot } from 'react-hot-loader';
import { connect } from 'react-redux';
import { NavModel, renderMarkdown } from '@grafana/data';
import { HorizontalGroup, Pagination, VerticalGroup } from '@grafana/ui';
import Page from 'app/core/components/Page/Page';
2018-10-03 02:43:10 -05:00
import UsersActionBar from './UsersActionBar';
2018-10-11 04:49:34 -05:00
import UsersTable from './UsersTable';
2018-10-03 02:43:10 -05:00
import InviteesTable from './InviteesTable';
import { Invitee, OrgUser, OrgRole } from 'app/types';
import { loadInvitees, loadUsers, removeUser, updateUser } from './state/actions';
import { getNavModel } from 'app/core/selectors/navModel';
import { getInvitees, getUsers, getUsersSearchQuery, getUsersSearchPage } from './state/selectors';
import { setUsersSearchQuery, setUsersSearchPage } from './state/reducers';
2018-10-03 02:43:10 -05:00
export interface Props {
navModel: NavModel;
invitees: Invitee[];
users: OrgUser[];
searchQuery: string;
searchPage: number;
2018-10-03 02:43:10 -05:00
externalUserMngInfo: string;
2018-10-11 04:49:34 -05:00
hasFetched: boolean;
2018-10-03 02:43:10 -05:00
loadUsers: typeof loadUsers;
loadInvitees: typeof loadInvitees;
setUsersSearchQuery: typeof setUsersSearchQuery;
setUsersSearchPage: typeof setUsersSearchPage;
2018-10-03 02:43:10 -05:00
updateUser: typeof updateUser;
removeUser: typeof removeUser;
}
export interface State {
showInvites: boolean;
}
const pageLimit = 30;
2018-10-03 02:43:10 -05:00
export class UsersListPage extends PureComponent<Props, State> {
externalUserMngInfoHtml: string;
constructor(props: Props) {
super(props);
if (this.props.externalUserMngInfo) {
this.externalUserMngInfoHtml = renderMarkdown(this.props.externalUserMngInfo);
}
this.state = {
showInvites: false,
};
}
2018-10-03 02:43:10 -05:00
componentDidMount() {
this.fetchUsers();
this.fetchInvitees();
}
async fetchUsers() {
return await this.props.loadUsers();
}
async fetchInvitees() {
return await this.props.loadInvitees();
}
onRoleChange = (role: OrgRole, user: OrgUser) => {
2018-10-03 02:43:10 -05:00
const updatedUser = { ...user, role: role };
this.props.updateUser(updatedUser);
};
2018-10-03 03:54:15 -05:00
onShowInvites = () => {
this.setState((prevState) => ({
2018-10-03 02:43:10 -05:00
showInvites: !prevState.showInvites,
}));
};
getPaginatedUsers = (users: OrgUser[]) => {
const offset = (this.props.searchPage - 1) * pageLimit;
return users.slice(offset, offset + pageLimit);
};
2018-10-11 04:49:34 -05:00
renderTable() {
const { invitees, users, setUsersSearchPage } = this.props;
const paginatedUsers = this.getPaginatedUsers(users);
const totalPages = Math.ceil(users.length / pageLimit);
2018-10-11 04:49:34 -05:00
if (this.state.showInvites) {
2018-12-14 06:49:14 -06:00
return <InviteesTable invitees={invitees} />;
2018-10-11 04:49:34 -05:00
} else {
return (
<VerticalGroup spacing="md">
<UsersTable
users={paginatedUsers}
onRoleChange={(role, user) => this.onRoleChange(role, user)}
onRemoveUser={(user) => this.props.removeUser(user.userId)}
/>
<HorizontalGroup justify="flex-end">
<Pagination
onNavigate={setUsersSearchPage}
currentPage={this.props.searchPage}
numberOfPages={totalPages}
hideWhenSinglePage={true}
/>
</HorizontalGroup>
</VerticalGroup>
2018-10-11 04:49:34 -05:00
);
}
}
2018-10-03 02:43:10 -05:00
render() {
2018-10-11 04:49:34 -05:00
const { navModel, hasFetched } = this.props;
const externalUserMngInfoHtml = this.externalUserMngInfoHtml;
2018-10-03 02:43:10 -05:00
return (
<Page navModel={navModel}>
<Page.Contents isLoading={!hasFetched}>
<>
<UsersActionBar onShowInvites={this.onShowInvites} showInvites={this.state.showInvites} />
{externalUserMngInfoHtml && (
<div className="grafana-info-box" dangerouslySetInnerHTML={{ __html: externalUserMngInfoHtml }} />
)}
{hasFetched && this.renderTable()}
</>
</Page.Contents>
</Page>
2018-10-03 02:43:10 -05:00
);
}
}
function mapStateToProps(state: any) {
2018-10-03 02:43:10 -05:00
return {
navModel: getNavModel(state.navIndex, 'users'),
users: getUsers(state.users),
searchQuery: getUsersSearchQuery(state.users),
searchPage: getUsersSearchPage(state.users),
2018-10-03 02:43:10 -05:00
invitees: getInvitees(state.users),
externalUserMngInfo: state.users.externalUserMngInfo,
2018-10-11 04:49:34 -05:00
hasFetched: state.users.hasFetched,
2018-10-03 02:43:10 -05:00
};
}
const mapDispatchToProps = {
loadUsers,
loadInvitees,
setUsersSearchQuery,
setUsersSearchPage,
2018-10-03 02:43:10 -05:00
updateUser,
removeUser,
};
export default hot(module)(connect(mapStateToProps, mapDispatchToProps)(UsersListPage));