index.tsx 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. import Stack from '@mui/material/Stack';
  2. import IconButton from '@mui/material/IconButton';
  3. import MenuItem from '@mui/material/MenuItem';
  4. import ListItemText from '@mui/material/ListItemText';
  5. import CloseIcon from '@mui/icons-material/Close';
  6. import MenuOpenIcon from '@mui/icons-material/MenuOpen';
  7. import Button from '@mui/material/Button';
  8. import Avatar from '@mui/material/Avatar';
  9. import ListItemIcon from '@mui/material/ListItemIcon';
  10. import LibraryMusicIcon from '@mui/icons-material/LibraryMusic';
  11. import FolderIcon from '@mui/icons-material/Folder';
  12. import ImageIcon from '@mui/icons-material/Image';
  13. import ContentCopyIcon from '@mui/icons-material/ContentCopy';
  14. import VideoLibraryIcon from '@mui/icons-material/VideoLibrary';
  15. import ListItemAvatar from '@mui/material/ListItemAvatar';
  16. import { CopyToClipboard } from 'react-copy-to-clipboard';
  17. import { makeStyles } from '@material-ui/core'
  18. import { useSelector } from 'react-redux';
  19. import { useState,useEffect} from 'react';
  20. import { TMessages,TMessage } from '../../../../../typescript/redux/messages/types';
  21. import { getMessagesMemo } from '../../../../../redux/messages/selector';
  22. import { firstLetter,slicedWord,handleSort,prodAwsS3,copied,handleDownload } from '../../../../../helpers';
  23. import { pinMessageById } from '../../../../../api-data';
  24. const useStyles = makeStyles({
  25. container: {
  26. marginLeft: 20,
  27. display: 'flex',
  28. alignContent: 'center',
  29. alignItems:'center'
  30. },
  31. iconClose: {
  32. '&:hover': {
  33. transform: 'rotate(180deg)',
  34. transition: 'all 250ms ease-out ',
  35. }
  36. },
  37. listWrapper: {
  38. background: '#fdfdfd',
  39. padding:0
  40. },
  41. overlay: {
  42. position: 'fixed',
  43. top: 0,
  44. left: 0,
  45. width: '100vw',
  46. height: '100vh',
  47. zIndex: 100,
  48. backgroundColor: 'rgba(104, 105, 104, 0.6)',
  49. overflowY: 'hidden',
  50. },
  51. modalUnpin: {
  52. background: '#ffffff',
  53. position: 'absolute',
  54. content:'',
  55. width: '20%',
  56. height:'auto',
  57. left: '40%',
  58. bottom: '48.5%',
  59. borderRadius: 10,
  60. padding: 10,
  61. display: 'flex',
  62. flexDirection:'column'
  63. },
  64. folderIcon: {
  65. color: '#54b0fc',
  66. cursor: 'pointer',
  67. '&:hover': {
  68. color: '#016cc3'
  69. },
  70. },
  71. })
  72. const PinnedBar = ({divRef}:{divRef: any | null}) => {
  73. const classes = useStyles()
  74. const messagesMemo = useSelector(getMessagesMemo)
  75. const [pinnedArr, setPinnedArr] = useState<TMessages>([])
  76. const [openedPin, setOpenedPin] = useState<TMessage | null>(null)
  77. const [openedIndex, setOpenedIndex] = useState<number>(0)
  78. const [modal, setModal] = useState<boolean>(false)
  79. const handleActivePin = () => {
  80. const childNodes = divRef.current.childNodes[0].childNodes
  81. let toScrollNode:any
  82. if (pinnedArr.length - 1 === openedIndex) {
  83. setOpenedIndex(0)
  84. toScrollNode = [ ...childNodes ].find((el:any) => el.id === pinnedArr[0]._id)
  85. } else {
  86. setOpenedIndex(prevState => prevState + 1)
  87. toScrollNode = [...childNodes].find((el: any) => el.id === pinnedArr[openedIndex + 1]._id)
  88. }
  89. toScrollNode.style.boxShadow = '0px 0px 6px 0px #555555'
  90. toScrollNode.scrollIntoView()
  91. setTimeout(() => {
  92. toScrollNode.style.boxShadow = 'unset'
  93. }, 2000)
  94. }
  95. const handleUnpin = (e: any) => {
  96. const id = e.target.id
  97. if (id === 'overlay' || id === 'cancel') return setModal(false)
  98. if (id === 'unpin' && openedPin) {
  99. pinMessageById(openedPin._id, !openedPin.pinned)
  100. setModal(false)
  101. if (openedIndex - 1 >= 0) {
  102. return setOpenedIndex(prevState => prevState - 1)
  103. } else {
  104. setOpenedIndex(0)
  105. }
  106. }
  107. }
  108. useEffect(() => {
  109. const messages = messagesMemo.filter((el) => el.pinned === true)
  110. setPinnedArr(handleSort('updatedAt', messages, false))
  111. }, [messagesMemo])
  112. useEffect(() => {
  113. setOpenedPin(pinnedArr[openedIndex])
  114. }, [openedIndex, pinnedArr])
  115. return openedPin ?
  116. <Stack className={classes.container} direction="row">
  117. <ul className={classes.listWrapper}>
  118. <MenuItem onClick={handleActivePin}>
  119. <ListItemIcon >
  120. <Avatar alt={openedPin.name} src={openedPin.avatarUrl?`${prodAwsS3}/${openedPin.avatarUrl}`:undefined}
  121. sx={{ background: openedPin.color, width: 44, height: 44 }}>
  122. {!openedPin.avatarUrl&&`${firstLetter(openedPin.name)}${firstLetter(openedPin.lastName)}`}
  123. </Avatar>
  124. </ListItemIcon>
  125. <ListItemText style={{marginLeft:20}}
  126. primary={`${firstLetter(openedPin.name)}${slicedWord(openedPin.name, 15, 1)}
  127. ${firstLetter(openedPin.lastName)}${slicedWord(openedPin.lastName, 15, 1)}`}
  128. primaryTypographyProps={{fontSize:16 }}
  129. secondary={slicedWord(openedPin.message, 20, 1)}
  130. secondaryTypographyProps={{fontSize: 12 }} />
  131. <ListItemText style={{marginLeft:20}}
  132. primary={`Pinned Message ${openedIndex + 1} of ${pinnedArr.length}`}
  133. primaryTypographyProps={{ color: "#0379af",fontSize:16 }}
  134. secondary={`Type : ${openedPin.type.toUpperCase()}`}
  135. secondaryTypographyProps={{ fontSize:16 }}/>
  136. </MenuItem>
  137. </ul>
  138. <ListItemAvatar style={{marginLeft:10}}>
  139. {openedPin.type === 'text' &&<CopyToClipboard onCopy={() => copied('Message')} text={openedPin.message}>
  140. <ContentCopyIcon className={classes.folderIcon} fontSize='large' />
  141. </CopyToClipboard>}
  142. {openedPin.type === 'audio' &&<LibraryMusicIcon onClick={() =>
  143. handleDownload(`${prodAwsS3}/${openedPin.message}`, openedPin.fullType)}
  144. className={classes.folderIcon} fontSize='large' />}
  145. {openedPin.type === 'video' &&<VideoLibraryIcon onClick={() =>
  146. handleDownload(`${prodAwsS3}/${openedPin.message}`, openedPin.fullType)}
  147. className={classes.folderIcon} fontSize='large' />}
  148. {openedPin.type === 'image' &&<ImageIcon onClick={() =>
  149. handleDownload(`${prodAwsS3}/${openedPin.message}`, openedPin.fullType)}
  150. className={classes.folderIcon} fontSize='large' />}
  151. {openedPin.type === 'pdf' ?<FolderIcon onClick={() =>
  152. handleDownload(`${prodAwsS3}/${openedPin.message}`, openedPin.fullType)}
  153. className={classes.folderIcon} fontSize='large' />:null}
  154. {openedPin.type === 'docx' ?<FolderIcon onClick={() =>
  155. handleDownload(`${prodAwsS3}/${openedPin.message}`, openedPin.fullType)}
  156. className={classes.folderIcon} fontSize='large' />:null}
  157. </ListItemAvatar>
  158. <IconButton onClick={() => setModal(true)} aria-label="delete" size="medium">
  159. <CloseIcon className={classes.iconClose} fontSize='medium'/>
  160. </IconButton>
  161. <IconButton aria-label="delete" size="medium">
  162. <MenuOpenIcon fontSize='medium'/>
  163. </IconButton>
  164. {modal &&
  165. <div onClick={handleUnpin} className={classes.overlay} id='overlay'>
  166. <div className={classes.modalUnpin}>
  167. <h3 style={{color: '#2c2c2c'}}>Telegram</h3>
  168. <p style={{ color: '#050505' }}>Would you like to unpin this message?</p>
  169. <Button id='unpin' variant="text" color="error" style={{fontWeight:500,fontSize:22}}>
  170. UNPIN
  171. </Button>
  172. <Button id='cancel' variant="text" style={{fontWeight:500,fontSize:22}}>
  173. CANCEL
  174. </Button>
  175. </div>
  176. </div>}
  177. </Stack> :
  178. null
  179. }
  180. export default PinnedBar