ChatPage.jsx 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. import { useEffect, useState} from 'react';
  2. import {Button, Box} from '@mui/material';
  3. import TextareaAutosize from '@mui/material/TextareaAutosize';
  4. import { MessageForm } from './messageForm/MessegaForm';
  5. import { UserInfo } from './userInfo/UserInfo';
  6. import { store } from '../../store';
  7. import { removeToken, isPrivatChat, privateMessage} from '../../reducers/userDataReducer'
  8. import { useDispatch, useSelector } from 'react-redux';
  9. import {getSocket} from'../../reducers/socketReducer';
  10. import { sendMessage, storeMessage, fileMessage } from '../../reducers/messageReducer';
  11. import { editMessage } from '../../reducers/messageReducer';
  12. import { SwitchButton } from './SwitchButton';
  13. import { MessageEditorMenu } from './MessageEditorMenu.jsx';
  14. import imgBtn from '../../assets/img/gg.png';
  15. import imgBtnPhoto from '../../assets/img/photo.png'
  16. import './chatPage.scss';
  17. import WebcamCapture from './service/webCam/WebcamCapture';
  18. import useSound from 'use-sound';
  19. import getNotif from '../../assets/sendSound.mp3'
  20. import { PrivateChat } from './privateChat/PrivateChat';
  21. import { PrivatChatHeader } from './privateChat/PrivatChatHeader';
  22. import { socketEvents } from '../../utils/socketsEvents';
  23. export const ChatPage = () => {
  24. console.log('render')
  25. const dispatch = useDispatch();
  26. const token = useSelector(state => localStorage.getItem('token') || state.userDataReducer.token);
  27. const user = useSelector(state => state.getUserSocketReducer.socketUserData)
  28. const socket = useSelector(state => state.getUserSocketReducer.socket)
  29. const editOldMessage = useSelector(state => state.messageReducer.editMessage)
  30. let showUserInfoBox = useSelector(state => state.messageReducer.showUserInfoBox)// || localStorage.getItem('showBox');
  31. const toUser = useSelector(state => state.userDataReducer.toUser)
  32. const chatId = useSelector(state => state.userDataReducer.toUser.socketId)
  33. const isPrivatChat = useSelector(state => state.userDataReducer.isPrivatChat)
  34. const newPrivateMessages = useSelector(state => state.getUserSocketReducer.newPrivateMessages)
  35. const [message, setMessage] = useState({message: ''});
  36. const [isUserTyping, setUserTyping] = useState([]);
  37. const [isCamActiv, setisCamActiv] = useState(false);
  38. const [showSpinner, setshowSpinner] = useState(false);
  39. const [loadingPercentage, setLoadPercentage] = useState(0)
  40. const isTabletorMobile = (window.screen.width < 730);
  41. const isNewMessage = newPrivateMessages.length > 0
  42. const [play] = useSound(getNotif, {volume: 0.005});
  43. const axiosConfig = {
  44. headers: {
  45. "Content-type": "multipart/form-data"
  46. },
  47. onUploadProgress: (progress) => {
  48. const {loaded, total} = progress;
  49. const loadStatus = Math.floor(loaded * 100 / total);
  50. setLoadPercentage(loadStatus)
  51. if(loadStatus == 100) {
  52. setshowSpinner(false)
  53. }
  54. }}
  55. const webcamEventHandler = async () => {
  56. let stream = null;
  57. try {
  58. stream = await navigator.mediaDevices.getUserMedia({ video: { width: 300 }});
  59. if (stream){
  60. setisCamActiv(!isCamActiv)
  61. }
  62. } catch(err) {
  63. console.log(err)
  64. }
  65. setisCamActiv(!isCamActiv) // test camera
  66. }
  67. console.log('************render*************')
  68. const sendPrivateMessage = () => {
  69. console.log(toUser.socketId)
  70. socket.emit("private message", {
  71. fromUser: user,
  72. message: message.message,
  73. to: chatId,
  74. toUser
  75. })
  76. }
  77. useEffect(() => {
  78. if(socket) {
  79. socket.on('writing', (data) => {
  80. setUserTyping(data)
  81. setTimeout(() => setUserTyping([]), 500 )
  82. })
  83. }
  84. }, [socket, token])
  85. useEffect(() => {
  86. if(token && socket){
  87. socketEvents(socket)
  88. }
  89. }, [token, socket, user])
  90. return (
  91. <div className='rootContainer'>
  92. <Box className = {isTabletorMobile?'mobileRootBox':'rootBox'}>
  93. <Box className={isTabletorMobile?'usersBoxMobile':'usersBox'}
  94. sx = {showUserInfoBox ? {
  95. transform: "translateX(100%)",
  96. display: "block"
  97. }: { }}>
  98. <UserInfo/>
  99. { isTabletorMobile ? <SwitchButton/> : null}
  100. <Button
  101. style={isTabletorMobile ?
  102. {
  103. maxHeight:'20px',
  104. maxWidth: '15px',
  105. fontSize: '10px',
  106. marginLeft: '25px',
  107. marginRight: '10px'}
  108. :{margin:'10px 5px'}}
  109. variant="contained"
  110. onClick={()=> {
  111. localStorage.removeItem('token');
  112. dispatch(removeToken());
  113. socket.disconnect();
  114. }}>
  115. Logout
  116. </Button>
  117. </Box>
  118. <Box className ={isTabletorMobile ? 'rootMessageFormMobile':'rootMessageForm'} >
  119. {isCamActiv ?
  120. <div>
  121. <Button
  122. variant="contained"
  123. component="label"
  124. onClick = {() => webcamEventHandler()}
  125. >
  126. close camera
  127. </Button>
  128. <WebcamCapture />
  129. </div>
  130. :
  131. ""}
  132. {isPrivatChat? <PrivateChat/> : <MessageForm/>}
  133. {isUserTyping.isTyping && (isUserTyping.userName !== user.userName)? <span> User {isUserTyping.userName} typing..</span> : ""}
  134. <Box
  135. component="form"
  136. onSubmit = {e => {
  137. e.preventDefault()
  138. if (message.message.length){
  139. isPrivatChat? sendPrivateMessage() : dispatch(sendMessage({user, socket}));
  140. // isPrivatChat && dispatch(getSocket('allmessages'))
  141. isPrivatChat &&dispatch(editMessage({editMessage: ''}))
  142. setMessage({message: ''})
  143. play()
  144. }
  145. }}
  146. sx={(isTabletorMobile)?{
  147. display: 'flex',
  148. margin: '10px 2px'}
  149. :{
  150. display: 'flex',
  151. margin: '20px 5px'}
  152. }>
  153. <Button
  154. disabled={showSpinner}
  155. variant="contained"
  156. component="label"
  157. sx = {showSpinner?
  158. {
  159. minWidth: 'auto'}
  160. :
  161. {
  162. minWidth: 'auto',
  163. backgroundImage:'url(' + imgBtn + ')' ,
  164. backgroundPosition: 'center',
  165. backgroundRepeat: "no-repeat",
  166. backgroundSize: '20px 20px'
  167. }}
  168. style = {{color: 'green'}}
  169. >
  170. <input
  171. onChange={e =>{
  172. setshowSpinner(true)
  173. dispatch(fileMessage({files: e.target.files, axiosConfig}))
  174. }}
  175. type="file"
  176. multiple
  177. hidden
  178. />
  179. {showSpinner? loadingPercentage : ""}
  180. </Button>
  181. <Button
  182. variant="contained"
  183. component="label"
  184. sx = {{
  185. minWidth: 'auto',
  186. backgroundImage:'url(' + imgBtnPhoto + ')' ,
  187. backgroundPosition: 'center',
  188. backgroundRepeat: "no-repeat",
  189. backgroundSize: '20px 20px'
  190. }}
  191. onClick = {() => webcamEventHandler()}
  192. >
  193. </Button>
  194. <TextareaAutosize
  195. id="outlined-basic"
  196. label="Type a message..."
  197. variant="outlined"
  198. value={message.message}
  199. placeholder='type you message...'
  200. minRows={2}
  201. maxRows={4}
  202. className='textArea'
  203. onKeyPress={(e) => {
  204. if (e.key === "Enter") {
  205. e.preventDefault();
  206. isPrivatChat? sendPrivateMessage() : dispatch(sendMessage({user, socket}));
  207. //dispatch(getSocket('allmessages'))
  208. dispatch(editMessage({editMessage: ''}))
  209. setMessage({message: ''})
  210. }
  211. }}
  212. onChange={e => {
  213. dispatch(storeMessage({message: e.target.value}))
  214. !isPrivatChat &&socket.emit('userWriting');
  215. setMessage({message: e.target.value})}
  216. }
  217. // onFocus={ e => {
  218. // if (isTabletorMobile) {
  219. // e.target.className = 'focus'
  220. // }
  221. // }}
  222. // onBlur={e=> {
  223. // e.target.className = 'blur'
  224. // }}
  225. />
  226. <Button
  227. variant="contained"
  228. type='submit'
  229. disabled={user?.isMutted || !message.message.length}
  230. style={{width: '20%'}}
  231. >
  232. Send
  233. </Button>
  234. </Box>
  235. </Box>
  236. </Box>
  237. </div>
  238. )
  239. }