## Context Adding icon column to role table to distinguish roles. <img width="621" alt="Screenshot 2025-03-13 at 11 02 37" src="https://github.com/user-attachments/assets/dc7b121c-2901-4599-9638-d5dcdf443999" />
125 lines
3.5 KiB
TypeScript
125 lines
3.5 KiB
TypeScript
import { SettingsPath } from '@/types/SettingsPath';
|
|
import { TableCell } from '@/ui/layout/table/components/TableCell';
|
|
import { TableRow } from '@/ui/layout/table/components/TableRow';
|
|
import { useTheme } from '@emotion/react';
|
|
import styled from '@emotion/styled';
|
|
import React from 'react';
|
|
import {
|
|
AppTooltip,
|
|
Avatar,
|
|
IconChevronRight,
|
|
IconLock,
|
|
TooltipDelay,
|
|
useIcons,
|
|
} from 'twenty-ui';
|
|
import { Role } from '~/generated-metadata/graphql';
|
|
import { useNavigateSettings } from '~/hooks/useNavigateSettings';
|
|
|
|
const StyledAssignedText = styled.div`
|
|
color: ${({ theme }) => theme.font.color.secondary};
|
|
font-size: ${({ theme }) => theme.font.size.sm};
|
|
`;
|
|
|
|
const StyledNameCell = styled.div`
|
|
color: ${({ theme }) => theme.font.color.primary};
|
|
display: flex;
|
|
gap: ${({ theme }) => theme.spacing(1)};
|
|
`;
|
|
|
|
const StyledAvatarGroup = styled.div`
|
|
display: flex;
|
|
justify-content: flex-end;
|
|
|
|
> * {
|
|
margin-left: -5px;
|
|
|
|
&:first-of-type {
|
|
margin-left: 0;
|
|
}
|
|
}
|
|
`;
|
|
|
|
const StyledIconLockContainer = styled.div`
|
|
align-items: center;
|
|
display: flex;
|
|
justify-content: flex-end;
|
|
`;
|
|
|
|
const StyledTableRow = styled(TableRow)`
|
|
&:hover {
|
|
background: ${({ theme }) => theme.background.transparent.light};
|
|
cursor: pointer;
|
|
}
|
|
`;
|
|
|
|
export const RolesTableRow = ({ role }: { role: Role }) => {
|
|
const theme = useTheme();
|
|
|
|
const navigateSettings = useNavigateSettings();
|
|
|
|
const handleRoleClick = (roleId: string) => {
|
|
navigateSettings(SettingsPath.RoleDetail, { roleId });
|
|
};
|
|
|
|
const { getIcon } = useIcons();
|
|
const Icon = getIcon(role.icon ?? 'IconUser');
|
|
|
|
return (
|
|
<StyledTableRow
|
|
key={role.id}
|
|
gridAutoColumns="332px 3fr 2fr 1fr"
|
|
onClick={() => handleRoleClick(role.id)}
|
|
>
|
|
<TableCell>
|
|
<StyledNameCell>
|
|
<Icon size={theme.icon.size.md} stroke={theme.icon.stroke.sm} />
|
|
{role.label}
|
|
{!role.isEditable && (
|
|
<StyledIconLockContainer>
|
|
<IconLock
|
|
color={theme.font.color.light}
|
|
stroke={theme.icon.stroke.sm}
|
|
size={theme.icon.size.sm}
|
|
/>
|
|
</StyledIconLockContainer>
|
|
)}
|
|
</StyledNameCell>
|
|
</TableCell>
|
|
<TableCell align={'right'}>
|
|
<StyledAvatarGroup>
|
|
{role.workspaceMembers.slice(0, 5).map((workspaceMember) => (
|
|
<React.Fragment key={workspaceMember.id}>
|
|
<div id={`avatar-${workspaceMember.id}`}>
|
|
<Avatar
|
|
avatarUrl={workspaceMember.avatarUrl}
|
|
placeholderColorSeed={workspaceMember.id}
|
|
placeholder={workspaceMember.name.firstName ?? ''}
|
|
type="rounded"
|
|
size="md"
|
|
/>
|
|
</div>
|
|
<AppTooltip
|
|
anchorSelect={`#avatar-${workspaceMember.id}`}
|
|
content={`${workspaceMember.name.firstName} ${workspaceMember.name.lastName}`}
|
|
noArrow
|
|
place="top"
|
|
positionStrategy="fixed"
|
|
delay={TooltipDelay.shortDelay}
|
|
/>
|
|
</React.Fragment>
|
|
))}
|
|
</StyledAvatarGroup>
|
|
</TableCell>
|
|
<TableCell align={'left'}>
|
|
<StyledAssignedText>{role.workspaceMembers.length}</StyledAssignedText>
|
|
</TableCell>
|
|
<TableCell align={'right'}>
|
|
<IconChevronRight
|
|
size={theme.icon.size.md}
|
|
color={theme.font.color.tertiary}
|
|
/>
|
|
</TableCell>
|
|
</StyledTableRow>
|
|
);
|
|
};
|