diff --git a/ts/components/conversation/Quote.tsx b/ts/components/conversation/Quote.tsx index 0193380ec..d5b4ce05e 100644 --- a/ts/components/conversation/Quote.tsx +++ b/ts/components/conversation/Quote.tsx @@ -1,6 +1,6 @@ // tslint:disable:react-this-binding-issue -import React from 'react'; +import React, { useState } from 'react'; import classNames from 'classnames'; import * as MIME from '../../../ts/types/MIME'; @@ -12,7 +12,9 @@ import { ContactName } from './ContactName'; import { PubKey } from '../../session/types'; import { ConversationTypeEnum } from '../../models/conversation'; -interface Props { +import { useEncryptedFileFetch } from '../../hooks/useEncryptedFileFetch'; + +interface QuoteProps { attachment?: QuotedAttachmentType; authorPhoneNumber: string; authorProfileName?: string; @@ -25,15 +27,10 @@ interface Props { isPublic?: boolean; withContentAbove: boolean; onClick?: (e: any) => void; - onClose?: () => void; text: string; referencedMessageNotFound: boolean; } -interface State { - imageBroken: boolean; -} - export interface QuotedAttachmentType { contentType: MIME.MIMEType; fileName: string; @@ -48,7 +45,7 @@ interface Attachment { objectUrl?: string; } -function validateQuote(quote: Props): boolean { +function validateQuote(quote: QuoteProps): boolean { if (quote.text) { return true; } @@ -92,30 +89,11 @@ function getTypeLabel({ return; } +export const QuoteIcon = (props: any) => { + const { icon } = props; -export class Quote extends React.Component { - public handleImageErrorBound: () => void; - - public constructor(props: Props) { - super(props); - - this.handleImageErrorBound = this.handleImageError.bind(this); - - this.state = { - imageBroken: false, - }; - } - - public handleImageError() { - // tslint:disable-next-line no-console - console.log('Message: Image failed to load; failing over to placeholder'); - this.setState({ - imageBroken: true, - }); - } - - public renderImage(url: string, i18n: LocalizerType, icon?: string) { - const iconElement = icon ? ( + return ( +
{ />
- ) : null; - - return ( -
- {i18n('quoteThumbnailAlt')} - {iconElement} -
- ); - } - - public renderIcon(icon: string) { - return ( -
-
-
-
-
-
-
- ); - } - - public renderGenericFile() { - const { attachment, isIncoming } = this.props; +
+ ); +}; - if (!attachment) { - return; - } +export const QuoteImage = (props: any) => { + const { url, i18n, icon, contentType, handleImageErrorBound } = props; - const { fileName, contentType } = attachment; - const isGenericFile = - !GoogleChrome.isVideoTypeSupported(contentType) && - !GoogleChrome.isImageTypeSupported(contentType) && - !MIME.isAudio(contentType); + const { loading, urlToLoad } = useEncryptedFileFetch(url, contentType); + const srcData = !loading ? urlToLoad : ''; - if (!isGenericFile) { - return null; - } - - return ( -
-
+ const iconElement = icon ? ( +
+
- {fileName} -
+ />
- ); +
+ ) : null; + + return ( +
+ {i18n('quoteThumbnailAlt')} + {iconElement} +
+ ); +}; + +export const QuoteGenericFile = (props: any) => { + const { attachment, isIncoming } = props; + + if (!attachment) { + return <>; } - public renderIconContainer() { - const { attachment, i18n } = this.props; - const { imageBroken } = this.state; - - if (!attachment) { - return null; - } - - const { contentType, thumbnail } = attachment; - const objectUrl = getObjectUrl(thumbnail); - - if (GoogleChrome.isVideoTypeSupported(contentType)) { - return objectUrl && !imageBroken - ? this.renderImage(objectUrl, i18n, 'play') - : this.renderIcon('movie'); - } - if (GoogleChrome.isImageTypeSupported(contentType)) { - return objectUrl && !imageBroken - ? this.renderImage(objectUrl, i18n) - : this.renderIcon('image'); - } - if (MIME.isAudio(contentType)) { - return this.renderIcon('microphone'); - } + const { fileName, contentType } = attachment; + const isGenericFile = + !GoogleChrome.isVideoTypeSupported(contentType) && + !GoogleChrome.isImageTypeSupported(contentType) && + !MIME.isAudio(contentType); - return null; + if (!isGenericFile) { + return <>; } - public renderText() { - const { i18n, text, attachment, isIncoming, conversationType, convoId } = this.props; - - if (text) { - return ( -
- -
- ); - } - - if (!attachment) { - return null; - } + return ( +
+
+
+ {fileName} +
+
+ ); +}; - const { contentType, isVoiceMessage } = attachment; - - const typeLabel = getTypeLabel({ i18n, contentType, isVoiceMessage }); - if (typeLabel) { - return ( -
- {typeLabel} -
- ); - } +export const QuoteIconContainer = (props: any) => { + const { attachment, i18n, imageBroken, handleImageErrorBound } = props; + if (!attachment) { return null; } - public renderClose() { - const { onClose } = this.props; + const { contentType, thumbnail } = attachment; + const objectUrl = getObjectUrl(thumbnail); - if (!onClose) { - return null; - } + if (GoogleChrome.isVideoTypeSupported(contentType)) { + return objectUrl && !imageBroken ? ( + + ) : ( + + ); + } + if (GoogleChrome.isImageTypeSupported(contentType)) { + return objectUrl && !imageBroken ? ( + + ) : ( + + ); + } + if (MIME.isAudio(contentType)) { + return ; + } + return null; +}; - // We don't want the overall click handler for the quote to fire, so we stop - // propagation before handing control to the caller's callback. - const onClick = (e: any): void => { - e.stopPropagation(); - onClose(); - }; +export const QuoteText = (props: any) => { + const { i18n, text, attachment, isIncoming, conversationType, convoId } = props; + const isGroup = conversationType === ConversationTypeEnum.GROUP; - // We need the container to give us the flexibility to implement the iOS design. + if (text) { return ( -
-
+
+
); } - public renderAuthor() { - const { - authorProfileName, - authorPhoneNumber, - authorName, - i18n, - isFromMe, - isIncoming, - isPublic, - } = this.props; + if (!attachment) { + return null; + } + + const { contentType, isVoiceMessage } = attachment; + const typeLabel = getTypeLabel({ i18n, contentType, isVoiceMessage }); + if (typeLabel) { return (
- {isFromMe ? ( - i18n('you') - ) : ( - - )} + {typeLabel}
); } - public renderReferenceWarning() { - const { i18n, isIncoming, referencedMessageNotFound } = this.props; + return null; +}; + +export const QuoteAuthor = (props: any) => { + const { + authorProfileName, + authorPhoneNumber, + authorName, + i18n, + isFromMe, + isIncoming, + isPublic, + } = props; + + return ( +
+ {isFromMe ? ( + i18n('you') + ) : ( + + )} +
+ ); +}; - if (!referencedMessageNotFound) { - return null; - } +export const QuoteReferenceWarning = (props: any) => { + const { i18n, isIncoming, referencedMessageNotFound } = props; - return ( + if (!referencedMessageNotFound) { + return null; + } + + return ( +
+
-
-
- {i18n('originalMessageNotFound')} -
+ {i18n('originalMessageNotFound')}
- ); - } +
+ ); +}; - public render() { - const { isIncoming, onClick, referencedMessageNotFound, withContentAbove } = this.props; +export const Quote = (props: QuoteProps) => { + const [imageBroken, setImageBroken] = useState(false); - if (!validateQuote(this.props)) { - return null; - } + const handleImageErrorBound = null; - return ( + const handleImageError = () => { + // tslint:disable-next-line no-console + console.log('Message: Image failed to load; failing over to placeholder'); + setImageBroken(true); + }; + + const { isIncoming, onClick, referencedMessageNotFound, withContentAbove } = props; + + if (!validateQuote(props)) { + return null; + } + + return ( + <>
{ )} >
- {this.renderAuthor()} - {this.renderGenericFile()} - {this.renderText()} + + +
- {this.renderIconContainer()} - {this.renderClose()} +
- {this.renderReferenceWarning()} +
- ); - } -} + + ); +}; diff --git a/ts/components/session/conversation/SessionQuotedMessageComposition.tsx b/ts/components/session/conversation/SessionQuotedMessageComposition.tsx index 8a7a2a240..140c37d2d 100644 --- a/ts/components/session/conversation/SessionQuotedMessageComposition.tsx +++ b/ts/components/session/conversation/SessionQuotedMessageComposition.tsx @@ -1,8 +1,10 @@ import React, { useContext } from 'react'; import { Flex } from '../../basic/Flex'; -import { SessionIconButton, SessionIconSize, SessionIconType } from '../icon'; +import { SessionIcon, SessionIconButton, SessionIconSize, SessionIconType } from '../icon'; import { ReplyingToMessageProps } from './SessionCompositionBox'; import styled, { DefaultTheme, ThemeContext } from 'styled-components'; +import { getAlt, isAudio, isImageAttachment } from '../../../types/Attachment'; +import { Image } from '../../conversation/Image'; // tslint:disable: react-unused-props-and-state interface Props { @@ -44,6 +46,18 @@ export const SessionQuotedMessageComposition = (props: Props) => { const { text: body, attachments } = quotedMessageProps; const hasAttachments = attachments && attachments.length > 0; + + let hasImageAttachment = false; + + let firstImageAttachment; + if (attachments && attachments.length > 0) { + firstImageAttachment = attachments[0]; + hasImageAttachment = true; + } + + const hasAudioAttachment = + hasAttachments && attachments && attachments.length > 0 && isAudio(attachments); + return ( { /> - {(hasAttachments && window.i18n('mediaMessage')) || body} + + {(hasAttachments && window.i18n('mediaMessage')) || body} + + {hasImageAttachment && ( + {getAlt(firstImageAttachment, + )} + + {hasAudioAttachment && ( + + )} + ); diff --git a/ts/receiver/queuedJob.ts b/ts/receiver/queuedJob.ts index 23ca84dbd..195a1ad0e 100644 --- a/ts/receiver/queuedJob.ts +++ b/ts/receiver/queuedJob.ts @@ -79,7 +79,7 @@ async function handleGroups( return groupUpdate; } -function contentTypeSupported(type: any): boolean { +function contentTypeSupported(type: string): boolean { const Chrome = window.Signal.Util.GoogleChrome; return Chrome.isImageTypeSupported(type) || Chrome.isVideoTypeSupported(type); } @@ -139,7 +139,7 @@ async function copyFromQuotedMessage( return; } - if (!firstAttachment || !contentTypeSupported(firstAttachment)) { + if (!firstAttachment || !contentTypeSupported(firstAttachment.contentType)) { return; }