fix: hide known message status except for last message

pull/2940/head
Audric Ackermann 1 year ago
parent 615722434b
commit b2d22b2a73

@ -5,7 +5,7 @@ import React, { createContext, useCallback, useContext, useLayoutEffect, useStat
import { InView } from 'react-intersection-observer'; import { InView } from 'react-intersection-observer';
import { useSelector } from 'react-redux'; import { useSelector } from 'react-redux';
import styled, { css, keyframes } from 'styled-components'; import styled, { css, keyframes } from 'styled-components';
import { MessageModelType, MessageRenderingProps } from '../../../../models/messageType'; import { MessageRenderingProps } from '../../../../models/messageType';
import { StateType } from '../../../../state/reducer'; import { StateType } from '../../../../state/reducer';
import { useMessageIsDeleted } from '../../../../state/selectors'; import { useMessageIsDeleted } from '../../../../state/selectors';
import { import {
@ -77,14 +77,14 @@ export const StyledMessageHighlighter = styled.div<{
`; `;
const StyledMessageOpaqueContent = styled(StyledMessageHighlighter)<{ const StyledMessageOpaqueContent = styled(StyledMessageHighlighter)<{
messageDirection: MessageModelType; isIncoming: boolean;
highlight: boolean; highlight: boolean;
}>` }>`
background: ${props => background: ${props =>
props.messageDirection === 'incoming' props.isIncoming
? 'var(--message-bubbles-received-background-color)' ? 'var(--message-bubbles-received-background-color)'
: 'var(--message-bubbles-sent-background-color)'}; : 'var(--message-bubbles-sent-background-color)'};
align-self: ${props => (props.messageDirection === 'incoming' ? 'flex-start' : 'flex-end')}; align-self: ${props => (props.isIncoming ? 'flex-start' : 'flex-end')};
padding: var(--padding-message-content); padding: var(--padding-message-content);
border-radius: var(--border-radius-message-box); border-radius: var(--border-radius-message-box);
width: 100%; width: 100%;
@ -183,7 +183,7 @@ export const MessageContent = (props: Props) => {
> >
<IsMessageVisibleContext.Provider value={isMessageVisible}> <IsMessageVisibleContext.Provider value={isMessageVisible}>
{hasContentBeforeAttachment && ( {hasContentBeforeAttachment && (
<StyledMessageOpaqueContent messageDirection={direction} highlight={highlight}> <StyledMessageOpaqueContent isIncoming={direction === 'incoming'} highlight={highlight}>
{!isDeleted && ( {!isDeleted && (
<> <>
<MessageQuote messageId={props.messageId} /> <MessageQuote messageId={props.messageId} />

@ -1,9 +1,11 @@
import { ipcRenderer } from 'electron'; import { ipcRenderer } from 'electron';
import React from 'react'; import React, { useCallback } from 'react';
import { useSelector } from 'react-redux';
import styled from 'styled-components'; import styled from 'styled-components';
import { useMessageExpirationPropsById } from '../../../../hooks/useParamSelector'; import { useMessageExpirationPropsById } from '../../../../hooks/useParamSelector';
import { useMessageStatus } from '../../../../state/selectors'; import { useMessageStatus } from '../../../../state/selectors';
import { getMostRecentMessageId } from '../../../../state/selectors/conversations';
import { SpacerXS } from '../../../basic/Text'; import { SpacerXS } from '../../../basic/Text';
import { SessionIcon, SessionIconType } from '../../../icon'; import { SessionIcon, SessionIconType } from '../../../icon';
import { ExpireTimer } from '../../ExpireTimer'; import { ExpireTimer } from '../../ExpireTimer';
@ -13,6 +15,19 @@ type Props = {
dataTestId?: string | undefined; dataTestId?: string | undefined;
}; };
/**
* MessageStatus is used to display the status of an outgoing OR incoming message.
* There are 3 parts to this status: a status text, a status icon and a expiring stopwatch.
* At all times, we either display `text + icon` OR `text + stopwatch`.
*
* The logic to display the text is :
* - if the message is expiring:
* - if the message is incoming: display its 'read' state and the stopwatch icon (1)
* - if the message is outgoing: display its status and the stopwatch, unless when the status is error or sending (just display icon and text in this case, no stopwatch) (2)
* - if the message is not expiring:
* - if the message is incoming: do not show anything (3)
* - if the message is outgoing: show the text for the last message, or a message sending, or in the error state. (4)
*/
export const MessageStatus = (props: Props) => { export const MessageStatus = (props: Props) => {
const { dataTestId, messageId } = props; const { dataTestId, messageId } = props;
const status = useMessageStatus(props.messageId); const status = useMessageStatus(props.messageId);
@ -25,40 +40,39 @@ export const MessageStatus = (props: Props) => {
if (isIncoming) { if (isIncoming) {
if (selected.isUnread || !selected.expirationDurationMs || !selected.expirationTimestamp) { if (selected.isUnread || !selected.expirationDurationMs || !selected.expirationTimestamp) {
return null; return null; // incoming and not expiring, this is case (3) above
} }
return ( // incoming and expiring, this is case (1) above
<MessageStatusRead dataTestId={dataTestId} messageId={messageId} reserveDirection={true} /> return <MessageStatusRead dataTestId={dataTestId} messageId={messageId} isIncoming={true} />;
);
} }
// this is the outgoing state: we display the text and the icon or the text and the expiretimer stopwatch when the message is expiring
switch (status) { switch (status) {
case 'sending': case 'sending':
return <MessageStatusSending dataTestId={dataTestId} messageId={messageId} />; return <MessageStatusSending dataTestId={dataTestId} messageId={messageId} />; // we always show sending state
case 'sent': case 'sent':
return <MessageStatusSent dataTestId={dataTestId} messageId={messageId} />; return <MessageStatusSent dataTestId={dataTestId} messageId={messageId} />;
case 'read': case 'read':
return <MessageStatusRead dataTestId={dataTestId} messageId={messageId} />; return <MessageStatusRead dataTestId={dataTestId} messageId={messageId} isIncoming={false} />; // read is used for both incoming and outgoing messages, but not with the same UI
case 'error': case 'error':
return <MessageStatusError dataTestId={dataTestId} messageId={messageId} />; return <MessageStatusError dataTestId={dataTestId} messageId={messageId} />; // we always show error state
default: default:
return null; return null;
} }
}; };
const MessageStatusContainer = styled.div<{ reserveDirection?: boolean }>` const MessageStatusContainer = styled.div<{ isIncoming: boolean }>`
display: inline-block; display: inline-block;
align-self: flex-end; align-self: ${props => (props.isIncoming ? 'flex-start' : 'flex-end')};
flex-direction: ${props =>
props.isIncoming
? 'row-reverse'
: 'row'}; // we want {icon}{text} for incoming read messages, but {text}{icon} for outgoing messages
margin-bottom: 2px; margin-bottom: 2px;
margin-inline-start: 5px; margin-inline-start: 5px;
cursor: pointer; cursor: pointer;
display: flex; display: flex;
align-items: baseline; align-items: baseline;
flex-direction: ${props =>
props.reserveDirection
? 'row-reverse'
: 'row'}; // we want {icon}{text} for incoming read messages, but {text}{icon} for outgoing messages
`; `;
const StyledStatusText = styled.div` const StyledStatusText = styled.div`
@ -103,6 +117,12 @@ function useIsExpiring(messageId: string) {
); );
} }
function useIsMostRecentMessage(messageId: string) {
const mostRecentMessageId = useSelector(getMostRecentMessageId);
const isMostRecentMessage = mostRecentMessageId === messageId;
return isMostRecentMessage;
}
function MessageStatusExpireTimer(props: Props) { function MessageStatusExpireTimer(props: Props) {
const selected = useMessageExpirationPropsById(props.messageId); const selected = useMessageExpirationPropsById(props.messageId);
if ( if (
@ -124,24 +144,42 @@ function MessageStatusExpireTimer(props: Props) {
const MessageStatusSending = ({ dataTestId }: Props) => { const MessageStatusSending = ({ dataTestId }: Props) => {
// while sending, we do not display the expire timer at all. // while sending, we do not display the expire timer at all.
return ( return (
<MessageStatusContainer data-testid={dataTestId} data-testtype="sending"> <MessageStatusContainer data-testid={dataTestId} data-testtype="sending" isIncoming={false}>
<TextDetails text={window.i18n('sending')} /> <TextDetails text={window.i18n('sending')} />
<IconNormal rotateDuration={2} iconType="sending" /> <IconNormal rotateDuration={2} iconType="sending" />
</MessageStatusContainer> </MessageStatusContainer>
); );
}; };
/**
* Returns the correct expiring stopwatch icon if this message is expiring, or a normal status icon otherwise.
* Only to be used with the status "read" and "sent"
*/
function IconForExpiringMessageId({
messageId,
iconType,
}: Pick<Props, 'messageId'> & { iconType: SessionIconType }) {
const isExpiring = useIsExpiring(messageId);
return isExpiring ? (
<MessageStatusExpireTimer messageId={messageId} />
) : (
<IconNormal iconType={iconType} />
);
}
const MessageStatusSent = ({ dataTestId, messageId }: Props) => { const MessageStatusSent = ({ dataTestId, messageId }: Props) => {
const isExpiring = useIsExpiring(messageId); const isExpiring = useIsExpiring(messageId);
const isMostRecentMessage = useIsMostRecentMessage(messageId);
// we hide a "sent" message status which is not expiring except for the most recent message
if (!isExpiring && !isMostRecentMessage) {
return null;
}
return ( return (
<MessageStatusContainer data-testid={dataTestId} data-testtype="sent"> <MessageStatusContainer data-testid={dataTestId} data-testtype="sent" isIncoming={false}>
<TextDetails text={window.i18n('sent')} /> <TextDetails text={window.i18n('sent')} />
{isExpiring ? ( <IconForExpiringMessageId messageId={messageId} iconType="circleCheck" />
<MessageStatusExpireTimer messageId={messageId} />
) : (
<IconNormal iconType="circleCheck" />
)}
</MessageStatusContainer> </MessageStatusContainer>
); );
}; };
@ -149,29 +187,29 @@ const MessageStatusSent = ({ dataTestId, messageId }: Props) => {
const MessageStatusRead = ({ const MessageStatusRead = ({
dataTestId, dataTestId,
messageId, messageId,
reserveDirection, isIncoming,
}: Props & { reserveDirection?: boolean }) => { }: Props & { isIncoming: boolean }) => {
const isExpiring = useIsExpiring(messageId); const isExpiring = useIsExpiring(messageId);
const isMostRecentMessage = useIsMostRecentMessage(messageId);
// we hide an outgoing "read" message status which is not expiring except for the most recent message
if (!isIncoming && !isExpiring && !isMostRecentMessage) {
return null;
}
return ( return (
<MessageStatusContainer <MessageStatusContainer data-testid={dataTestId} data-testtype="read" isIncoming={isIncoming}>
data-testid={dataTestId}
data-testtype="read"
reserveDirection={reserveDirection}
>
<TextDetails text={window.i18n('read')} /> <TextDetails text={window.i18n('read')} />
{isExpiring ? ( <IconForExpiringMessageId messageId={messageId} iconType="doubleCheckCircleFilled" />
<MessageStatusExpireTimer messageId={messageId} />
) : (
<IconNormal iconType="doubleCheckCircleFilled" />
)}
</MessageStatusContainer> </MessageStatusContainer>
); );
}; };
const MessageStatusError = ({ dataTestId }: Props) => { const MessageStatusError = ({ dataTestId }: Props) => {
const showDebugLog = () => { const showDebugLog = useCallback(() => {
ipcRenderer.send('show-debug-log'); ipcRenderer.send('show-debug-log');
}; }, []);
// when on errro, we do not display the expire timer at all. // when on errro, we do not display the expire timer at all.
return ( return (
@ -180,6 +218,7 @@ const MessageStatusError = ({ dataTestId }: Props) => {
data-testtype="failed" data-testtype="failed"
onClick={showDebugLog} onClick={showDebugLog}
title={window.i18n('sendFailed')} title={window.i18n('sendFailed')}
isIncoming={false}
> >
<TextDetails text={window.i18n('failed')} /> <TextDetails text={window.i18n('failed')} />
<IconDanger iconType="error" /> <IconDanger iconType="error" />

Loading…
Cancel
Save