|
|
import { compact, flatten, isEqual } from 'lodash';
|
|
|
import React, { useEffect, useState } from 'react';
|
|
|
|
|
|
// tslint:disable-next-line: no-submodule-imports
|
|
|
import { useDispatch } from 'react-redux';
|
|
|
import useInterval from 'react-use/lib/useInterval';
|
|
|
import styled from 'styled-components';
|
|
|
import { Data } from '../../../../data/data';
|
|
|
import { SessionIconButton } from '../../../icon';
|
|
|
|
|
|
import { useIsRightPanelShowing } from '../../../../hooks/useUI';
|
|
|
import {
|
|
|
deleteAllMessagesByConvoIdWithConfirmation,
|
|
|
showAddModeratorsByConvoId,
|
|
|
showInviteContactByConvoId,
|
|
|
showLeaveGroupByConvoId,
|
|
|
showRemoveModeratorsByConvoId,
|
|
|
showUpdateGroupMembersByConvoId,
|
|
|
showUpdateGroupNameByConvoId,
|
|
|
} from '../../../../interactions/conversationInteractions';
|
|
|
import { Constants } from '../../../../session';
|
|
|
import { closeRightPanel } from '../../../../state/ducks/conversations';
|
|
|
import { setRightOverlayMode } from '../../../../state/ducks/section';
|
|
|
import {
|
|
|
useSelectedConversationKey,
|
|
|
useSelectedDisplayNameInProfile,
|
|
|
useSelectedIsActive,
|
|
|
useSelectedIsBlocked,
|
|
|
useSelectedIsGroup,
|
|
|
useSelectedIsKickedFromGroup,
|
|
|
useSelectedIsLeft,
|
|
|
useSelectedIsPublic,
|
|
|
useSelectedSubscriberCount,
|
|
|
useSelectedWeAreAdmin,
|
|
|
} from '../../../../state/selectors/selectedConversation';
|
|
|
import { AttachmentTypeWithPath } from '../../../../types/Attachment';
|
|
|
import { getAbsoluteAttachmentPath } from '../../../../types/MessageAttachment';
|
|
|
import { Avatar, AvatarSize } from '../../../avatar/Avatar';
|
|
|
import { SessionButton, SessionButtonColor, SessionButtonType } from '../../../basic/SessionButton';
|
|
|
import { SpacerLG } from '../../../basic/Text';
|
|
|
import { PanelButtonGroup, PanelIconButton } from '../../../buttons';
|
|
|
import { MediaItemType } from '../../../lightbox/LightboxGallery';
|
|
|
import { MediaGallery } from '../../media-gallery/MediaGallery';
|
|
|
|
|
|
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 selectedConvoKey = useSelectedConversationKey();
|
|
|
const dispatch = useDispatch();
|
|
|
const isBlocked = useSelectedIsBlocked();
|
|
|
const isKickedFromGroup = useSelectedIsKickedFromGroup();
|
|
|
const left = useSelectedIsLeft();
|
|
|
const isGroup = useSelectedIsGroup();
|
|
|
|
|
|
if (!selectedConvoKey) {
|
|
|
return null;
|
|
|
}
|
|
|
|
|
|
const showInviteContacts = isGroup && !isKickedFromGroup && !isBlocked && !left;
|
|
|
|
|
|
return (
|
|
|
<div className="right-panel-header">
|
|
|
<SessionIconButton
|
|
|
iconType="chevron"
|
|
|
iconSize="medium"
|
|
|
iconRotation={270}
|
|
|
onClick={() => {
|
|
|
dispatch(closeRightPanel());
|
|
|
}}
|
|
|
style={{ position: 'absolute' }}
|
|
|
dataTestId="back-button-conversation-options"
|
|
|
/>
|
|
|
<Avatar size={AvatarSize.XL} pubkey={selectedConvoKey} />
|
|
|
{showInviteContacts && (
|
|
|
<SessionIconButton
|
|
|
iconType="addUser"
|
|
|
iconSize="medium"
|
|
|
onClick={() => {
|
|
|
if (selectedConvoKey) {
|
|
|
showInviteContactByConvoId(selectedConvoKey);
|
|
|
}
|
|
|
}}
|
|
|
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);
|
|
|
}
|
|
|
`;
|
|
|
|
|
|
const StyledName = styled.h4`
|
|
|
padding-inline: var(--margins-md);
|
|
|
font-size: var(--font-size-md);
|
|
|
`;
|
|
|
|
|
|
// tslint:disable: cyclomatic-complexity
|
|
|
// tslint:disable: max-func-body-length
|
|
|
export const OverlayRightPanelSettings = () => {
|
|
|
const [documents, setDocuments] = useState<Array<MediaItemType>>([]);
|
|
|
const [media, setMedia] = useState<Array<MediaItemType>>([]);
|
|
|
|
|
|
const selectedConvoKey = useSelectedConversationKey();
|
|
|
const dispatch = useDispatch();
|
|
|
const isShowing = useIsRightPanelShowing();
|
|
|
const subscriberCount = useSelectedSubscriberCount();
|
|
|
|
|
|
const isActive = useSelectedIsActive();
|
|
|
const displayNameInProfile = useSelectedDisplayNameInProfile();
|
|
|
const isBlocked = useSelectedIsBlocked();
|
|
|
const isKickedFromGroup = useSelectedIsKickedFromGroup();
|
|
|
const left = useSelectedIsLeft();
|
|
|
const isGroup = useSelectedIsGroup();
|
|
|
const isPublic = useSelectedIsPublic();
|
|
|
const weAreAdmin = useSelectedWeAreAdmin();
|
|
|
|
|
|
useEffect(() => {
|
|
|
let isCancelled = false;
|
|
|
|
|
|
const loadDocumentsOrMedia = async () => {
|
|
|
try {
|
|
|
if (isShowing && selectedConvoKey) {
|
|
|
const results = await getMediaGalleryProps(selectedConvoKey);
|
|
|
|
|
|
if (!isCancelled) {
|
|
|
if (!isEqual(documents, results.documents)) {
|
|
|
setDocuments(results.documents);
|
|
|
}
|
|
|
|
|
|
if (!isEqual(media, results.media)) {
|
|
|
setMedia(results.media);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
} catch (error) {
|
|
|
if (!isCancelled) {
|
|
|
window.log.debug(`OverlayRightPanelSettings loadDocumentsOrMedia: ${error}`);
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
void loadDocumentsOrMedia();
|
|
|
|
|
|
return () => {
|
|
|
isCancelled = true;
|
|
|
};
|
|
|
}, [documents, isShowing, media, selectedConvoKey]);
|
|
|
|
|
|
useInterval(async () => {
|
|
|
if (isShowing && selectedConvoKey) {
|
|
|
const results = await getMediaGalleryProps(selectedConvoKey);
|
|
|
if (results.documents.length !== documents.length || results.media.length !== media.length) {
|
|
|
setDocuments(results.documents);
|
|
|
setMedia(results.media);
|
|
|
}
|
|
|
}
|
|
|
}, 10000);
|
|
|
|
|
|
if (!selectedConvoKey) {
|
|
|
return null;
|
|
|
}
|
|
|
|
|
|
const showMemberCount = !!(subscriberCount && subscriberCount > 0);
|
|
|
const commonNoShow = isKickedFromGroup || left || isBlocked || !isActive;
|
|
|
const hasDisappearingMessages = !isPublic && !commonNoShow;
|
|
|
const leaveGroupString = isPublic
|
|
|
? window.i18n('leaveGroup')
|
|
|
: isKickedFromGroup
|
|
|
? window.i18n('youGotKickedFromGroup')
|
|
|
: left
|
|
|
? window.i18n('youLeftTheGroup')
|
|
|
: window.i18n('leaveGroup');
|
|
|
|
|
|
const showUpdateGroupNameButton =
|
|
|
isGroup && (!isPublic || (isPublic && weAreAdmin)) && !commonNoShow;
|
|
|
const showAddRemoveModeratorsButton = weAreAdmin && !commonNoShow && isPublic;
|
|
|
const showUpdateGroupMembersButton = !isPublic && isGroup && !commonNoShow;
|
|
|
|
|
|
const deleteConvoAction = isPublic
|
|
|
? () => {
|
|
|
deleteAllMessagesByConvoIdWithConfirmation(selectedConvoKey); // TODOLATER this does not delete the public group and showLeaveGroupByConvoId is not only working for closed groups
|
|
|
}
|
|
|
: () => {
|
|
|
showLeaveGroupByConvoId(selectedConvoKey);
|
|
|
};
|
|
|
|
|
|
return (
|
|
|
<>
|
|
|
<HeaderItem />
|
|
|
<StyledName data-testid="right-panel-group-name">{displayNameInProfile}</StyledName>
|
|
|
{showMemberCount && (
|
|
|
<>
|
|
|
<SpacerLG />
|
|
|
<div role="button" className="subtle">
|
|
|
{window.i18n('members', [`${subscriberCount}`])}
|
|
|
</div>
|
|
|
<SpacerLG />
|
|
|
</>
|
|
|
)}
|
|
|
{showUpdateGroupNameButton && (
|
|
|
<StyledGroupSettingsItem
|
|
|
className="right-panel-item"
|
|
|
role="button"
|
|
|
onClick={() => {
|
|
|
void showUpdateGroupNameByConvoId(selectedConvoKey);
|
|
|
}}
|
|
|
>
|
|
|
{isPublic ? window.i18n('editGroup') : window.i18n('editGroupName')}
|
|
|
</StyledGroupSettingsItem>
|
|
|
)}
|
|
|
{showAddRemoveModeratorsButton && (
|
|
|
<>
|
|
|
<StyledGroupSettingsItem
|
|
|
className="right-panel-item"
|
|
|
role="button"
|
|
|
onClick={() => {
|
|
|
showAddModeratorsByConvoId(selectedConvoKey);
|
|
|
}}
|
|
|
>
|
|
|
{window.i18n('addModerators')}
|
|
|
</StyledGroupSettingsItem>
|
|
|
<StyledGroupSettingsItem
|
|
|
className="right-panel-item"
|
|
|
role="button"
|
|
|
onClick={() => {
|
|
|
showRemoveModeratorsByConvoId(selectedConvoKey);
|
|
|
}}
|
|
|
>
|
|
|
{window.i18n('removeModerators')}
|
|
|
</StyledGroupSettingsItem>
|
|
|
</>
|
|
|
)}
|
|
|
|
|
|
{showUpdateGroupMembersButton && (
|
|
|
<StyledGroupSettingsItem
|
|
|
className="right-panel-item"
|
|
|
role="button"
|
|
|
onClick={() => {
|
|
|
void showUpdateGroupMembersByConvoId(selectedConvoKey);
|
|
|
}}
|
|
|
>
|
|
|
{window.i18n('groupMembers')}
|
|
|
</StyledGroupSettingsItem>
|
|
|
)}
|
|
|
|
|
|
{hasDisappearingMessages && (
|
|
|
/* TODO Move ButtonGroup around all settings items */
|
|
|
<PanelButtonGroup>
|
|
|
<PanelIconButton
|
|
|
iconType={'timer50'}
|
|
|
text={window.i18n('disappearingMessages')}
|
|
|
onClick={() => {
|
|
|
dispatch(setRightOverlayMode('disappearing-messages'));
|
|
|
}}
|
|
|
/>
|
|
|
</PanelButtonGroup>
|
|
|
)}
|
|
|
|
|
|
<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>
|
|
|
)}
|
|
|
</>
|
|
|
);
|
|
|
};
|