diff --git a/_locales/en/messages.json b/_locales/en/messages.json index b650be5d9142..38fb23901606 100644 --- a/_locales/en/messages.json +++ b/_locales/en/messages.json @@ -1281,6 +1281,10 @@ "message": "Send failed", "description": "Shown on outgoing message if it fails to send" }, + "partiallySent": { + "message": "Partially sent, click for details", + "description": "Shown on outgoing message if it is partially sent" + }, "showMore": { "message": "Details", "description": "Displays the details of a key change" diff --git a/js/models/messages.js b/js/models/messages.js index d2ba8d96c16a..c8b1b00f3055 100644 --- a/js/models/messages.js +++ b/js/models/messages.js @@ -688,7 +688,13 @@ return this.get('type') === 'incoming'; }, getMessagePropStatus() { + const sent = this.get('sent'); + const sentTo = this.get('sent_to') || []; + if (this.hasErrors()) { + if (sent || sentTo.length > 0) { + return 'partial-sent'; + } return 'error'; } if (!this.isOutgoing()) { @@ -704,8 +710,6 @@ if (delivered || deliveredTo.length > 0) { return 'delivered'; } - const sent = this.get('sent'); - const sentTo = this.get('sent_to') || []; if (sent || sentTo.length > 0) { return 'sent'; } diff --git a/stylesheets/_modules.scss b/stylesheets/_modules.scss index d427b6487901..2c14ad1edfda 100644 --- a/stylesheets/_modules.scss +++ b/stylesheets/_modules.scss @@ -1170,6 +1170,9 @@ color: $color-white-alpha-80; } } +.module-message__metadata__tapable { + @include button-reset; +} .module-message__metadata__date--incoming { color: $color-white-alpha-80; @@ -3853,7 +3856,8 @@ $timer-icons: '55', '50', '45', '40', '35', '30', '25', '20', '15', '10', '05', } width: 18px; } -.module-conversation-list-item__message__status-icon--error { +.module-conversation-list-item__message__status-icon--error, +.module-conversation-list-item__message__status-icon--partial-sent { @include light-theme { @include color-svg( '../images/icons/v2/error-outline-12.svg', diff --git a/ts/components/ConversationListItem.md b/ts/components/ConversationListItem.md index 2c43d076b145..6d5e9a393cd9 100644 --- a/ts/components/ConversationListItem.md +++ b/ts/components/ConversationListItem.md @@ -153,6 +153,20 @@ onClick={result => console.log('onClick', result)} i18n={util.i18n} /> + console.log('onClick', result)} + i18n={util.i18n} + /> ``` diff --git a/ts/components/ConversationListItem.tsx b/ts/components/ConversationListItem.tsx index 642054335984..aa303a383bf0 100644 --- a/ts/components/ConversationListItem.tsx +++ b/ts/components/ConversationListItem.tsx @@ -32,7 +32,13 @@ export type PropsData = { typingContact?: Object; lastMessage?: { - status: 'sending' | 'sent' | 'delivered' | 'read' | 'error'; + status: + | 'sending' + | 'sent' + | 'delivered' + | 'read' + | 'error' + | 'partial-sent'; text: string; deletedForEveryone?: boolean; }; diff --git a/ts/components/conversation/Message.md b/ts/components/conversation/Message.md index 1ee01f9d172d..f65abaaff95f 100644 --- a/ts/components/conversation/Message.md +++ b/ts/components/conversation/Message.md @@ -200,6 +200,16 @@ Note that timestamp and status can be hidden with the `collapseMetadata` boolean onRetrySend={() => console.log('onRetrySend')} /> +
+ +
+
+ console.log('onRetrySend')} + /> +
+
+ console.log('onRetrySend')} + /> +
+
+ console.log('onRetrySend')} + /> +
+
+ console.log('onRetrySend')} + /> +
{ } } + public renderTimestamp() { + const { + direction, + i18n, + id, + isSticker, + isTapToViewExpired, + showMessageDetail, + status, + text, + timestamp, + } = this.props; + + const isShowingImage = this.isShowingImage(); + const withImageNoCaption = Boolean(!isSticker && !text && isShowingImage); + + const isError = status === 'error' && direction === 'outgoing'; + const isPartiallySent = + status === 'partial-sent' && direction === 'outgoing'; + + if (isError || isPartiallySent) { + return ( + + {isError ? ( + i18n('sendFailed') + ) : ( + + )} + + ); + } + + const metadataDirection = isSticker ? undefined : direction; + + return ( + + ); + } + // tslint:disable-next-line cyclomatic-complexity public renderMetadata() { const { @@ -388,14 +453,12 @@ export class Message extends React.PureComponent { direction, expirationLength, expirationTimestamp, - i18n, isSticker, isTapToViewExpired, reactions, status, text, textPending, - timestamp, } = this.props; if (collapseMetadata) { @@ -405,7 +468,6 @@ export class Message extends React.PureComponent { const isShowingImage = this.isShowingImage(); const withImageNoCaption = Boolean(!isSticker && !text && isShowingImage); const withReactions = reactions && reactions.length > 0; - const showError = status === 'error' && direction === 'outgoing'; const metadataDirection = isSticker ? undefined : direction; return ( @@ -419,33 +481,7 @@ export class Message extends React.PureComponent { : null )} > - {showError ? ( - - {i18n('sendFailed')} - - ) : ( - - )} + {this.renderTimestamp()} {expirationLength && expirationTimestamp ? ( {
) : null} - {!textPending && direction === 'outgoing' && status !== 'error' ? ( + {!textPending && + direction === 'outgoing' && + status !== 'error' && + status !== 'partial-sent' ? (
{ public renderError(isCorrectSide: boolean) { const { status, direction } = this.props; - if (!isCorrectSide || status !== 'error') { + if (!isCorrectSide || (status !== 'error' && status !== 'partial-sent')) { return null; } diff --git a/ts/state/ducks/conversations.ts b/ts/state/ducks/conversations.ts index 97b2f96e93ec..c24fe6b606d0 100644 --- a/ts/state/ducks/conversations.ts +++ b/ts/state/ducks/conversations.ts @@ -38,7 +38,13 @@ export type ConversationType = { timestamp?: number; inboxPosition?: number; lastMessage?: { - status: 'error' | 'sending' | 'sent' | 'delivered' | 'read'; + status: + | 'error' + | 'partial-sent' + | 'sending' + | 'sent' + | 'delivered' + | 'read'; text: string; }; phoneNumber?: string;