index.tsx 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. import { makeStyles } from "@material-ui/core/styles";
  2. import { useState, useEffect, useRef, useCallback } from "react";
  3. import { useSelector,useDispatch } from "react-redux";
  4. import SendMessage from "./SendMessage";
  5. import MessageLeftText from './Messages/MessageLeftText'
  6. import MessageLeftImage from './Messages/MessageLeftImage'
  7. import MessageLeftAudio from './Messages/MessageLeftAudio'
  8. import MessageLeftVideo from './Messages/MessageLeftVideo'
  9. import MessageLeftFile from "./Messages/MessageLeftFile";
  10. import MessageRightText from './Messages/MessageRightText'
  11. import MessageRightImage from './Messages/MessageRightImage'
  12. import MessageRightAudio from './Messages/MessageRightAudio'
  13. import MessageRightVideo from './Messages/MessageRightVideo'
  14. import MessageRightFile from "./Messages/MessageRightFile";
  15. import MessageTime from "./Messages/MessageTime";
  16. import AlertInfo from "../../../reusableComponents/AlertInfo";
  17. import { getMessagesMemo } from '../../../../redux/messages/selector'
  18. import { getNumber } from '../../../../redux/authorization/selector'
  19. import { getChat } from '../../../../redux/chat/selector'
  20. import { getScroll } from '../../../../redux/control/selector'
  21. import { actionScroll } from '../../../../redux/control/action'
  22. import { asyncGetMessagesById } from '../../../../redux/messages/operations'
  23. import { seenChat } from "../../../../api-data";
  24. import { timeStampFilter } from "../../../../helpers";
  25. const debounce = require('lodash.debounce');
  26. const useStyles = makeStyles({
  27. container: {
  28. width: "100%",
  29. height: "100%",
  30. maxWidth: "100%",
  31. maxHeight: "100%",
  32. display: "flex",
  33. alignItems: "center",
  34. alignContent:"center",
  35. flexDirection: "column",
  36. position: "relative",
  37. overflowY: "scroll",
  38. paddingTop: 60,
  39. },
  40. messagesBody: {
  41. width: "60%",
  42. height: "80%",
  43. },
  44. });
  45. const ChatBar = () => {
  46. const classes = useStyles();
  47. const dispatch = useDispatch()
  48. const messages = useSelector(getMessagesMemo)
  49. const userNumber = useSelector(getNumber)
  50. const { companionId } = useSelector(getChat)
  51. const scroll = useSelector(getScroll)
  52. const [isArrow, setIsArrow] = useState<boolean>(false)
  53. const [isNew, setIsNew] = useState<{new:number,mute:boolean}>({new:0,mute:false})
  54. const divRef = useRef<any | null>(null)
  55. let time:any
  56. const handleScrollTo = () => {
  57. divRef.current&&divRef.current.scrollTo({
  58. top: divRef.current.scrollHeight,
  59. behavior: 'smooth'
  60. })
  61. }
  62. const handleScroll = ({ target }: any) => {
  63. const different = target.scrollHeight - target.scrollTop
  64. if (different < 900) seenChat(companionId)
  65. setIsArrow(different < 1500 ? false : true)
  66. }
  67. const debouncedHandleScroll = useCallback(debounce(handleScroll, 300), []);
  68. useEffect(() => {
  69. if (scroll) {
  70. seenChat(companionId)
  71. dispatch(asyncGetMessagesById(companionId, handleScrollTo))
  72. dispatch(actionScroll(false))
  73. }
  74. }, [dispatch,scroll, companionId])
  75. // () =>
  76. // divRef.current.scrollHeight < 900 && !watched&& seenChat(companionId))
  77. useEffect(() => {
  78. dispatch(asyncGetMessagesById(companionId, handleScrollTo))
  79. const handleReset = () => {
  80. dispatch(asyncGetMessagesById(companionId, null))
  81. const arr: any = localStorage.getItem('isNew')
  82. if(arr) setIsNew(JSON.parse(arr))
  83. }
  84. const idInterval = setInterval(handleReset, 3000);
  85. return () => clearInterval(idInterval);
  86. }, [dispatch, companionId]);
  87. return (
  88. <div ref={divRef} className={classes.container} onScroll={debouncedHandleScroll}>
  89. <div className={classes.messagesBody}>
  90. {messages.length > 0 ? messages.map(({ message, name, lastName, color,
  91. updatedAt, createdAt,number, type,fullType }) => {
  92. let isTime
  93. if (!time) {
  94. isTime = true
  95. time = updatedAt
  96. } else if (timeStampFilter(time) !== timeStampFilter(updatedAt)) {
  97. time = updatedAt
  98. isTime = true
  99. }
  100. const url = `http://localhost:3000/${message}`
  101. if (number !== userNumber) {
  102. if (type === 'text') return (<div key={createdAt}>
  103. {isTime&&<MessageTime message={timeStampFilter(updatedAt)}/>}
  104. <MessageLeftText
  105. message={message}
  106. updatedAt={updatedAt}
  107. name={name}
  108. lastName={lastName}
  109. /></div>)
  110. if (type === 'image') return (<div key={createdAt}>
  111. {isTime&&<MessageTime message={timeStampFilter(updatedAt)}/>}
  112. <MessageLeftImage
  113. url={url}
  114. updatedAt={updatedAt}
  115. color={color}
  116. message={message}
  117. messages={messages}
  118. fullType={fullType}
  119. /></div>)
  120. if (type === 'audio') return (<div key={createdAt}>
  121. {isTime&&<MessageTime message={timeStampFilter(updatedAt)}/>}
  122. <MessageLeftAudio
  123. url={url}
  124. updatedAt={updatedAt}
  125. fullType={fullType}
  126. /></div>)
  127. if (type === 'video') return (<div key={createdAt}>
  128. {isTime&&<MessageTime message={timeStampFilter(updatedAt)}/>}
  129. <MessageLeftVideo
  130. url={url}
  131. updatedAt={updatedAt}
  132. fullType={fullType}
  133. /></div>)
  134. if (type) return (<div key={createdAt}>
  135. {isTime&&<MessageTime message={timeStampFilter(updatedAt)}/>}
  136. <MessageLeftFile
  137. url={url}
  138. updatedAt={updatedAt}
  139. type={type}
  140. /></div>)
  141. } else {
  142. if (type === 'text') return (<div key={createdAt}>
  143. {isTime&&<MessageTime message={timeStampFilter(updatedAt)}/>}
  144. <MessageRightText
  145. message={message}
  146. updatedAt={updatedAt}
  147. name={name}
  148. lastName={lastName}
  149. /></div>)
  150. if (type === 'image') return (<div key={createdAt}>
  151. {isTime&&<MessageTime message={timeStampFilter(updatedAt)}/>}
  152. <MessageRightImage
  153. url={url}
  154. updatedAt={updatedAt}
  155. color={color}
  156. message={message}
  157. messages={messages}
  158. fullType={fullType}
  159. /></div>)
  160. if (type === 'audio') return (<div key={createdAt}>
  161. {isTime&&<MessageTime message={timeStampFilter(updatedAt)}/>}
  162. <MessageRightAudio
  163. url={url}
  164. updatedAt={updatedAt}
  165. fullType={fullType}
  166. /></div>)
  167. if (type === 'video') return (<div key={createdAt}>
  168. {isTime&&<MessageTime message={timeStampFilter(updatedAt)}/>}
  169. <MessageRightVideo
  170. url={url}
  171. updatedAt={updatedAt}
  172. fullType={fullType}
  173. /></div>)
  174. if (type) return (<div key={createdAt}>
  175. {isTime&&<MessageTime message={timeStampFilter(updatedAt)}/>}
  176. <MessageRightFile
  177. url={url}
  178. updatedAt={updatedAt}
  179. type={type}
  180. /></div>)
  181. }
  182. }) : <AlertInfo name='You do not have messages yet!' />}
  183. </div>
  184. <SendMessage isArrow={isArrow} isNew={isNew} handleScrollTo={handleScrollTo}/>
  185. </div>
  186. );
  187. }
  188. export default ChatBar