|  |  | import React, { useEffect, useState } from 'react';
 | 
						
						
						
							|  |  | import { SessionIconButton } from '../icon';
 | 
						
						
						
							|  |  | import _ from 'lodash';
 | 
						
						
						
							|  |  | // tslint:disable-next-line: no-submodule-imports
 | 
						
						
						
							|  |  | import useInterval from 'react-use/lib/useInterval';
 | 
						
						
						
							|  |  | import { useDispatch, useSelector } from 'react-redux';
 | 
						
						
						
							|  |  | import { Data } from '../../data/data';
 | 
						
						
						
							|  |  | import {
 | 
						
						
						
							|  |  |   deleteAllMessagesByConvoIdWithConfirmation,
 | 
						
						
						
							|  |  |   setDisappearingMessagesByConvoId,
 | 
						
						
						
							|  |  |   showAddModeratorsByConvoId,
 | 
						
						
						
							|  |  |   showInviteContactByConvoId,
 | 
						
						
						
							|  |  |   showLeaveGroupByConvoId,
 | 
						
						
						
							|  |  |   showRemoveModeratorsByConvoId,
 | 
						
						
						
							|  |  |   showUpdateGroupMembersByConvoId,
 | 
						
						
						
							|  |  |   showUpdateGroupNameByConvoId,
 | 
						
						
						
							|  |  | } from '../../interactions/conversationInteractions';
 | 
						
						
						
							|  |  | import { Constants } from '../../session';
 | 
						
						
						
							|  |  | import { closeRightPanel } from '../../state/ducks/conversations';
 | 
						
						
						
							|  |  | import { getSelectedConversation, isRightPanelShowing } from '../../state/selectors/conversations';
 | 
						
						
						
							|  |  | import { getTimerOptions } from '../../state/selectors/timerOptions';
 | 
						
						
						
							|  |  | import { AttachmentTypeWithPath } from '../../types/Attachment';
 | 
						
						
						
							|  |  | import { Avatar, AvatarSize } from '../avatar/Avatar';
 | 
						
						
						
							|  |  | import { SessionButton, SessionButtonColor, SessionButtonType } from '../basic/SessionButton';
 | 
						
						
						
							|  |  | import { SessionDropdown } from '../basic/SessionDropdown';
 | 
						
						
						
							|  |  | import { SpacerLG } from '../basic/Text';
 | 
						
						
						
							|  |  | import { MediaItemType } from '../lightbox/LightboxGallery';
 | 
						
						
						
							|  |  | import { MediaGallery } from './media-gallery/MediaGallery';
 | 
						
						
						
							|  |  | import { getAbsoluteAttachmentPath } from '../../types/MessageAttachment';
 | 
						
						
						
							|  |  | import styled from 'styled-components';
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  | async function getMediaGalleryProps(
 | 
						
						
						
							|  |  |   conversationId: string
 | 
						
						
						
							|  |  | ): Promise<{
 | 
						
						
						
							|  |  |   documents: Array<MediaItemType>;
 | 
						
						
						
							|  |  |   media: Array<MediaItemType>;
 | 
						
						
						
							|  |  | }> {
 | 
						
						
						
							|  |  |   // We fetch more documents than media as they don’t require to be loaded
 | 
						
						
						
							|  |  |   // into memory right away. Revisit this once we have infinite scrolling:
 | 
						
						
						
							|  |  |   const rawMedia = await Data.getMessagesWithVisualMediaAttachments(
 | 
						
						
						
							|  |  |     conversationId,
 | 
						
						
						
							|  |  |     Constants.CONVERSATION.DEFAULT_MEDIA_FETCH_COUNT
 | 
						
						
						
							|  |  |   );
 | 
						
						
						
							|  |  |   const rawDocuments = await Data.getMessagesWithFileAttachments(
 | 
						
						
						
							|  |  |     conversationId,
 | 
						
						
						
							|  |  |     Constants.CONVERSATION.DEFAULT_DOCUMENTS_FETCH_COUNT
 | 
						
						
						
							|  |  |   );
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   const media = _.flatten(
 | 
						
						
						
							|  |  |     rawMedia.map(attributes => {
 | 
						
						
						
							|  |  |       const { attachments, source, id, timestamp, serverTimestamp, received_at } = attributes;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |       return (attachments || [])
 | 
						
						
						
							|  |  |         .filter(
 | 
						
						
						
							|  |  |           (attachment: AttachmentTypeWithPath) =>
 | 
						
						
						
							|  |  |             attachment.thumbnail && !attachment.pending && !attachment.error
 | 
						
						
						
							|  |  |         )
 | 
						
						
						
							|  |  |         .map((attachment: AttachmentTypeWithPath, index: number) => {
 | 
						
						
						
							|  |  |           const { thumbnail } = attachment;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |           const mediaItem: MediaItemType = {
 | 
						
						
						
							|  |  |             objectURL: getAbsoluteAttachmentPath(attachment.path),
 | 
						
						
						
							|  |  |             thumbnailObjectUrl: thumbnail ? getAbsoluteAttachmentPath(thumbnail.path) : undefined,
 | 
						
						
						
							|  |  |             contentType: attachment.contentType || '',
 | 
						
						
						
							|  |  |             index,
 | 
						
						
						
							|  |  |             messageTimestamp: timestamp || serverTimestamp || received_at || 0,
 | 
						
						
						
							|  |  |             messageSender: source,
 | 
						
						
						
							|  |  |             messageId: id,
 | 
						
						
						
							|  |  |             attachment,
 | 
						
						
						
							|  |  |           };
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |           return mediaItem;
 | 
						
						
						
							|  |  |         });
 | 
						
						
						
							|  |  |     })
 | 
						
						
						
							|  |  |   );
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   // Unlike visual media, only one non-image attachment is supported
 | 
						
						
						
							|  |  |   const documents = rawDocuments.map(attributes => {
 | 
						
						
						
							|  |  |     // this is to not fail if the attachment is invalid (could be a Long Attachment type which is not supported)
 | 
						
						
						
							|  |  |     if (!attributes.attachments?.length) {
 | 
						
						
						
							|  |  |       // window?.log?.info(
 | 
						
						
						
							|  |  |       //   'Got a message with an empty list of attachment. Skipping...'
 | 
						
						
						
							|  |  |       // );
 | 
						
						
						
							|  |  |       return null;
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  |     const attachment = attributes.attachments[0];
 | 
						
						
						
							|  |  |     const { source, id, timestamp, serverTimestamp, received_at } = attributes;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     return {
 | 
						
						
						
							|  |  |       contentType: attachment.contentType,
 | 
						
						
						
							|  |  |       index: 0,
 | 
						
						
						
							|  |  |       attachment,
 | 
						
						
						
							|  |  |       messageTimestamp: timestamp || serverTimestamp || received_at || 0,
 | 
						
						
						
							|  |  |       messageSender: source,
 | 
						
						
						
							|  |  |       messageId: id,
 | 
						
						
						
							|  |  |     };
 | 
						
						
						
							|  |  |   });
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   return {
 | 
						
						
						
							|  |  |     media,
 | 
						
						
						
							|  |  |     documents: _.compact(documents), // remove null
 | 
						
						
						
							|  |  |   };
 | 
						
						
						
							|  |  | }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  | const HeaderItem = () => {
 | 
						
						
						
							|  |  |   const selectedConversation = useSelector(getSelectedConversation);
 | 
						
						
						
							|  |  |   const dispatch = useDispatch();
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   if (!selectedConversation) {
 | 
						
						
						
							|  |  |     return null;
 | 
						
						
						
							|  |  |   }
 | 
						
						
						
							|  |  |   const { id, isGroup, isKickedFromGroup, isBlocked, left } = selectedConversation;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   const showInviteContacts = isGroup && !isKickedFromGroup && !isBlocked && !left;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   return (
 | 
						
						
						
							|  |  |     <div className="group-settings-header">
 | 
						
						
						
							|  |  |       <SessionIconButton
 | 
						
						
						
							|  |  |         iconType="chevron"
 | 
						
						
						
							|  |  |         iconSize="medium"
 | 
						
						
						
							|  |  |         iconRotation={270}
 | 
						
						
						
							|  |  |         onClick={() => {
 | 
						
						
						
							|  |  |           dispatch(closeRightPanel());
 | 
						
						
						
							|  |  |         }}
 | 
						
						
						
							|  |  |         style={{ position: 'absolute' }}
 | 
						
						
						
							|  |  |         dataTestId="back-button-conversation-options"
 | 
						
						
						
							|  |  |       />
 | 
						
						
						
							|  |  |       <Avatar size={AvatarSize.XL} pubkey={id} />
 | 
						
						
						
							|  |  |       {showInviteContacts && (
 | 
						
						
						
							|  |  |         <SessionIconButton
 | 
						
						
						
							|  |  |           iconType="addUser"
 | 
						
						
						
							|  |  |           iconSize="medium"
 | 
						
						
						
							|  |  |           onClick={() => {
 | 
						
						
						
							|  |  |             if (selectedConversation) {
 | 
						
						
						
							|  |  |               showInviteContactByConvoId(selectedConversation.id);
 | 
						
						
						
							|  |  |             }
 | 
						
						
						
							|  |  |           }}
 | 
						
						
						
							|  |  |           dataTestId="add-user-button"
 | 
						
						
						
							|  |  |         />
 | 
						
						
						
							|  |  |       )}
 | 
						
						
						
							|  |  |     </div>
 | 
						
						
						
							|  |  |   );
 | 
						
						
						
							|  |  | };
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  | const StyledLeaveButton = styled.div`
 | 
						
						
						
							|  |  |   width: 100%;
 | 
						
						
						
							|  |  |   .session-button {
 | 
						
						
						
							|  |  |     margin-top: auto;
 | 
						
						
						
							|  |  |     width: 100%;
 | 
						
						
						
							|  |  |     min-height: calc(var(--composition-container-height) + 1px); // include border in height
 | 
						
						
						
							|  |  |     flex-shrink: 0;
 | 
						
						
						
							|  |  |     align-items: center;
 | 
						
						
						
							|  |  |     border-top: 1px solid var(--border-color);
 | 
						
						
						
							|  |  |     border-radius: 0px;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     &:not(.disabled) {
 | 
						
						
						
							|  |  |       &:hover {
 | 
						
						
						
							|  |  |         background-color: var(--button-solid-background-hover-color);
 | 
						
						
						
							|  |  |       }
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  |   }
 | 
						
						
						
							|  |  | `;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  | const StyledGroupSettingsItem = styled.div`
 | 
						
						
						
							|  |  |   display: flex;
 | 
						
						
						
							|  |  |   align-items: center;
 | 
						
						
						
							|  |  |   min-height: 3rem;
 | 
						
						
						
							|  |  |   font-size: var(--font-size-md);
 | 
						
						
						
							|  |  |   color: var(--right-panel-item-text-color);
 | 
						
						
						
							|  |  |   background-color: var(--right-panel-item-background-color);
 | 
						
						
						
							|  |  |   border-top: 1px solid var(--border-color);
 | 
						
						
						
							|  |  |   border-bottom: 1px solid var(--border-color);
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   width: -webkit-fill-available;
 | 
						
						
						
							|  |  |   padding: 0 var(--margins-md);
 | 
						
						
						
							|  |  |   transition: var(--default-duration);
 | 
						
						
						
							|  |  |   cursor: pointer;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   &:hover {
 | 
						
						
						
							|  |  |     background-color: var(--right-panel-item-background-hover-color);
 | 
						
						
						
							|  |  |   }
 | 
						
						
						
							|  |  | `;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  | // tslint:disable: cyclomatic-complexity
 | 
						
						
						
							|  |  | // tslint:disable: max-func-body-length
 | 
						
						
						
							|  |  | export const SessionRightPanelWithDetails = () => {
 | 
						
						
						
							|  |  |   const [documents, setDocuments] = useState<Array<MediaItemType>>([]);
 | 
						
						
						
							|  |  |   const [media, setMedia] = useState<Array<MediaItemType>>([]);
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   const selectedConversation = useSelector(getSelectedConversation);
 | 
						
						
						
							|  |  |   const isShowing = useSelector(isRightPanelShowing);
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   useEffect(() => {
 | 
						
						
						
							|  |  |     let isRunning = true;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     if (isShowing && selectedConversation) {
 | 
						
						
						
							|  |  |       void getMediaGalleryProps(selectedConversation.id).then(results => {
 | 
						
						
						
							|  |  |         if (isRunning) {
 | 
						
						
						
							|  |  |           if (!_.isEqual(documents, results.documents)) {
 | 
						
						
						
							|  |  |             setDocuments(results.documents);
 | 
						
						
						
							|  |  |           }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |           if (!_.isEqual(media, results.media)) {
 | 
						
						
						
							|  |  |             setMedia(results.media);
 | 
						
						
						
							|  |  |           }
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |       });
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     return () => {
 | 
						
						
						
							|  |  |       isRunning = false;
 | 
						
						
						
							|  |  |       return;
 | 
						
						
						
							|  |  |     };
 | 
						
						
						
							|  |  |   }, [isShowing, selectedConversation?.id]);
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   useInterval(async () => {
 | 
						
						
						
							|  |  |     if (isShowing && selectedConversation) {
 | 
						
						
						
							|  |  |       const results = await getMediaGalleryProps(selectedConversation.id);
 | 
						
						
						
							|  |  |       if (results.documents.length !== documents.length || results.media.length !== media.length) {
 | 
						
						
						
							|  |  |         setDocuments(results.documents);
 | 
						
						
						
							|  |  |         setMedia(results.media);
 | 
						
						
						
							|  |  |       }
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  |   }, 10000);
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   if (!selectedConversation) {
 | 
						
						
						
							|  |  |     return null;
 | 
						
						
						
							|  |  |   }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   const {
 | 
						
						
						
							|  |  |     id,
 | 
						
						
						
							|  |  |     subscriberCount,
 | 
						
						
						
							|  |  |     displayNameInProfile,
 | 
						
						
						
							|  |  |     isKickedFromGroup,
 | 
						
						
						
							|  |  |     left,
 | 
						
						
						
							|  |  |     isPublic,
 | 
						
						
						
							|  |  |     weAreAdmin,
 | 
						
						
						
							|  |  |     isBlocked,
 | 
						
						
						
							|  |  |     isGroup,
 | 
						
						
						
							|  |  |     activeAt,
 | 
						
						
						
							|  |  |   } = selectedConversation;
 | 
						
						
						
							|  |  |   const showMemberCount = !!(subscriberCount && subscriberCount > 0);
 | 
						
						
						
							|  |  |   const commonNoShow = isKickedFromGroup || left || isBlocked || !activeAt;
 | 
						
						
						
							|  |  |   const hasDisappearingMessages = !isPublic && !commonNoShow;
 | 
						
						
						
							|  |  |   const leaveGroupString = isPublic
 | 
						
						
						
							|  |  |     ? window.i18n('leaveGroup')
 | 
						
						
						
							|  |  |     : isKickedFromGroup
 | 
						
						
						
							|  |  |     ? window.i18n('youGotKickedFromGroup')
 | 
						
						
						
							|  |  |     : left
 | 
						
						
						
							|  |  |     ? window.i18n('youLeftTheGroup')
 | 
						
						
						
							|  |  |     : window.i18n('leaveGroup');
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   const timerOptions = useSelector(getTimerOptions).timerOptions;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   const disappearingMessagesOptions = timerOptions.map(option => {
 | 
						
						
						
							|  |  |     return {
 | 
						
						
						
							|  |  |       content: option.name,
 | 
						
						
						
							|  |  |       onClick: () => {
 | 
						
						
						
							|  |  |         void setDisappearingMessagesByConvoId(id, option.value);
 | 
						
						
						
							|  |  |       },
 | 
						
						
						
							|  |  |     };
 | 
						
						
						
							|  |  |   });
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   const showUpdateGroupNameButton =
 | 
						
						
						
							|  |  |     isGroup && (!isPublic || (isPublic && weAreAdmin)) && !commonNoShow;
 | 
						
						
						
							|  |  |   const showAddRemoveModeratorsButton = weAreAdmin && !commonNoShow && isPublic;
 | 
						
						
						
							|  |  |   const showUpdateGroupMembersButton = !isPublic && isGroup && !commonNoShow;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |   const deleteConvoAction = isPublic
 | 
						
						
						
							|  |  |     ? () => {
 | 
						
						
						
							|  |  |         deleteAllMessagesByConvoIdWithConfirmation(id);
 | 
						
						
						
							|  |  |       }
 | 
						
						
						
							|  |  |     : () => {
 | 
						
						
						
							|  |  |         showLeaveGroupByConvoId(id);
 | 
						
						
						
							|  |  |       };
 | 
						
						
						
							|  |  |   return (
 | 
						
						
						
							|  |  |     <div className="group-settings">
 | 
						
						
						
							|  |  |       <HeaderItem />
 | 
						
						
						
							|  |  |       <h2 data-testid="right-panel-group-name">{displayNameInProfile}</h2>
 | 
						
						
						
							|  |  |       {showMemberCount && (
 | 
						
						
						
							|  |  |         <>
 | 
						
						
						
							|  |  |           <SpacerLG />
 | 
						
						
						
							|  |  |           <div role="button" className="subtle">
 | 
						
						
						
							|  |  |             {window.i18n('members', [`${subscriberCount}`])}
 | 
						
						
						
							|  |  |           </div>
 | 
						
						
						
							|  |  |           <SpacerLG />
 | 
						
						
						
							|  |  |         </>
 | 
						
						
						
							|  |  |       )}
 | 
						
						
						
							|  |  |       {showUpdateGroupNameButton && (
 | 
						
						
						
							|  |  |         <StyledGroupSettingsItem
 | 
						
						
						
							|  |  |           className="group-settings-item"
 | 
						
						
						
							|  |  |           role="button"
 | 
						
						
						
							|  |  |           onClick={async () => {
 | 
						
						
						
							|  |  |             await showUpdateGroupNameByConvoId(id);
 | 
						
						
						
							|  |  |           }}
 | 
						
						
						
							|  |  |         >
 | 
						
						
						
							|  |  |           {isPublic ? window.i18n('editGroup') : window.i18n('editGroupName')}
 | 
						
						
						
							|  |  |         </StyledGroupSettingsItem>
 | 
						
						
						
							|  |  |       )}
 | 
						
						
						
							|  |  |       {showAddRemoveModeratorsButton && (
 | 
						
						
						
							|  |  |         <>
 | 
						
						
						
							|  |  |           <StyledGroupSettingsItem
 | 
						
						
						
							|  |  |             className="group-settings-item"
 | 
						
						
						
							|  |  |             role="button"
 | 
						
						
						
							|  |  |             onClick={() => {
 | 
						
						
						
							|  |  |               showAddModeratorsByConvoId(id);
 | 
						
						
						
							|  |  |             }}
 | 
						
						
						
							|  |  |           >
 | 
						
						
						
							|  |  |             {window.i18n('addModerators')}
 | 
						
						
						
							|  |  |           </StyledGroupSettingsItem>
 | 
						
						
						
							|  |  |           <StyledGroupSettingsItem
 | 
						
						
						
							|  |  |             className="group-settings-item"
 | 
						
						
						
							|  |  |             role="button"
 | 
						
						
						
							|  |  |             onClick={() => {
 | 
						
						
						
							|  |  |               showRemoveModeratorsByConvoId(id);
 | 
						
						
						
							|  |  |             }}
 | 
						
						
						
							|  |  |           >
 | 
						
						
						
							|  |  |             {window.i18n('removeModerators')}
 | 
						
						
						
							|  |  |           </StyledGroupSettingsItem>
 | 
						
						
						
							|  |  |         </>
 | 
						
						
						
							|  |  |       )}
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |       {showUpdateGroupMembersButton && (
 | 
						
						
						
							|  |  |         <StyledGroupSettingsItem
 | 
						
						
						
							|  |  |           className="group-settings-item"
 | 
						
						
						
							|  |  |           role="button"
 | 
						
						
						
							|  |  |           onClick={async () => {
 | 
						
						
						
							|  |  |             await showUpdateGroupMembersByConvoId(id);
 | 
						
						
						
							|  |  |           }}
 | 
						
						
						
							|  |  |         >
 | 
						
						
						
							|  |  |           {window.i18n('groupMembers')}
 | 
						
						
						
							|  |  |         </StyledGroupSettingsItem>
 | 
						
						
						
							|  |  |       )}
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |       {hasDisappearingMessages && (
 | 
						
						
						
							|  |  |         <SessionDropdown
 | 
						
						
						
							|  |  |           label={window.i18n('disappearingMessages')}
 | 
						
						
						
							|  |  |           options={disappearingMessagesOptions}
 | 
						
						
						
							|  |  |         />
 | 
						
						
						
							|  |  |       )}
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |       <MediaGallery documents={documents} media={media} />
 | 
						
						
						
							|  |  |       {isGroup && (
 | 
						
						
						
							|  |  |         // tslint:disable-next-line: use-simple-attributes
 | 
						
						
						
							|  |  |         <StyledLeaveButton>
 | 
						
						
						
							|  |  |           <SessionButton
 | 
						
						
						
							|  |  |             text={leaveGroupString}
 | 
						
						
						
							|  |  |             buttonColor={SessionButtonColor.Danger}
 | 
						
						
						
							|  |  |             buttonType={SessionButtonType.Simple}
 | 
						
						
						
							|  |  |             disabled={isKickedFromGroup || left}
 | 
						
						
						
							|  |  |             onClick={deleteConvoAction}
 | 
						
						
						
							|  |  |           />
 | 
						
						
						
							|  |  |         </StyledLeaveButton>
 | 
						
						
						
							|  |  |       )}
 | 
						
						
						
							|  |  |     </div>
 | 
						
						
						
							|  |  |   );
 | 
						
						
						
							|  |  | };
 |