Proyectos de Subversion LeadersLinked - Antes de SPA

Rev

Rev 7323 | Rev 7325 | Ir a la última revisión | | Comparar con el anterior | Ultima modificación | Ver Log |

Rev Autor Línea Nro. Línea
7317 stevensc 1
import React, { useEffect, useRef, useState } from 'react'
2
import { axios } from '../../../utils'
3
import { useSelector } from 'react-redux'
4
import styled from 'styled-components'
5
 
6
import TextBox from './TextBox'
7
import ChatList from './ChatList'
8
import EmptySection from '../../UI/EmptySection'
9
import ConfirmModal from '../../modals/ConfirmModal'
10
import ConferenceModal from '../../modals/ConferenceModal'
7324 stevensc 11
import { Avatar, IconButton } from '@mui/material'
7322 stevensc 12
import { Add, ArrowBackIos } from '@mui/icons-material'
13
import SearchInput from '../../UI/SearchInput'
7317 stevensc 14
 
15
const StyledShowOptions = styled.div`
16
  height: 342px;
17
  flex-direction: column;
18
  gap: 0.5rem;
19
  overflow-y: auto;
20
  position: relative;
21
  &.show {
22
    display: flex;
23
  }
24
  &.hide {
25
    display: none;
26
  }
7324 stevensc 27
  .option-tab {
28
    align-items: center;
29
    border-bottom: 1px solid #e2e2e2;
7317 stevensc 30
    cursor: pointer;
7324 stevensc 31
    display: flex;
32
    gap: 0.5rem;
33
    padding: 0.5rem;
34
    &:hover {
35
      background-color: #e2e2e2;
7317 stevensc 36
    }
37
  }
7322 stevensc 38
  .contact-tab {
39
    align-items: center;
40
    border-bottom: 1px solid #e2e2e2;
7317 stevensc 41
    display: flex;
7322 stevensc 42
    justify-content: space-between;
43
    padding: 0.2rem 0.5rem;
44
    .info {
7317 stevensc 45
      display: flex;
46
      align-items: center;
7322 stevensc 47
      gap: 0.5rem;
7317 stevensc 48
    }
49
  }
50
`
51
 
7324 stevensc 52
const StyledComeBack = styled(IconButton)`
53
  border-radius: 0;
54
  font-size: 1rem;
55
  flex-grow: 1;
56
`
57
 
7317 stevensc 58
const OPTIONS = {
59
  GROUP: 'group',
60
  CONFERENCE: 'conference',
61
  ADD_CONTACTS: 'addContacts',
62
  LIST_CONTACTS: 'listContacts',
63
  INITIAL: null,
64
}
65
 
66
const PersonalChat = ({
67
  entity,
68
  onClose,
69
  onMinimize,
70
  onRead,
71
  not_seen_messages,
72
  minimized,
73
}) => {
74
  const [optionTab, setOptionTab] = useState(OPTIONS.INITIAL)
75
  const [availableContactsToAdd, setAvailableContactsToAdd] = useState([])
76
  const [groupContactsList, setGroupContactsList] = useState([])
77
  const [confirmModalShow, setConfirmModalShow] = useState(false)
78
  const [showConferenceModal, setShowConferenceModal] = useState(false)
79
  const [search, setSearch] = useState('')
80
  const [messages, setMessages] = useState([])
81
  const [oldMessages, setOldMessages] = useState([])
82
  const [currentPage, setCurrentPage] = useState(1)
83
  const [lastPage, setLastPage] = useState(1)
84
  const [loading, setLoading] = useState(false)
85
  const labels = useSelector(({ intl }) => intl.labels)
86
 
87
  const filtredGroupList = groupContactsList.filter((conversation) =>
88
    conversation.name.toLowerCase().includes(search.toLowerCase())
89
  )
90
 
91
  // refs
92
  const conversationListEl = useRef(null)
93
  const loader = useRef(null)
94
  const modalActionUrl = useRef('')
95
 
96
  const handleActive = () => {
97
    onRead(entity)
98
    onMinimize(entity)
99
  }
100
 
101
  const handleGetMessages = async () => {
102
    setLoading(true)
103
    const response = await axios.get(entity?.url_get_all_messages)
104
    const resData = response.data
105
    if (!resData.success) {
106
      return 'ha ocurrido un error'
107
    }
108
    const updatedMessages = [...resData.data.items].reverse()
109
    const newMessages = updatedMessages.reduce((acum, updatedMessage) => {
110
      if (
111
        messages.findIndex((message) => message.id === updatedMessage.id) === -1
112
      ) {
113
        acum = [...acum, updatedMessage]
114
      }
115
      return acum
116
    }, [])
117
 
118
    if (newMessages.length > 0) {
119
      setMessages([...messages, ...newMessages])
120
      setLoading(false)
121
      setLastPage(resData.data.pages)
122
      scrollToBottom()
123
    } else {
124
      setLoading(false)
125
    }
126
  }
127
 
128
  const handleLoadMore = async () => {
129
    await axios
130
      .get(`${entity?.url_get_all_messages}?page=${currentPage}`)
131
      .then((response) => {
132
        const resData = response.data
133
        if (resData.success) {
134
          if (resData.data.page > 1) {
135
            const updatedOldMessages = [...resData.data.items].reverse()
136
            setOldMessages([...updatedOldMessages, ...oldMessages])
137
            /* scrollDownBy(100); */
138
          }
139
        }
140
      })
141
  }
142
 
143
  const handleCloseChat = () => onClose(entity)
144
 
145
  const sendMessage = async (message) => {
146
    const formData = new FormData()
147
    formData.append('message', emojione.toShort(message))
148
    await axios.post(entity?.url_send, formData).then((response) => {
149
      const resData = response.data
150
      if (resData.success) {
151
        let newMessage = resData.data
152
 
153
        entity?.online
154
          ? (newMessage = { ...newMessage, not_received: false })
155
          : (newMessage = { ...newMessage, not_received: true })
156
 
157
        setMessages([...messages, newMessage])
158
      }
159
    })
160
    // await handleGetMessages()
161
    // setResponseMessage(null)
162
  }
163
 
164
  const showOptions = (tab = OPTIONS.INITIAL) => {
165
    onMinimize(entity, false)
166
    setOptionTab(tab)
167
  }
168
 
169
  const handleAddPersonToGroup = async (id) => {
170
    const formData = new FormData()
171
    formData.append('uid', id)
172
    await axios
173
      .post(entity?.url_add_user_to_group, formData)
174
      .then((response) => {
175
        const resData = response.data
176
        if (resData.success) {
177
          loadPersonsAvailable()
178
        }
179
      })
180
  }
181
 
182
  const handleConfirmModalAction = async () => {
183
    try {
184
      const { data } = await axios.post(modalActionUrl.current)
185
      if (!data.success) console.log('Error in confirm modal action')
186
      handleConfirmModalShow()
187
      onClose(entity)
188
      return
189
    } catch (error) {
190
      console.log(error)
191
    }
192
  }
193
 
194
  const handleObserver = (entities) => {
195
    const target = entities[0]
196
    if (target.isIntersecting) {
197
      setCurrentPage((prevState) => prevState + 1)
198
    }
199
  }
200
 
201
  const scrollToBottom = () => {
202
    if (conversationListEl.current) {
203
      conversationListEl.current.scrollTop =
204
        conversationListEl.current.scrollHeight * 9
205
    }
206
  }
207
 
208
  const handleConfirmModalShow = () => setConfirmModalShow(!confirmModalShow)
209
 
210
  const handleConfirmModalAccept = () => handleConfirmModalAction()
211
 
212
  /* const handleResponseMessage = (element) => {
213
    element.mtype === 'text'
214
      ? setResponseMessage(element)
215
      : setResponseMessage({ ...element, m: 'Archivo adjunto' })
216
 
217
    textAreaEl.current && textAreaEl.current.focus()
218
  } */
219
 
220
  const displayConferenceModal = () =>
221
    setShowConferenceModal(!showConferenceModal)
222
 
223
  const loadPersonsAvailable = async () => {
224
    await axios
225
      .get(entity?.url_get_contacts_availables_for_group)
226
      .then((response) => {
227
        const resData = response.data
228
        if (resData.success) {
229
          setAvailableContactsToAdd(resData.data)
230
        }
231
      })
232
  }
233
 
234
  const loadGroupContacts = async () => {
235
    await axios.get(entity?.url_get_contact_group_list).then((response) => {
236
      const resData = response.data
237
      if (resData.success) {
238
        setGroupContactsList(resData.data)
239
      }
240
    })
241
  }
242
 
243
  const handleDeletePersonFromGroup = async (urlDeletePersonFromGroup) => {
244
    await axios.post(urlDeletePersonFromGroup).then((response) => {
245
      const resData = response.data
246
      if (resData.success) {
247
        loadGroupContacts()
248
      }
249
    })
250
  }
251
 
252
  const tabsOptions = {
253
    group: (
254
      <ul>
255
        {entity?.url_get_contact_group_list && (
7324 stevensc 256
          <li>
257
            <button
258
              className="option-tab"
259
              onClick={() => showOptions(OPTIONS.LIST_CONTACTS)}
260
            >
7317 stevensc 261
              <i className="fa fa-group" />
262
              Integrantes
7324 stevensc 263
            </button>
7317 stevensc 264
          </li>
265
        )}
266
        {entity?.url_add_user_to_group && (
7324 stevensc 267
          <li>
268
            <button
269
              className="option-tab"
270
              onClick={() => showOptions(OPTIONS.ADD_CONTACTS)}
271
            >
7317 stevensc 272
              <i className="fa fa-user-plus"></i>
7324 stevensc 273
              {labels.add_contacts}
274
            </button>
7317 stevensc 275
          </li>
276
        )}
277
        {entity?.url_zoom && (
7324 stevensc 278
          <li>
279
            <button className="option-tab" onClick={displayConferenceModal}>
7317 stevensc 280
              <i className="fa fa-user-plus" />
7324 stevensc 281
              {labels.create_conference}
282
            </button>
7317 stevensc 283
          </li>
284
        )}
285
        {entity?.url_delete && (
7324 stevensc 286
          <li>
287
            <button
288
              className="option-tab"
289
              onClick={() => {
290
                handleConfirmModalShow()
291
                modalActionUrl.current = entity?.url_delete
292
              }}
293
            >
7317 stevensc 294
              <i className="fa fa-trash"></i>
7324 stevensc 295
              Eliminar grupo
296
            </button>
7317 stevensc 297
          </li>
298
        )}
299
        {!!entity?.url_leave && (
7324 stevensc 300
          <li>
301
            <button
302
              className="option-tab"
303
              onClick={() => {
304
                handleConfirmModalShow()
305
                modalActionUrl.current = entity?.url_leave
306
              }}
307
            >
7317 stevensc 308
              <i className="fa fa-user-times"></i>
7324 stevensc 309
              Dejar grupo
310
            </button>
7317 stevensc 311
          </li>
312
        )}
313
      </ul>
314
    ),
315
    addContacts: (
316
      <>
7322 stevensc 317
        {availableContactsToAdd.length ? (
7323 stevensc 318
          <ul>
319
            {availableContactsToAdd.map(({ image, name, id }) => (
320
              <li key={id}>
321
                <div className="contact-tab">
322
                  <div className="info">
323
                    <Avatar
324
                      src={image}
325
                      alt={`${name} profile image`}
326
                      sx={{
327
                        height: '36px',
328
                        width: '36px',
329
                      }}
330
                    />
331
                    <span>{name}</span>
7322 stevensc 332
                  </div>
7323 stevensc 333
                  <IconButton onClick={() => handleAddPersonToGroup(id)}>
334
                    <Add />
335
                  </IconButton>
336
                </div>
337
              </li>
338
            ))}
339
          </ul>
7322 stevensc 340
        ) : (
7320 stevensc 341
          <EmptySection message={labels.not_contacts} />
7317 stevensc 342
        )}
343
      </>
344
    ),
345
    listContacts: (
346
      <>
7322 stevensc 347
        <SearchInput onChange={(e) => setSearch(e.target.value)} />
7317 stevensc 348
        {filtredGroupList.length ? (
7322 stevensc 349
          <ul>
350
            {filtredGroupList.map(
351
              ({ image, name, url_remove_from_group, id }) => {
352
                return (
353
                  <li key={id}>
354
                    <div className="contact-tab">
355
                      <div className="info">
356
                        <Avatar
357
                          src={image}
358
                          alt={`${name} profile image`}
359
                          sx={{
360
                            height: '36px',
361
                            width: '36px',
362
                          }}
363
                        />
364
                        <span>{name}</span>
365
                      </div>
366
                      {url_remove_from_group && (
367
                        <IconButton
368
                          onClick={() =>
369
                            handleDeletePersonFromGroup(url_remove_from_group)
370
                          }
371
                        >
372
                          <i className="fa fa-user-times" />
373
                        </IconButton>
374
                      )}
375
                    </div>
376
                  </li>
377
                )
378
              }
379
            )}
380
          </ul>
7317 stevensc 381
        ) : (
7322 stevensc 382
          <EmptySection message={labels.not_contacts} />
7317 stevensc 383
        )}
384
      </>
385
    ),
386
  }
387
 
388
  // getMessageOnMaximize and subscribe to infinite Loader
389
  useEffect(async () => {
390
    const options = {
391
      root: null,
392
      rootMargin: '20px',
393
      treshold: 1.0,
394
    }
395
    const observer = new IntersectionObserver(handleObserver, options)
396
    if (!minimized) {
397
      await handleGetMessages()
398
      // loader observer
399
      if (loader.current) {
400
        observer.observe(loader.current)
401
      }
402
    }
403
    return () => {
404
      if (loader.current) {
405
        observer.unobserve(loader.current)
406
      }
407
    }
408
  }, [minimized])
409
 
410
  // LoadMore on change page
411
  useEffect(() => {
412
    let loadMore = () => handleLoadMore()
413
    loadMore()
414
    return () => {
415
      loadMore = null
416
    }
417
  }, [currentPage])
418
 
419
  // getMessagesInterval
420
  useEffect(() => {
421
    if (window.location.pathname === '/group/my-groups') {
422
      const items = document.getElementsByClassName('sc-jSgupP')
423
      if (items && items.length > 0) {
424
        items[0].style.display = 'none'
425
      }
426
    }
427
  }, [minimized])
428
 
429
  useEffect(() => {
430
    let timer
431
    if (!minimized && !loading) {
432
      timer = setTimeout(() => handleGetMessages(), 1000)
433
    }
434
    return () => {
435
      clearTimeout(timer)
436
    }
437
  }, [minimized, loading])
438
 
439
  // useEffect for tabs changing
440
  useEffect(() => {
441
    if (optionTab === 'addContacts') loadPersonsAvailable()
442
    if (optionTab === 'listContacts') loadGroupContacts()
443
  }, [optionTab])
444
 
445
  return (
446
    <>
447
      <div className="personal-chat">
448
        <div className={`chat-header ${not_seen_messages ? 'notify' : ''}`}>
7324 stevensc 449
          <Avatar
450
            src={entity?.image || '/images/users-group.png'}
451
            sx={{ width: '36px', height: '36px' }}
452
          />
7317 stevensc 453
          <div className="info-content">
454
            <a href={entity?.profile} target="_blank" rel="noreferrer">
7323 stevensc 455
              {entity?.name}
7317 stevensc 456
            </a>
457
            {entity?.type === 'user' && (
458
              <small className={entity?.online ? 'online' : 'offline'}>
459
                {entity?.online ? 'En línea' : 'Desconectado'}
460
              </small>
461
            )}
462
          </div>
463
          <div className="options ml-auto">
464
            <i
465
              className="cursor-pointer fa fa-gear"
466
              onClick={() =>
467
                showOptions(
468
                  entity?.type === 'user' ? OPTIONS.CONFERENCE : OPTIONS.GROUP
469
                )
470
              }
471
            />
472
            <i
473
              className={'cursor-pointer fa fa-minus-circle'}
474
              onClick={handleActive}
475
            />
476
            <i
477
              className="cursor-pointer fa fa-times-circle"
478
              onClick={handleCloseChat}
479
            />
480
          </div>
481
        </div>
482
 
483
        <div
484
          className="panel-body"
485
          style={{ display: !minimized ? 'block' : 'none' }}
486
        >
487
          {optionTab ? (
488
            <StyledShowOptions>
7324 stevensc 489
              <StyledComeBack onClick={() => showOptions(OPTIONS.INITIAL)}>
7323 stevensc 490
                <ArrowBackIos />
7324 stevensc 491
                {labels.back}
492
              </StyledComeBack>
7317 stevensc 493
              <div className="optionsTab">{tabsOptions[optionTab]}</div>
494
            </StyledShowOptions>
495
          ) : (
496
            <>
497
              <ChatList
498
                messages={[...oldMessages, ...messages]}
499
                currentPage={currentPage}
500
                lastPage={lastPage}
501
                conversationList={conversationListEl}
502
                loader={loader}
503
              />
504
              <TextBox
505
                uploadUrl={entity?.url_upload}
506
                isNotSeen={not_seen_messages}
507
                markRead={() => onRead(entity)}
508
                onSend={sendMessage}
509
              />
510
            </>
511
          )}
512
        </div>
513
      </div>
514
      <ConfirmModal
515
        show={confirmModalShow}
516
        onClose={handleConfirmModalShow}
517
        onAccept={handleConfirmModalAccept}
518
      />
519
      <ConferenceModal
520
        show={showConferenceModal}
521
        zoomUrl={entity?.url_zoom}
522
        onClose={() => {
523
          showOptions()
524
          displayConferenceModal()
525
        }}
526
      />
527
    </>
528
  )
529
}
530
 
531
export default React.memo(PersonalChat)