grafana/public/app/features/admin/Users/UsersTable.tsx
2023-11-17 14:08:41 +02:00

156 lines
4.6 KiB
TypeScript

import React, { useMemo } from 'react';
import {
Avatar,
CellProps,
Column,
FetchDataFunc,
Icon,
InteractiveTable,
Pagination,
Stack,
Tag,
Text,
Tooltip,
} from '@grafana/ui';
import { TagBadge } from 'app/core/components/TagFilter/TagBadge';
import { UserDTO } from 'app/types';
import { OrgUnits } from './OrgUnits';
type Cell<T extends keyof UserDTO = keyof UserDTO> = CellProps<UserDTO, UserDTO[T]>;
interface UsersTableProps {
users: UserDTO[];
showPaging?: boolean;
totalPages: number;
onChangePage: (page: number) => void;
currentPage: number;
fetchData?: FetchDataFunc<UserDTO>;
}
export const UsersTable = ({
users,
showPaging,
totalPages,
onChangePage,
currentPage,
fetchData,
}: UsersTableProps) => {
const showLicensedRole = useMemo(() => users.some((user) => user.licensedRole), [users]);
const columns: Array<Column<UserDTO>> = useMemo(
() => [
{
id: 'avatarUrl',
header: '',
cell: ({ cell: { value } }: Cell<'avatarUrl'>) => value && <Avatar src={value} alt={'User avatar'} />,
},
{
id: 'login',
header: 'Login',
cell: ({ cell: { value } }: Cell<'login'>) => value,
sortType: 'string',
},
{
id: 'email',
header: 'Email',
cell: ({ cell: { value } }: Cell<'email'>) => value,
sortType: 'string',
},
{
id: 'name',
header: 'Name',
cell: ({ cell: { value } }: Cell<'name'>) => value,
sortType: 'string',
},
{
id: 'orgs',
header: 'Belongs to',
cell: ({ cell: { value, row } }: Cell<'orgs'>) => {
return (
<Stack alignItems={'center'}>
<OrgUnits units={value} icon={'building'} />
{row.original.isAdmin && (
<Tooltip placement="top" content="Grafana Admin">
<Icon name="shield" />
</Tooltip>
)}
</Stack>
);
},
sortType: (a, b) => (a.original.orgs?.length || 0) - (b.original.orgs?.length || 0),
},
...(showLicensedRole
? [
{
id: 'licensedRole',
header: 'Licensed role',
cell: ({ cell: { value } }: Cell<'licensedRole'>) => {
return value === 'None' ? (
<Text color={'disabled'}>
Not assigned{' '}
<Tooltip placement="top" content="A licensed role will be assigned when this user signs in">
<Icon name="question-circle" />
</Tooltip>
</Text>
) : (
value
);
},
// Needs the assertion here, the types are not inferred correctly due to the conditional assignment
sortType: 'string' as const,
},
]
: []),
{
id: 'lastSeenAtAge',
header: 'Last active',
headerTooltip: {
content: 'Time since user was seen using Grafana',
iconName: 'question-circle',
},
cell: ({ cell: { value } }: Cell<'lastSeenAtAge'>) => {
return <>{value && <>{value === '10 years' ? <Text color={'disabled'}>Never</Text> : value}</>}</>;
},
sortType: (a, b) => new Date(a.original.lastSeenAt!).getTime() - new Date(b.original.lastSeenAt!).getTime(),
},
{
id: 'authLabels',
header: 'Origin',
cell: ({ cell: { value } }: Cell<'authLabels'>) => (
<>{Array.isArray(value) && value.length > 0 && <TagBadge label={value[0]} removeIcon={false} count={0} />}</>
),
},
{
id: 'isDisabled',
header: '',
cell: ({ cell: { value } }: Cell<'isDisabled'>) => <>{value && <Tag colorIndex={9} name={'Disabled'} />}</>,
},
{
id: 'edit',
header: '',
cell: ({ row: { original } }: Cell) => {
return (
<a href={`admin/users/edit/${original.id}`} aria-label={`Edit team ${original.name}`}>
<Tooltip content={'Edit user'}>
<Icon name={'pen'} />
</Tooltip>
</a>
);
},
},
],
[showLicensedRole]
);
return (
<Stack direction={'column'} gap={2}>
<InteractiveTable columns={columns} data={users} getRowId={(user) => String(user.id)} fetchData={fetchData} />
{showPaging && (
<Stack justifyContent={'flex-end'}>
<Pagination numberOfPages={totalPages} currentPage={currentPage} onNavigate={onChangePage} />
</Stack>
)}
</Stack>
);
};