index.tsx 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. import List from '@mui/material/List';
  2. import ListItemButton from '@mui/material/ListItemButton';
  3. import Avatar from '@mui/material/Avatar';
  4. import ListItemText from '@mui/material/ListItemText';
  5. import ListItemIcon from '@mui/material/ListItemIcon';
  6. import { styled } from '@mui/material/styles';
  7. import Badge from '@mui/material/Badge';
  8. import VolumeOffIcon from '@mui/icons-material/VolumeOff';
  9. import { makeStyles, Typography } from '@material-ui/core'
  10. import { useState,useEffect,useRef } from 'react';
  11. import { useSelector, useDispatch } from 'react-redux';
  12. import AlertInfo from '../../../reusableComponents/AlertInfo'
  13. import DoneAllIcon from '@mui/icons-material/DoneAll';
  14. import { firstLetter, slicedWord, timeStampEU,notification,playNotificationWithoutPermission } from '../../../../helpers'
  15. import { getState } from '../../../../redux/chats/selector'
  16. import { getChatMemo } from '../../../../redux/chat/selector'
  17. import { asyncGetChats } from '../../../../redux/chats/operations'
  18. import { asyncStartChatById } from '../../../../redux/chat/operations'
  19. import { actionRemoveChat } from '../../../../redux/chat/action'
  20. import { actionScroll } from '../../../../redux/control/action'
  21. const StyledBadge = styled(Badge)(({ theme }) => ({
  22. '& .MuiBadge-badge': {
  23. backgroundColor: '#44b700',
  24. color: '#44b700',
  25. boxShadow: `0 0 0 2px ${theme.palette.background.paper}`,
  26. '&::after': {
  27. position: 'absolute',
  28. top: 0,
  29. left: 0,
  30. width: '100%',
  31. height: '100%',
  32. borderRadius: '50%',
  33. animation: 'ripple 1.2s infinite ease-in-out',
  34. border: '1px solid currentColor',
  35. content: '""',
  36. },
  37. },
  38. '@keyframes ripple': {
  39. '0%': {
  40. transform: 'scale(.8)',
  41. opacity: 1,
  42. },
  43. '100%': {
  44. transform: 'scale(2.4)',
  45. opacity: 0,
  46. },
  47. },
  48. }));
  49. const useStyles = makeStyles({
  50. list: {
  51. width: '100%',
  52. maxHeight: '890px',
  53. overflowY: 'scroll',
  54. '&::-webkit-scrollbar': {
  55. width: '0.4em'
  56. },
  57. '&::-webkit-scrollbar-track': {
  58. boxShadow: 'inset 0 0 6px rgba(0,0,0,0.00)',
  59. webkitBoxShadow: 'inset 0 0 6px rgba(0,0,0,0.00)',
  60. backgroundColor: '#eceeec',
  61. },
  62. '&::-webkit-scrollbar-thumb': {
  63. backgroundColor: '#ccc8c8',
  64. },
  65. "&::-webkit-scrollbar-thumb:focus": {
  66. backgroundColor: "#959595",
  67. },
  68. "&::-webkit-scrollbar-thumb:active": {
  69. backgroundColor: "#959595",
  70. },
  71. },
  72. listItemInnerText: {
  73. display: 'flex',
  74. alignContent: 'center',
  75. alignItems: 'center',
  76. flexWrap: 'nowrap',
  77. },
  78. listItemInnerText__icon: {
  79. marginLeft: 5,
  80. color: '#959595',
  81. },
  82. listItem_iconAvatar: {
  83. marginRight:10
  84. },
  85. listItem_iconRight: {
  86. marginRight: 10,
  87. display: 'flex',
  88. alignItems: 'center',
  89. justifyContent: 'center',
  90. alignContent: 'center',
  91. flexDirection: 'column'
  92. },
  93. listItem_iconTimeChecked: {
  94. display: 'flex',
  95. flexWrap: 'nowrap',
  96. alignItems: 'center',
  97. justifyContent: 'center',
  98. alignContent: 'center',
  99. marginBottom:2
  100. },
  101. listItem_iconRightBtn: {
  102. background: '#0ac40a',
  103. borderRadius: '50%',
  104. color: '#ffffff',
  105. border: 'none',
  106. height: 24,
  107. width: 24,
  108. textAlign: 'center',
  109. display: 'flex',
  110. alignItems: 'center',
  111. justifyContent: 'center',
  112. alignContent: 'center',
  113. fontSize: 12,
  114. marginLeft: 'auto',
  115. '&:hover': {
  116. outline: 'solid 3px #3ee415',
  117. }
  118. },
  119. listItem_iconRightBtnMute: {
  120. background: '#a7aaa7',
  121. borderRadius: '50%',
  122. color: '#ffffff',
  123. border: 'none',
  124. height: 24,
  125. width: 24,
  126. textAlign: 'center',
  127. display: 'flex',
  128. alignItems: 'center',
  129. justifyContent: 'center',
  130. alignContent: 'center',
  131. fontSize: 12,
  132. marginLeft: 'auto',
  133. '&:hover': {
  134. outline: 'solid 3px #cccbcb',
  135. }
  136. },
  137. listItem_iconRightBtnHidden: {
  138. background: 'inherit',
  139. borderRadius: '50%',
  140. border: 'none',
  141. height: 24,
  142. width: 24,
  143. textAlign: 'center',
  144. display: 'flex',
  145. alignItems: 'center',
  146. justifyContent: 'center',
  147. alignContent: 'center',
  148. fontSize: 12,
  149. marginLeft: 'auto',
  150. },
  151. listItem_icon_time: {
  152. fontSize: 12,
  153. marginLeft: 5,
  154. color: '#1b1b1b'
  155. },
  156. listItem_typing: {
  157. color: '#4d4d4d',
  158. animation: 'ripple 4s infinite ease-in-out',
  159. },
  160. listItem_dots: {
  161. color: '#1b1b1b',
  162. fontWeight: 'bold',
  163. display:'inline-block',
  164. fontFamily: 'monospace',
  165. clipPath: 'inset(0 3ch 0 0)',
  166. animation: `$run 2s steps(5) infinite`,
  167. },
  168. '@keyframes run': {
  169. to: {
  170. clipPath: 'inset(0 -1ch 0 0)'
  171. },
  172. },
  173. })
  174. const ChatsList = () => {
  175. const classes = useStyles()
  176. const dispatch = useDispatch()
  177. const ref = useRef<any>(null)
  178. const [selectedIndex, setSelectedIndex] = useState<number>(1);
  179. const { total, chats } = useSelector(getState)
  180. const chat = useSelector(getChatMemo)
  181. const handleListItemClick = (i: number, companionId: string) => {
  182. dispatch(asyncStartChatById(companionId))
  183. dispatch(actionScroll(false))
  184. setSelectedIndex(i);
  185. }
  186. const handleNewMsgs = (e: React.MouseEvent<HTMLButtonElement, MouseEvent>, i: number,companionId: string) => {
  187. e.stopPropagation()
  188. dispatch(asyncStartChatById(companionId))
  189. dispatch(actionScroll(true))
  190. }
  191. useEffect(() => {
  192. dispatch(asyncGetChats())
  193. const handleReset = () => dispatch(asyncGetChats())
  194. const idInterval = setInterval(handleReset, 3000);
  195. return () => clearInterval(idInterval);
  196. }, [dispatch]);
  197. useEffect(() => {
  198. const handleNotification= (companionId: string) => {
  199. dispatch(asyncStartChatById(companionId))
  200. dispatch(actionScroll(true))
  201. }
  202. if (chat.companionId&&!chats.find((el) => el.companionId === chat.companionId)) {
  203. dispatch(actionRemoveChat())
  204. }
  205. if (ref.current) {
  206. ref.current.forEach(({total,seen}: any,i:number) => {
  207. const oldDifferent = total - seen
  208. const chat = chats[i]
  209. if(chat === undefined) return
  210. const newDifferent = chat.total - chat.seen
  211. if (newDifferent > oldDifferent && !chat.mute) {
  212. playNotificationWithoutPermission('http://localhost:3000/recive.mp3')
  213. notification(chat.name,() => handleNotification(chat.companionId))
  214. }
  215. })
  216. }
  217. ref.current = chats
  218. }, [chats,chat,dispatch])
  219. return total !== '0' ? (
  220. <List className={classes.list} component="nav"
  221. aria-label="main mailbox folders">
  222. {chats.map(({ name, lastName, avatarUrl, updatedAt, color, companionId, mute, seen, total,
  223. watched, typing, number, online, lastMessage }, i: number) =>
  224. <ListItemButton
  225. key={number}
  226. selected={selectedIndex === i}
  227. onClick={() => handleListItemClick(i,companionId)}
  228. >
  229. <ListItemIcon className={classes.listItem_iconAvatar}>
  230. <StyledBadge overlap="circular" variant={online === 'true'?'dot':'standard'}
  231. anchorOrigin={{ vertical: 'bottom', horizontal: 'right' }}>
  232. <Avatar alt={name} src={avatarUrl?`http://localhost:3000/${avatarUrl}`:undefined}
  233. sx={{ background: color, width: 54, height: 54 }}>
  234. {!avatarUrl&&`${firstLetter(name)}${firstLetter(lastName)}`}
  235. </Avatar>
  236. </StyledBadge>
  237. </ListItemIcon>
  238. <ListItemText primary={<div className={classes.listItemInnerText}>
  239. <span>{`${firstLetter(name)}${slicedWord(name, 15, 1)}
  240. ${firstLetter(lastName)}${slicedWord(lastName, 15, 1)}`}</span>
  241. {mute&&<VolumeOffIcon className={classes.listItemInnerText__icon} fontSize='small' />}</div>}
  242. secondary={typing ? <span className={classes.listItem_typing}>
  243. typing<span className={classes.listItem_dots}>...</span></span> :
  244. lastMessage ? slicedWord(lastMessage, 35) :
  245. `${firstLetter(name)}${slicedWord(name, 8, 1)} joined Telegram`}/>
  246. <ListItemIcon className={classes.listItem_iconRight}>
  247. <div className={classes.listItem_iconTimeChecked}>
  248. {watched&& <DoneAllIcon style={{ color: '#18bd03' }} fontSize='small' />}
  249. <Typography className={classes.listItem_icon_time} variant="h6" color="initial">
  250. {timeStampEU(updatedAt)}
  251. </Typography>
  252. </div>
  253. {lastMessage && total > seen ? <button onClick={(e) => handleNewMsgs(e, i,companionId)}
  254. className={mute?classes.listItem_iconRightBtnMute:classes.listItem_iconRightBtn}>{total-seen}</button> :
  255. <button className={classes.listItem_iconRightBtnHidden}/>}
  256. </ListItemIcon>
  257. </ListItemButton>)}
  258. </List>
  259. ):<AlertInfo name='You do not have any chats yet!' />;
  260. }
  261. export default ChatsList