2023-03-01 16:05:39 +01:00
|
|
|
import {
|
|
|
|
Box,
|
|
|
|
Typography,
|
|
|
|
styled,
|
|
|
|
Avatar,
|
|
|
|
ListItemButton,
|
|
|
|
useTheme,
|
|
|
|
} from '@mui/material';
|
2023-02-27 13:04:09 +01:00
|
|
|
import {
|
|
|
|
NotificationsSchemaItem,
|
|
|
|
NotificationsSchemaItemNotificationType,
|
|
|
|
} from 'openapi';
|
|
|
|
import { ReactComponent as ChangesAppliedIcon } from 'assets/icons/merge.svg';
|
|
|
|
import TimeAgo from 'react-timeago';
|
|
|
|
import { ToggleOffOutlined } from '@mui/icons-material';
|
2023-03-01 16:05:39 +01:00
|
|
|
import { flexRow } from 'themes/themeStyles';
|
2023-02-27 13:04:09 +01:00
|
|
|
|
|
|
|
const StyledContainerBox = styled(Box, {
|
|
|
|
shouldForwardProp: prop => prop !== 'readAt',
|
|
|
|
})<{ readAt: boolean }>(({ theme, readAt }) => ({
|
|
|
|
padding: theme.spacing(0.5),
|
|
|
|
backgroundColor: readAt
|
|
|
|
? theme.palette.neutral.light
|
|
|
|
: theme.palette.secondary.light,
|
|
|
|
width: '30px',
|
|
|
|
height: '30px',
|
|
|
|
display: 'flex',
|
|
|
|
justifyContent: 'center',
|
|
|
|
alignItems: 'center',
|
|
|
|
borderRadius: `${theme.shape.borderRadius}px`,
|
|
|
|
position: 'absolute',
|
|
|
|
top: 3,
|
|
|
|
left: 7,
|
|
|
|
}));
|
|
|
|
|
2023-03-01 14:28:05 +01:00
|
|
|
const StyledListItemButton = styled(ListItemButton)(({ theme }) => ({
|
2023-02-27 13:04:09 +01:00
|
|
|
position: 'relative',
|
|
|
|
cursor: 'pointer',
|
|
|
|
margin: theme.spacing(2, 0),
|
|
|
|
'&:not(:last-child)': {
|
2023-03-06 11:58:36 +01:00
|
|
|
borderBottom: `2px solid ${theme.palette.divider}`,
|
2023-02-27 13:04:09 +01:00
|
|
|
},
|
|
|
|
width: '100%',
|
|
|
|
}));
|
|
|
|
|
|
|
|
const StyledNotificationMessageBox = styled(Box)(({ theme }) => ({
|
|
|
|
marginLeft: theme.spacing(4),
|
|
|
|
display: 'flex',
|
|
|
|
flexDirection: 'column',
|
|
|
|
width: '100%',
|
|
|
|
}));
|
|
|
|
|
|
|
|
const StyledSecondaryInfoBox = styled(Box)(({ theme }) => ({
|
|
|
|
display: 'flex',
|
2023-03-01 16:05:39 +01:00
|
|
|
justifyContent: 'space-between',
|
|
|
|
alignItems: 'center',
|
|
|
|
margin: theme.spacing(1.5, 0, 1.5, 0),
|
2023-02-27 13:04:09 +01:00
|
|
|
}));
|
|
|
|
|
|
|
|
const StyledMessageTypography = styled(Typography, {
|
|
|
|
shouldForwardProp: prop => prop !== 'readAt',
|
|
|
|
})<{ readAt: boolean }>(({ theme, readAt }) => ({
|
|
|
|
fontSize: theme.fontSizes.smallBody,
|
|
|
|
fontWeight: readAt ? 'normal' : 'bold',
|
|
|
|
textDecoration: 'none',
|
|
|
|
color: 'inherit',
|
|
|
|
}));
|
|
|
|
|
|
|
|
const StyledTimeAgoTypography = styled(Typography)(({ theme }) => ({
|
|
|
|
fontSize: theme.fontSizes.smallerBody,
|
|
|
|
color: theme.palette.neutral.main,
|
|
|
|
}));
|
|
|
|
|
2023-03-01 16:05:39 +01:00
|
|
|
const StyledUserContainer = styled(Box)(({ theme }) => ({
|
|
|
|
...flexRow,
|
|
|
|
}));
|
|
|
|
|
|
|
|
const StyledAvatar = styled(Avatar)(({ theme }) => ({
|
|
|
|
width: '18px',
|
|
|
|
height: '18px',
|
|
|
|
}));
|
|
|
|
|
|
|
|
const StyledCreatedBy = styled(Typography)(({ theme }) => ({
|
|
|
|
fontSize: theme.fontSizes.smallerBody,
|
|
|
|
color: theme.palette.neutral.main,
|
|
|
|
marginLeft: theme.spacing(1),
|
|
|
|
}));
|
|
|
|
|
2023-02-27 13:04:09 +01:00
|
|
|
interface INotificationProps {
|
|
|
|
notification: NotificationsSchemaItem;
|
|
|
|
onNotificationClick: (notification: NotificationsSchemaItem) => void;
|
|
|
|
}
|
|
|
|
|
|
|
|
export const Notification = ({
|
|
|
|
notification,
|
|
|
|
onNotificationClick,
|
|
|
|
}: INotificationProps) => {
|
|
|
|
const theme = useTheme();
|
|
|
|
const { readAt } = notification;
|
|
|
|
|
|
|
|
const resolveIcon = (type: NotificationsSchemaItemNotificationType) => {
|
|
|
|
if (type === 'change-request') {
|
|
|
|
return (
|
|
|
|
<StyledContainerBox readAt={Boolean(readAt)}>
|
|
|
|
<ChangesAppliedIcon
|
|
|
|
color={
|
|
|
|
notification.readAt
|
|
|
|
? theme.palette.neutral.main
|
|
|
|
: theme.palette.primary.main
|
|
|
|
}
|
|
|
|
style={{ transform: 'scale(0.8)' }}
|
|
|
|
/>
|
|
|
|
</StyledContainerBox>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type === 'toggle') {
|
|
|
|
return (
|
|
|
|
<StyledContainerBox readAt={Boolean(readAt)}>
|
|
|
|
<ToggleOffOutlined
|
|
|
|
sx={theme => ({
|
|
|
|
height: '20px',
|
|
|
|
width: '20px',
|
|
|
|
color: Boolean(readAt)
|
|
|
|
? theme.palette.neutral.main
|
|
|
|
: theme.palette.primary.main,
|
|
|
|
})}
|
|
|
|
/>
|
|
|
|
</StyledContainerBox>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
2023-03-01 14:28:05 +01:00
|
|
|
<StyledListItemButton onClick={() => onNotificationClick(notification)}>
|
2023-02-27 13:04:09 +01:00
|
|
|
{resolveIcon(notification.notificationType)}{' '}
|
|
|
|
<StyledNotificationMessageBox>
|
|
|
|
<StyledMessageTypography readAt={Boolean(readAt)}>
|
|
|
|
{notification.message}
|
|
|
|
</StyledMessageTypography>
|
|
|
|
<StyledSecondaryInfoBox>
|
2023-03-01 16:05:39 +01:00
|
|
|
<StyledUserContainer>
|
|
|
|
<StyledAvatar
|
|
|
|
src={notification.createdBy.imageUrl || ''}
|
|
|
|
/>
|
|
|
|
<StyledCreatedBy>
|
|
|
|
Created by {notification.createdBy.username}
|
|
|
|
</StyledCreatedBy>
|
|
|
|
</StyledUserContainer>
|
|
|
|
|
2023-02-27 13:04:09 +01:00
|
|
|
<StyledTimeAgoTypography>
|
|
|
|
<TimeAgo date={new Date(notification.createdAt)} />
|
|
|
|
</StyledTimeAgoTypography>
|
|
|
|
</StyledSecondaryInfoBox>
|
|
|
|
</StyledNotificationMessageBox>
|
2023-03-01 14:28:05 +01:00
|
|
|
</StyledListItemButton>
|
2023-02-27 13:04:09 +01:00
|
|
|
);
|
|
|
|
};
|