You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-desktop/ts/components/conversation/Message.tsx

878 lines
25 KiB
TypeScript

import React from 'react';
import classNames from 'classnames';
import { Avatar, AvatarSize } from '../Avatar';
import { Spinner } from '../basic/Spinner';
import { MessageBody } from './MessageBody';
import { ImageGrid } from './ImageGrid';
import { Image } from './Image';
import { ContactName } from './ContactName';
import { Quote } from './Quote';
import {
AttachmentType,
AttachmentTypeWithPath,
canDisplayImage,
getExtensionForDisplay,
getGridDimensions,
getImageDimensions,
hasImage,
hasVideoScreenshot,
isAudio,
isImage,
isImageAttachment,
isVideo,
} from '../../../ts/types/Attachment';
import { getIncrement } from '../../util/timer';
import { isFileDangerous } from '../../util/isFileDangerous';
import _ from 'lodash';
import { contextMenu } from 'react-contexify';
import uuid from 'uuid';
import { PubKey } from '../../session/types';
import { MessageRenderingProps } from '../../models/messageType';
import { updateUserDetailsModal } from '../../state/ducks/modalDialog';
import autoBind from 'auto-bind';
import { AudioPlayerWithEncryptedFile } from './H5AudioPlayer';
import { ClickToTrustSender } from './message/ClickToTrustSender';
import { getMessageById } from '../../data/data';
import { connect } from 'react-redux';
import { StateType } from '../../state/reducer';
import {
getQuotedMessageToAnimate,
getSelectedMessageIds,
} from '../../state/selectors/conversations';
import {
messageExpired,
showLightBox,
toggleSelectedMessageId,
} from '../../state/ducks/conversations';
import { saveAttachmentToDisk } from '../../util/attachmentsUtil';
import { LightBoxOptions } from '../session/conversation/SessionConversation';
import { MessageContextMenu } from './MessageContextMenu';
import { ReadableMessage } from './ReadableMessage';
import { getConversationController } from '../../session/conversations';
import { MessageMetadata } from './message/MessageMetadata';
// Same as MIN_WIDTH in ImageGrid.tsx
const MINIMUM_LINK_PREVIEW_IMAGE_WIDTH = 200;
interface State {
expiring: boolean;
expired: boolean;
imageBroken: boolean;
}
const EXPIRATION_CHECK_MINIMUM = 2000;
const EXPIRED_DELAY = 600;
type Props = MessageRenderingProps & {
selectedMessages: Array<string>;
quotedMessageToAnimate: string | undefined;
};
function attachmentIsAttachmentTypeWithPath(attac: any): attac is AttachmentTypeWithPath {
return attac.path !== undefined;
}
const onClickAttachment = async (onClickProps: {
attachment: AttachmentTypeWithPath | AttachmentType;
messageId: string;
}) => {
let index = -1;
const found = await getMessageById(onClickProps.messageId);
if (!found) {
window.log.warn('Such message not found');
return;
}
const msgAttachments = found.getPropsForMessage().attachments;
const media = (msgAttachments || []).map(attachmentForMedia => {
index++;
const messageTimestamp =
found.get('timestamp') || found.get('serverTimestamp') || found.get('received_at');
return {
index: _.clone(index),
objectURL: attachmentForMedia.url || undefined,
contentType: attachmentForMedia.contentType,
attachment: attachmentForMedia,
messageSender: found.getSource(),
messageTimestamp,
messageId: onClickProps.messageId,
};
});
if (attachmentIsAttachmentTypeWithPath(onClickProps.attachment)) {
const lightBoxOptions: LightBoxOptions = {
media: media as any,
attachment: onClickProps.attachment,
};
window.inboxStore?.dispatch(showLightBox(lightBoxOptions));
} else {
window.log.warn('Attachment is not of the right type');
}
};
class MessageInner extends React.PureComponent<Props, State> {
public expirationCheckInterval: any;
public expiredTimeout: any;
public ctxMenuID: string;
public constructor(props: Props) {
super(props);
autoBind(this);
this.state = {
expiring: false,
expired: false,
imageBroken: false,
};
this.ctxMenuID = `ctx-menu-message-${uuid()}`;
}
public componentDidMount() {
const { expirationLength } = this.props;
if (!expirationLength) {
return;
}
const increment = getIncrement(expirationLength);
const checkFrequency = Math.max(EXPIRATION_CHECK_MINIMUM, increment);
this.checkExpired();
this.expirationCheckInterval = setInterval(() => {
this.checkExpired();
}, checkFrequency);
}
public componentWillUnmount() {
if (this.expirationCheckInterval) {
clearInterval(this.expirationCheckInterval);
}
if (this.expiredTimeout) {
global.clearTimeout(this.expiredTimeout);
}
}
public componentDidUpdate() {
this.checkExpired();
}
public checkExpired() {
const now = Date.now();
const { isExpired, expirationTimestamp, expirationLength, convoId, id } = this.props;
if (!expirationTimestamp || !expirationLength) {
return;
}
if (this.expiredTimeout) {
return;
}
if (isExpired || now >= expirationTimestamp) {
this.setState({
expiring: true,
});
const setExpired = async () => {
this.setState({
expired: true,
});
await window.Signal.Data.removeMessage(id);
window.inboxStore?.dispatch(
messageExpired({
conversationKey: convoId,
messageId: id,
})
);
const convo = getConversationController().get(convoId);
convo?.updateLastMessage();
};
// as 'checkExpired' is potentially called more than once (componentDidUpdate & componentDidMount),
// we need to clear the timeout call to 'setExpired' first to avoid multiple calls to 'setExpired'.
global.clearTimeout(this.expiredTimeout);
this.expiredTimeout = setTimeout(setExpired, EXPIRED_DELAY);
}
}
public handleImageError() {
this.setState({
imageBroken: true,
});
}
// tslint:disable-next-line max-func-body-length cyclomatic-complexity
public renderAttachment() {
const {
id,
attachments,
text,
collapseMetadata,
conversationType,
direction,
quote,
isTrustedForAttachmentDownload,
} = this.props;
const { imageBroken } = this.state;
if (!attachments || !attachments[0]) {
return null;
}
const firstAttachment = attachments[0];
// For attachments which aren't full-frame
const withContentBelow = Boolean(text);
const withContentAbove =
Boolean(quote) || (conversationType === 'group' && direction === 'incoming');
const displayImage = canDisplayImage(attachments);
if (!isTrustedForAttachmentDownload) {
return <ClickToTrustSender messageId={id} />;
}
if (
displayImage &&
!imageBroken &&
((isImage(attachments) && hasImage(attachments)) ||
(isVideo(attachments) && hasVideoScreenshot(attachments)))
) {
return (
<div
className={classNames(
'module-message__attachment-container',
withContentAbove ? 'module-message__attachment-container--with-content-above' : null,
withContentBelow ? 'module-message__attachment-container--with-content-below' : null
)}
>
<ImageGrid
attachments={attachments}
withContentAbove={withContentAbove}
withContentBelow={withContentBelow}
bottomOverlay={!collapseMetadata}
onError={this.handleImageError}
onClickAttachment={this.onClickOnImageGrid}
/>
</div>
);
} else if (!firstAttachment.pending && isAudio(attachments)) {
return (
<div
role="main"
onClick={(e: any) => {
e.stopPropagation();
}}
>
<AudioPlayerWithEncryptedFile
src={firstAttachment.url}
contentType={firstAttachment.contentType}
messageId={this.props.id}
/>
</div>
);
} else {
const { pending, fileName, fileSize, contentType } = firstAttachment;
const extension = getExtensionForDisplay({ contentType, fileName });
const isDangerous = isFileDangerous(fileName || '');
return (
<div
className={classNames(
'module-message__generic-attachment',
withContentBelow ? 'module-message__generic-attachment--with-content-below' : null,
withContentAbove ? 'module-message__generic-attachment--with-content-above' : null
)}
>
{pending ? (
<div className="module-message__generic-attachment__spinner-container">
<Spinner size="small" direction={direction} />
</div>
) : (
<div className="module-message__generic-attachment__icon-container">
<div
role="button"
className="module-message__generic-attachment__icon"
onClick={this.onClickOnGenericAttachment}
>
{extension ? (
<div className="module-message__generic-attachment__icon__extension">
{extension}
</div>
) : null}
</div>
{isDangerous ? (
<div className="module-message__generic-attachment__icon-dangerous-container">
<div className="module-message__generic-attachment__icon-dangerous" />
</div>
) : null}
</div>
)}
<div className="module-message__generic-attachment__text">
<div
className={classNames(
'module-message__generic-attachment__file-name',
`module-message__generic-attachment__file-name--${direction}`
)}
>
{fileName}
</div>
<div
className={classNames(
'module-message__generic-attachment__file-size',
`module-message__generic-attachment__file-size--${direction}`
)}
>
{fileSize}
</div>
</div>
</div>
);
}
}
// tslint:disable-next-line cyclomatic-complexity
public renderPreview() {
const { attachments, conversationType, direction, previews, quote } = this.props;
// Attachments take precedence over Link Previews
if (attachments && attachments.length) {
return null;
}
if (!previews || previews.length < 1) {
return null;
}
const first = previews[0];
if (!first) {
return null;
}
const withContentAbove =
Boolean(quote) || (conversationType === 'group' && direction === 'incoming');
const previewHasImage = first.image && isImageAttachment(first.image);
const width = first.image && first.image.width;
const isFullSizeImage = width && width >= MINIMUM_LINK_PREVIEW_IMAGE_WIDTH;
return (
<div
role="button"
className={classNames(
'module-message__link-preview',
withContentAbove ? 'module-message__link-preview--with-content-above' : null
)}
>
{first.image && previewHasImage && isFullSizeImage ? (
<ImageGrid
attachments={[first.image]}
withContentAbove={withContentAbove}
withContentBelow={true}
onError={this.handleImageError}
/>
) : null}
<div
className={classNames(
'module-message__link-preview__content',
withContentAbove || isFullSizeImage
? 'module-message__link-preview__content--with-content-above'
: null
)}
>
{first.image && previewHasImage && !isFullSizeImage ? (
<div className="module-message__link-preview__icon_container">
<Image
smallCurveTopLeft={!withContentAbove}
softCorners={true}
alt={window.i18n('previewThumbnail', [first.domain])}
height={72}
width={72}
url={first.image.url}
attachment={first.image}
onError={this.handleImageError}
/>
</div>
) : null}
<div
className={classNames(
'module-message__link-preview__text',
previewHasImage && !isFullSizeImage
? 'module-message__link-preview__text--with-icon'
: null
)}
>
<div className="module-message__link-preview__title">{first.title}</div>
<div className="module-message__link-preview__location">{first.domain}</div>
</div>
</div>
</div>
);
}
public renderQuote() {
const { conversationType, direction, quote, isPublic, convoId } = this.props;
if (!quote || !quote.authorPhoneNumber || !quote.messageId) {
return null;
}
const withContentAbove = conversationType === 'group' && direction === 'incoming';
const shortenedPubkey = PubKey.shorten(quote.authorPhoneNumber);
const displayedPubkey = quote.authorProfileName ? shortenedPubkey : quote.authorPhoneNumber;
return (
<Quote
onClick={this.onQuoteClick}
text={quote.text}
attachment={quote.attachment}
isIncoming={direction === 'incoming'}
conversationType={conversationType}
convoId={convoId}
isPublic={isPublic}
authorPhoneNumber={displayedPubkey}
authorProfileName={quote.authorProfileName}
authorName={quote.authorName}
referencedMessageNotFound={quote.referencedMessageNotFound}
isFromMe={quote.isFromMe}
withContentAbove={withContentAbove}
/>
);
}
public renderAvatar() {
const {
authorAvatarPath,
authorName,
authorPhoneNumber,
authorProfileName,
collapseMetadata,
isSenderAdmin,
conversationType,
direction,
isPublic,
firstMessageOfSeries,
} = this.props;
if (collapseMetadata || conversationType !== 'group' || direction === 'outgoing') {
return;
}
const userName = authorName || authorProfileName || authorPhoneNumber;
if (!firstMessageOfSeries) {
return <div style={{ marginInlineEnd: '60px' }} key={`msg-avatar-${authorPhoneNumber}`} />;
}
return (
<div className="module-message__author-avatar" key={`msg-avatar-${authorPhoneNumber}`}>
<Avatar
avatarPath={authorAvatarPath}
name={userName}
size={AvatarSize.S}
onAvatarClick={this.onMessageAvatarClick}
pubkey={authorPhoneNumber}
/>
{isPublic && isSenderAdmin && (
<div className="module-avatar__icon--crown-wrapper">
<div className="module-avatar__icon--crown" />
</div>
)}
</div>
);
}
public renderText() {
const { text, direction, status, conversationType, convoId, multiSelectMode } = this.props;
const contents =
direction === 'incoming' && status === 'error' ? window.i18n('incomingError') : text;
if (!contents) {
return null;
}
return (
<div
dir="auto"
className={classNames(
'module-message__text',
`module-message__text--${direction}`,
status === 'error' && direction === 'incoming' ? 'module-message__text--error' : null
)}
>
<MessageBody
text={contents || ''}
isGroup={conversationType === 'group'}
convoId={convoId}
disableLinks={multiSelectMode}
/>
</div>
);
}
public renderError(isCorrectSide: boolean) {
const { status, direction } = this.props;
if (!isCorrectSide || status !== 'error') {
return null;
}
return (
<div className="module-message__error-container">
<div
className={classNames('module-message__error', `module-message__error--${direction}`)}
/>
</div>
);
}
public getWidth(): number | undefined {
const { attachments, previews } = this.props;
if (attachments && attachments.length) {
const dimensions = getGridDimensions(attachments);
if (dimensions) {
return dimensions.width;
}
}
if (previews && previews.length) {
const first = previews[0];
if (!first || !first.image) {
return;
}
const { width } = first.image;
if (isImageAttachment(first.image) && width && width >= MINIMUM_LINK_PREVIEW_IMAGE_WIDTH) {
const dimensions = getImageDimensions(first.image);
if (dimensions) {
return dimensions.width;
}
}
}
return;
}
public isShowingImage(): boolean {
const { attachments, previews, text } = this.props;
const { imageBroken } = this.state;
if (imageBroken) {
return false;
}
if (attachments && attachments.length) {
const displayImage = canDisplayImage(attachments);
const hasText = text?.length;
return Boolean(
displayImage &&
!hasText &&
((isImage(attachments) && hasImage(attachments)) ||
(isVideo(attachments) && hasVideoScreenshot(attachments)))
);
}
if (previews && previews.length) {
const first = previews[0];
const { image } = first;
if (!image) {
return false;
}
return isImageAttachment(image);
}
return false;
}
// tslint:disable-next-line: cyclomatic-complexity
// tslint:disable-next-line: max-func-body-length
public render() {
const {
direction,
id: messageId,
conversationType,
selectedMessages,
receivedAt,
isUnread,
} = this.props;
const { expired, expiring } = this.state;
if (expired) {
return null;
}
const selected = selectedMessages.includes(messageId) || false;
const width = this.getWidth();
const isShowingImage = this.isShowingImage();
const divClasses = ['session-message-wrapper'];
if (selected) {
divClasses.push('message-selected');
}
if (conversationType === 'group') {
divClasses.push('public-chat-message-wrapper');
}
if (this.props.quotedMessageToAnimate === messageId) {
divClasses.push('flash-green-once');
}
const isIncoming = direction === 'incoming';
return (
<ReadableMessage
messageId={messageId}
className={classNames(divClasses)}
onContextMenu={this.handleContextMenu}
receivedAt={receivedAt}
isUnread={isUnread}
key={`readable-message-${messageId}`}
>
{this.renderAvatar()}
<div
className={classNames(
'module-message',
`module-message--${direction}`,
expiring ? 'module-message--expired' : null
)}
role="button"
onClick={this.onClickOnMessageOuterContainer}
>
{this.renderError(isIncoming)}
<div
className={classNames(
'module-message__container',
`module-message__container--${direction}`,
isShowingImage
? `module-message__container--${direction}--transparent`
: `module-message__container--${direction}--opaque`
)}
style={{
width: isShowingImage ? width : undefined,
}}
role="button"
onClick={this.onClickOnMessageInnerContainer}
>
{this.renderAuthor()}
{this.renderQuote()}
{this.renderAttachment()}
{this.renderPreview()}
{this.renderText()}
<MessageMetadata
direction={this.props.direction}
messageId={this.props.id}
timestamp={this.props.timestamp}
collapseMetadata={this.props.collapseMetadata}
expirationLength={this.props.expirationLength}
isAdmin={this.props.isSenderAdmin}
serverTimestamp={this.props.serverTimestamp}
isPublic={this.props.isPublic}
status={this.props.status}
expirationTimestamp={this.props.expirationTimestamp}
text={this.props.text}
isShowingImage={this.isShowingImage()}
/>
</div>
{this.renderError(!isIncoming)}
<MessageContextMenu
authorPhoneNumber={this.props.authorPhoneNumber}
convoId={this.props.convoId}
contextMenuId={this.ctxMenuID}
direction={this.props.direction}
isBlocked={this.props.isBlocked}
isDeletable={this.props.isDeletable}
messageId={this.props.id}
text={this.props.text}
timestamp={this.props.timestamp}
serverTimestamp={this.props.serverTimestamp}
attachments={this.props.attachments}
isAdmin={this.props.isSenderAdmin}
isOpenGroupV2={this.props.isOpenGroupV2}
isPublic={this.props.isPublic}
status={this.props.status}
weAreAdmin={this.props.weAreAdmin}
/>
</div>
</ReadableMessage>
);
}
private handleContextMenu(e: any) {
e.preventDefault();
e.stopPropagation();
const { multiSelectMode, isKickedFromGroup } = this.props;
const enableContextMenu = !multiSelectMode && !isKickedFromGroup;
if (enableContextMenu) {
// Don't forget to pass the id and the event and voila!
contextMenu.hideAll();
contextMenu.show({
id: this.ctxMenuID,
event: e,
});
}
}
private onQuoteClick(e: any) {
const { quote, multiSelectMode, id } = this.props;
e.preventDefault();
e.stopPropagation();
if (!quote) {
window.log.warn('onQuoteClick: quote not valid');
return;
}
const quoteId = _.toNumber(quote.messageId);
const { authorPhoneNumber, referencedMessageNotFound } = quote;
if (multiSelectMode && id) {
window.inboxStore?.dispatch(toggleSelectedMessageId(id));
return;
}
void this.props.onQuoteClick?.({
quoteAuthor: authorPhoneNumber,
quoteId,
referencedMessageNotFound,
});
}
private renderAuthor() {
const {
authorName,
authorPhoneNumber,
authorProfileName,
conversationType,
direction,
isPublic,
} = this.props;
const title = authorName ? authorName : authorPhoneNumber;
if (direction !== 'incoming' || conversationType !== 'group' || !title) {
return null;
}
const shortenedPubkey = PubKey.shorten(authorPhoneNumber);
const displayedPubkey = authorProfileName ? shortenedPubkey : authorPhoneNumber;
return (
<div className="module-message__author">
<ContactName
phoneNumber={displayedPubkey}
name={authorName}
profileName={authorProfileName}
module="module-message__author"
boldProfileName={true}
shouldShowPubkey={Boolean(isPublic)}
/>
</div>
);
}
private onMessageAvatarClick() {
const userName =
this.props.authorName || this.props.authorProfileName || this.props.authorPhoneNumber;
window.inboxStore?.dispatch(
updateUserDetailsModal({
conversationId: this.props.authorPhoneNumber,
userName,
authorAvatarPath: this.props.authorAvatarPath,
})
);
}
private onClickOnImageGrid(attachment: AttachmentTypeWithPath | AttachmentType) {
const { multiSelectMode, id } = this.props;
if (multiSelectMode) {
window.inboxStore?.dispatch(toggleSelectedMessageId(id));
} else {
void onClickAttachment({
attachment,
messageId: id,
});
}
}
private onClickOnGenericAttachment(e: any) {
const { timestamp, serverTimestamp, authorPhoneNumber, attachments, convoId } = this.props;
e.stopPropagation();
e.preventDefault();
if (!attachments?.length) {
return;
}
const firstAttachment = attachments[0];
const messageTimestamp = timestamp || serverTimestamp || 0;
void saveAttachmentToDisk({
attachment: firstAttachment,
messageTimestamp,
messageSender: authorPhoneNumber,
conversationId: convoId,
});
}
private onClickOnMessageOuterContainer(event: any) {
const { multiSelectMode, id } = this.props;
const selection = window.getSelection();
// Text is being selected
if (selection && selection.type === 'Range') {
return;
}
// User clicked on message body
const target = event.target as HTMLDivElement;
if ((!multiSelectMode && target.className === 'text-selectable') || window.contextMenuShown) {
return;
}
event.preventDefault();
event.stopPropagation();
if (id) {
window.inboxStore?.dispatch(toggleSelectedMessageId(id));
}
}
private onClickOnMessageInnerContainer(event: any) {
const selection = window.getSelection();
// Text is being selected
if (selection && selection.type === 'Range') {
return;
}
// User clicked on message body
const target = event.target as HTMLDivElement;
if (target.className === 'text-selectable' || window.contextMenuShown) {
return;
}
event.preventDefault();
event.stopPropagation();
}
}
const mapStateToProps = (state: StateType) => {
return {
selectedMessages: getSelectedMessageIds(state),
quotedMessageToAnimate: getQuotedMessageToAnimate(state),
};
};
const smart = connect(mapStateToProps);
export const Message = smart(MessageInner);