mirror of
https://github.com/Unleash/unleash.git
synced 2025-01-06 00:07:44 +01:00
165 lines
5.4 KiB
TypeScript
165 lines
5.4 KiB
TypeScript
|
import { useMemo, useState, VFC } from 'react';
|
||
|
import { useSortBy, useTable } from 'react-table';
|
||
|
import {
|
||
|
Table,
|
||
|
TableBody,
|
||
|
TableRow,
|
||
|
TableCell,
|
||
|
SortableTableHeader,
|
||
|
} from 'component/common/Table';
|
||
|
import { Avatar, Box, SelectChangeEvent } from '@mui/material';
|
||
|
import { Delete } from '@mui/icons-material';
|
||
|
import { sortTypes } from 'utils/sortTypes';
|
||
|
import {
|
||
|
IProjectAccessOutput,
|
||
|
IProjectAccessUser,
|
||
|
} from 'hooks/api/getters/useProjectAccess/useProjectAccess';
|
||
|
import { ProjectRoleCell } from './ProjectRoleCell/ProjectRoleCell';
|
||
|
import PermissionIconButton from 'component/common/PermissionIconButton/PermissionIconButton';
|
||
|
import { UPDATE_PROJECT } from 'component/providers/AccessProvider/permissions';
|
||
|
import { TextCell } from 'component/common/Table/cells/TextCell/TextCell';
|
||
|
|
||
|
interface IProjectAccessTableProps {
|
||
|
access: IProjectAccessOutput;
|
||
|
projectId: string;
|
||
|
handleRoleChange: (
|
||
|
userId: number
|
||
|
) => (event: SelectChangeEvent) => Promise<void>;
|
||
|
handleRemoveAccess: (user: IProjectAccessUser) => void;
|
||
|
}
|
||
|
|
||
|
export const ProjectAccessTable: VFC<IProjectAccessTableProps> = ({
|
||
|
access,
|
||
|
projectId,
|
||
|
handleRoleChange,
|
||
|
handleRemoveAccess,
|
||
|
}) => {
|
||
|
const [initialState] = useState({});
|
||
|
const data = access.users;
|
||
|
|
||
|
const columns = useMemo(
|
||
|
() => [
|
||
|
{
|
||
|
Header: 'Avatar',
|
||
|
accessor: 'imageUrl',
|
||
|
disableSortBy: true,
|
||
|
width: 80,
|
||
|
Cell: ({ value }: { value: string }) => (
|
||
|
<Avatar
|
||
|
alt="Gravatar"
|
||
|
src={value}
|
||
|
sx={{ width: 32, height: 32, mx: 'auto' }}
|
||
|
/>
|
||
|
),
|
||
|
align: 'center',
|
||
|
},
|
||
|
{
|
||
|
Header: 'Name',
|
||
|
accessor: 'name',
|
||
|
},
|
||
|
{
|
||
|
Header: 'Username',
|
||
|
id: 'username',
|
||
|
accessor: 'email',
|
||
|
Cell: ({ row: { original: user } }: any) => (
|
||
|
<TextCell>{user.email || user.username}</TextCell>
|
||
|
),
|
||
|
},
|
||
|
{
|
||
|
Header: 'Role',
|
||
|
accessor: 'roleId',
|
||
|
Cell: ({
|
||
|
value,
|
||
|
row: { original: user },
|
||
|
}: {
|
||
|
value: number;
|
||
|
row: { original: IProjectAccessUser };
|
||
|
}) => (
|
||
|
<ProjectRoleCell
|
||
|
value={value}
|
||
|
user={user}
|
||
|
roles={access.roles}
|
||
|
onChange={handleRoleChange(user.id)}
|
||
|
/>
|
||
|
),
|
||
|
},
|
||
|
{
|
||
|
Header: 'Actions',
|
||
|
id: 'actions',
|
||
|
disableSortBy: true,
|
||
|
align: 'center',
|
||
|
width: 80,
|
||
|
Cell: ({ row: { original: user } }: any) => (
|
||
|
<Box
|
||
|
sx={{
|
||
|
display: 'flex',
|
||
|
justifyContent: 'center',
|
||
|
}}
|
||
|
>
|
||
|
<PermissionIconButton
|
||
|
permission={UPDATE_PROJECT}
|
||
|
projectId={projectId}
|
||
|
edge="end"
|
||
|
onClick={() => handleRemoveAccess(user)}
|
||
|
disabled={access.users.length === 1}
|
||
|
tooltipProps={{
|
||
|
title:
|
||
|
access.users.length === 1
|
||
|
? 'Cannot remove access. A project must have at least one owner'
|
||
|
: 'Remove access',
|
||
|
}}
|
||
|
>
|
||
|
<Delete />
|
||
|
</PermissionIconButton>
|
||
|
</Box>
|
||
|
),
|
||
|
},
|
||
|
],
|
||
|
[
|
||
|
access.roles,
|
||
|
access.users.length,
|
||
|
handleRemoveAccess,
|
||
|
handleRoleChange,
|
||
|
projectId,
|
||
|
]
|
||
|
);
|
||
|
|
||
|
const { getTableProps, getTableBodyProps, headerGroups, rows, prepareRow } =
|
||
|
useTable(
|
||
|
{
|
||
|
columns: columns as any[], // TODO: fix after `react-table` v8 update
|
||
|
data,
|
||
|
initialState,
|
||
|
sortTypes,
|
||
|
autoResetGlobalFilter: false,
|
||
|
autoResetSortBy: false,
|
||
|
disableSortRemove: true,
|
||
|
defaultColumn: {
|
||
|
Cell: TextCell,
|
||
|
},
|
||
|
},
|
||
|
useSortBy
|
||
|
);
|
||
|
|
||
|
return (
|
||
|
<Table {...getTableProps()}>
|
||
|
{/* @ts-expect-error -- react-table */}
|
||
|
<SortableTableHeader headerGroups={headerGroups} />
|
||
|
<TableBody {...getTableBodyProps()}>
|
||
|
{rows.map(row => {
|
||
|
prepareRow(row);
|
||
|
return (
|
||
|
<TableRow hover {...row.getRowProps()}>
|
||
|
{row.cells.map(cell => (
|
||
|
<TableCell {...cell.getCellProps()}>
|
||
|
{cell.render('Cell')}
|
||
|
</TableCell>
|
||
|
))}
|
||
|
</TableRow>
|
||
|
);
|
||
|
})}
|
||
|
</TableBody>
|
||
|
</Table>
|
||
|
);
|
||
|
};
|