2022-01-07 18:01:23 +00:00
|
|
|
|
// Copyright 2020-2022 Signal Messenger, LLC
|
2020-10-30 20:34:04 +00:00
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
2021-10-19 00:09:55 +00:00
|
|
|
|
/* eslint-disable camelcase */
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
import type * as Backbone from 'backbone';
|
|
|
|
|
import type { ComponentProps } from 'react';
|
|
|
|
|
import * as React from 'react';
|
2022-12-08 23:56:17 +00:00
|
|
|
|
import { flatten } from 'lodash';
|
2021-08-30 21:32:56 +00:00
|
|
|
|
import { render } from 'mustache';
|
2021-08-11 16:23:21 +00:00
|
|
|
|
|
2021-12-03 01:05:32 +00:00
|
|
|
|
import type { AttachmentType } from '../types/Attachment';
|
2021-10-26 19:15:33 +00:00
|
|
|
|
import { isGIF } from '../types/Attachment';
|
2021-07-09 19:36:10 +00:00
|
|
|
|
import * as Stickers from '../types/Stickers';
|
2021-10-26 19:15:33 +00:00
|
|
|
|
import type { MIMEType } from '../types/MIME';
|
|
|
|
|
import type { ConversationModel } from '../models/conversations';
|
2022-12-08 06:41:37 +00:00
|
|
|
|
import type { MessageAttributesType } from '../model-types.d';
|
2022-06-03 16:33:39 +00:00
|
|
|
|
import type { MediaItemType, MediaItemMessageType } from '../types/MediaItem';
|
2022-01-31 21:52:09 +00:00
|
|
|
|
import { getMessageById } from '../messages/getMessageById';
|
2021-12-10 22:51:54 +00:00
|
|
|
|
import { getContactId } from '../messages/helpers';
|
2021-08-30 21:32:56 +00:00
|
|
|
|
import { strictAssert } from '../util/assert';
|
2021-10-29 23:19:44 +00:00
|
|
|
|
import { enqueueReactionForSend } from '../reactions/enqueueReactionForSend';
|
2021-10-26 19:15:33 +00:00
|
|
|
|
import type { GroupNameCollisionsWithIdsByTitle } from '../util/groupMemberNameCollisions';
|
2022-12-08 06:41:37 +00:00
|
|
|
|
import { isDirectConversation, isGroup } from '../util/whatTypeOfConversation';
|
2021-06-22 23:16:50 +00:00
|
|
|
|
import { findAndFormatContact } from '../util/findAndFormatContact';
|
2022-06-03 16:33:39 +00:00
|
|
|
|
import { getPreferredBadgeSelector } from '../state/selectors/badges';
|
2021-06-17 17:15:10 +00:00
|
|
|
|
import {
|
|
|
|
|
canReply,
|
2021-11-03 17:02:26 +00:00
|
|
|
|
isIncoming,
|
2021-06-17 17:15:10 +00:00
|
|
|
|
isOutgoing,
|
|
|
|
|
isTapToView,
|
|
|
|
|
} from '../state/selectors/message';
|
2021-08-23 23:14:53 +00:00
|
|
|
|
import {
|
|
|
|
|
getConversationSelector,
|
|
|
|
|
getMessagesByConversation,
|
|
|
|
|
} from '../state/selectors/conversations';
|
2022-02-15 23:00:47 +00:00
|
|
|
|
import { getActiveCallState } from '../state/selectors/calling';
|
2022-06-03 16:33:39 +00:00
|
|
|
|
import { getTheme } from '../state/selectors/user';
|
|
|
|
|
import { ReactWrapperView } from './ReactWrapperView';
|
2022-06-16 19:12:50 +00:00
|
|
|
|
import type { Lightbox } from '../components/Lightbox';
|
2021-06-17 21:15:09 +00:00
|
|
|
|
import { ConversationDetailsMembershipList } from '../components/conversation/conversation-details/ConversationDetailsMembershipList';
|
2021-09-29 20:23:06 +00:00
|
|
|
|
import * as log from '../logging/log';
|
|
|
|
|
import type { EmbeddedContactType } from '../types/EmbeddedContact';
|
2021-10-05 16:47:06 +00:00
|
|
|
|
import { createConversationView } from '../state/roots/createConversationView';
|
2021-09-22 20:59:54 +00:00
|
|
|
|
import { ToastConversationArchived } from '../components/ToastConversationArchived';
|
|
|
|
|
import { ToastConversationMarkedUnread } from '../components/ToastConversationMarkedUnread';
|
|
|
|
|
import { ToastConversationUnarchived } from '../components/ToastConversationUnarchived';
|
|
|
|
|
import { ToastDangerousFileType } from '../components/ToastDangerousFileType';
|
|
|
|
|
import { ToastMessageBodyTooLong } from '../components/ToastMessageBodyTooLong';
|
|
|
|
|
import { ToastOriginalMessageNotFound } from '../components/ToastOriginalMessageNotFound';
|
|
|
|
|
import { ToastReactionFailed } from '../components/ToastReactionFailed';
|
|
|
|
|
import { ToastTapToViewExpiredIncoming } from '../components/ToastTapToViewExpiredIncoming';
|
|
|
|
|
import { ToastTapToViewExpiredOutgoing } from '../components/ToastTapToViewExpiredOutgoing';
|
2021-09-24 20:02:30 +00:00
|
|
|
|
import { ToastUnableToLoadAttachment } from '../components/ToastUnableToLoadAttachment';
|
2022-05-11 20:59:58 +00:00
|
|
|
|
import { ToastCannotOpenGiftBadge } from '../components/ToastCannotOpenGiftBadge';
|
2021-09-24 20:02:30 +00:00
|
|
|
|
import { deleteDraftAttachment } from '../util/deleteDraftAttachment';
|
2021-09-27 16:29:00 +00:00
|
|
|
|
import { retryMessageSend } from '../util/retryMessageSend';
|
2021-09-29 20:23:06 +00:00
|
|
|
|
import { isNotNil } from '../util/isNotNil';
|
|
|
|
|
import { openLinkInWebBrowser } from '../util/openLinkInWebBrowser';
|
|
|
|
|
import { showToast } from '../util/showToast';
|
2022-02-23 01:06:19 +00:00
|
|
|
|
import { UUIDKind } from '../types/UUID';
|
2022-03-16 00:11:28 +00:00
|
|
|
|
import type { UUIDStringType } from '../types/UUID';
|
2022-03-04 19:22:31 +00:00
|
|
|
|
import { retryDeleteForEveryone } from '../util/retryDeleteForEveryone';
|
2022-06-03 16:33:39 +00:00
|
|
|
|
import { ContactDetail } from '../components/conversation/ContactDetail';
|
|
|
|
|
import { MediaGallery } from '../components/conversation/media-gallery/MediaGallery';
|
|
|
|
|
import type { ItemClickEvent } from '../components/conversation/media-gallery/types/ItemClickEvent';
|
2022-06-17 00:48:57 +00:00
|
|
|
|
import {
|
|
|
|
|
removeLinkPreview,
|
|
|
|
|
suspendLinkPreviews,
|
|
|
|
|
} from '../services/LinkPreview';
|
2022-06-23 00:21:38 +00:00
|
|
|
|
import { closeLightbox, showLightbox } from '../util/showLightbox';
|
2022-07-01 00:52:03 +00:00
|
|
|
|
import { saveAttachment } from '../util/saveAttachment';
|
2022-07-05 16:30:55 +00:00
|
|
|
|
import { SECOND } from '../util/durations';
|
2022-12-05 22:56:23 +00:00
|
|
|
|
import { startConversation } from '../util/startConversation';
|
2022-12-06 17:31:44 +00:00
|
|
|
|
import { longRunningTaskWrapper } from '../util/longRunningTaskWrapper';
|
2022-12-08 23:56:17 +00:00
|
|
|
|
import { hasDraftAttachments } from '../util/hasDraftAttachments';
|
2020-09-28 23:46:31 +00:00
|
|
|
|
|
2021-03-22 18:51:53 +00:00
|
|
|
|
type AttachmentOptions = {
|
|
|
|
|
messageId: string;
|
|
|
|
|
attachment: AttachmentType;
|
|
|
|
|
};
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
type PanelType = { view: Backbone.View; headerTitle?: string };
|
|
|
|
|
|
2021-09-24 00:49:05 +00:00
|
|
|
|
const { Message } = window.Signal.Types;
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
|
|
|
|
const {
|
|
|
|
|
copyIntoTempDirectory,
|
|
|
|
|
deleteTempFile,
|
|
|
|
|
getAbsoluteAttachmentPath,
|
|
|
|
|
getAbsoluteTempPath,
|
|
|
|
|
upgradeMessageSchema,
|
|
|
|
|
} = window.Signal.Migrations;
|
|
|
|
|
|
2022-01-31 21:52:09 +00:00
|
|
|
|
const { getMessagesBySentAt } = window.Signal.Data;
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
type MessageActionsType = {
|
|
|
|
|
deleteMessage: (messageId: string) => unknown;
|
|
|
|
|
displayTapToViewMessage: (messageId: string) => unknown;
|
|
|
|
|
downloadAttachment: (options: {
|
|
|
|
|
attachment: AttachmentType;
|
|
|
|
|
timestamp: number;
|
|
|
|
|
isDangerous: boolean;
|
|
|
|
|
}) => unknown;
|
|
|
|
|
downloadNewVersion: () => unknown;
|
|
|
|
|
kickOffAttachmentDownload: (
|
|
|
|
|
options: Readonly<{ messageId: string }>
|
|
|
|
|
) => unknown;
|
|
|
|
|
markAttachmentAsCorrupted: (options: AttachmentOptions) => unknown;
|
|
|
|
|
openConversation: (conversationId: string, messageId?: string) => unknown;
|
2022-05-11 20:59:58 +00:00
|
|
|
|
openGiftBadge: (messageId: string) => unknown;
|
2021-08-30 21:32:56 +00:00
|
|
|
|
openLink: (url: string) => unknown;
|
|
|
|
|
reactToMessage: (
|
|
|
|
|
messageId: string,
|
|
|
|
|
reaction: { emoji: string; remove: boolean }
|
|
|
|
|
) => unknown;
|
|
|
|
|
replyToMessage: (messageId: string) => unknown;
|
|
|
|
|
retrySend: (messageId: string) => unknown;
|
2022-03-04 19:22:31 +00:00
|
|
|
|
retryDeleteForEveryone: (messageId: string) => unknown;
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showContactDetail: (options: {
|
|
|
|
|
contact: EmbeddedContactType;
|
2022-04-12 00:26:09 +00:00
|
|
|
|
signalAccount?: {
|
|
|
|
|
phoneNumber: string;
|
|
|
|
|
uuid: UUIDStringType;
|
|
|
|
|
};
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}) => unknown;
|
|
|
|
|
showContactModal: (contactId: string) => unknown;
|
|
|
|
|
showSafetyNumber: (contactId: string) => unknown;
|
|
|
|
|
showExpiredIncomingTapToViewToast: () => unknown;
|
|
|
|
|
showExpiredOutgoingTapToViewToast: () => unknown;
|
|
|
|
|
showIdentity: (conversationId: string) => unknown;
|
|
|
|
|
showMessageDetail: (messageId: string) => unknown;
|
|
|
|
|
showVisualAttachment: (options: {
|
|
|
|
|
attachment: AttachmentType;
|
|
|
|
|
messageId: string;
|
|
|
|
|
showSingle?: boolean;
|
|
|
|
|
}) => unknown;
|
2022-04-12 00:26:09 +00:00
|
|
|
|
startConversation: (e164: string, uuid: UUIDStringType) => unknown;
|
2021-08-30 21:32:56 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
type MediaType = {
|
|
|
|
|
path: string;
|
|
|
|
|
objectURL: string;
|
|
|
|
|
thumbnailObjectUrl?: string;
|
|
|
|
|
contentType: MIMEType;
|
|
|
|
|
index: number;
|
|
|
|
|
attachment: AttachmentType;
|
|
|
|
|
message: {
|
|
|
|
|
attachments: Array<AttachmentType>;
|
|
|
|
|
conversationId: string;
|
|
|
|
|
id: string;
|
|
|
|
|
received_at: number;
|
|
|
|
|
received_at_ms: number;
|
|
|
|
|
sent_at: number;
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export class ConversationView extends window.Backbone.View<ConversationModel> {
|
|
|
|
|
// Sub-views
|
|
|
|
|
private contactModalView?: Backbone.View;
|
2022-06-03 16:33:39 +00:00
|
|
|
|
private conversationView?: Backbone.View;
|
|
|
|
|
private lightboxView?: ReactWrapperView;
|
2021-08-30 21:32:56 +00:00
|
|
|
|
private stickerPreviewModalView?: Backbone.View;
|
|
|
|
|
|
|
|
|
|
// Panel support
|
2022-06-03 16:33:39 +00:00
|
|
|
|
private panels: Array<PanelType> = [];
|
2021-08-30 21:32:56 +00:00
|
|
|
|
private previousFocus?: HTMLElement;
|
|
|
|
|
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
|
constructor(...args: Array<any>) {
|
|
|
|
|
super(...args);
|
|
|
|
|
|
2020-09-24 20:57:54 +00:00
|
|
|
|
// Events on Conversation model
|
2021-06-16 00:44:14 +00:00
|
|
|
|
this.listenTo(this.model, 'destroy', this.stopListening);
|
|
|
|
|
|
|
|
|
|
// These are triggered by InboxView
|
|
|
|
|
this.listenTo(this.model, 'opened', this.onOpened);
|
|
|
|
|
this.listenTo(this.model, 'scroll-to-message', this.scrollToMessage);
|
|
|
|
|
this.listenTo(this.model, 'unload', (reason: string) =>
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.unload(`model trigger - ${reason}`)
|
|
|
|
|
);
|
2021-06-16 00:44:14 +00:00
|
|
|
|
|
|
|
|
|
// These are triggered by background.ts for keyboard handling
|
|
|
|
|
this.listenTo(this.model, 'open-all-media', this.showAllMedia);
|
|
|
|
|
this.listenTo(this.model, 'escape-pressed', this.resetPanel);
|
|
|
|
|
this.listenTo(this.model, 'show-message-details', this.showMessageDetail);
|
|
|
|
|
this.listenTo(this.model, 'show-contact-modal', this.showContactModal);
|
|
|
|
|
this.listenTo(
|
|
|
|
|
this.model,
|
|
|
|
|
'toggle-reply',
|
|
|
|
|
(messageId: string | undefined) => {
|
2022-12-08 01:26:59 +00:00
|
|
|
|
const composerState = window.reduxStore
|
|
|
|
|
? window.reduxStore.getState().composer
|
|
|
|
|
: undefined;
|
|
|
|
|
const quote = composerState?.quotedMessage?.quote;
|
|
|
|
|
|
|
|
|
|
this.setQuoteMessage(quote ? undefined : messageId);
|
2021-06-16 00:44:14 +00:00
|
|
|
|
}
|
|
|
|
|
);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.listenTo(
|
|
|
|
|
this.model,
|
|
|
|
|
'save-attachment',
|
|
|
|
|
this.downloadAttachmentWrapper
|
2020-09-24 20:57:54 +00:00
|
|
|
|
);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.listenTo(this.model, 'delete-message', this.deleteMessage);
|
2022-06-17 00:48:57 +00:00
|
|
|
|
this.listenTo(this.model, 'remove-link-review', removeLinkPreview);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.listenTo(
|
|
|
|
|
this.model,
|
|
|
|
|
'remove-all-draft-attachments',
|
|
|
|
|
this.clearAttachments
|
2020-09-24 20:57:54 +00:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
this.render();
|
|
|
|
|
|
2021-10-05 16:47:06 +00:00
|
|
|
|
this.setupConversationView();
|
2021-06-25 16:08:16 +00:00
|
|
|
|
this.updateAttachmentsView();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-11-12 23:44:20 +00:00
|
|
|
|
override events(): Record<string, string> {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
return {
|
|
|
|
|
drop: 'onDrop',
|
|
|
|
|
paste: 'onPaste',
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// We need this ignore because the backbone types really want this to be a string
|
|
|
|
|
// property, but the property isn't set until after super() is run, meaning that this
|
|
|
|
|
// classname wouldn't be applied when Backbone creates our el.
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
className(): string {
|
|
|
|
|
return 'conversation';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Same situation as className().
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
id(): string {
|
|
|
|
|
return `conversation-${this.model.cid}`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Backbone.View<ConversationModel> is demanded as the return type here, and we can't
|
|
|
|
|
// satisfy it because of the above difference in signature: className is a function
|
|
|
|
|
// when it should be a plain string property.
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
|
|
|
// @ts-ignore
|
|
|
|
|
render(): ConversationView {
|
|
|
|
|
const template = $('#conversation').html();
|
|
|
|
|
this.$el.html(render(template, {}));
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-05 16:47:06 +00:00
|
|
|
|
setupConversationView(): void {
|
|
|
|
|
// setupHeader
|
|
|
|
|
const conversationHeaderProps = {
|
|
|
|
|
id: this.model.id,
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-10-05 16:47:06 +00:00
|
|
|
|
onSearchInConversation: () => {
|
|
|
|
|
const { searchInConversation } = window.reduxActions.search;
|
2021-11-01 18:43:02 +00:00
|
|
|
|
searchInConversation(this.model.id);
|
2021-10-05 16:47:06 +00:00
|
|
|
|
},
|
|
|
|
|
onShowConversationDetails: () => {
|
|
|
|
|
this.showConversationDetails();
|
|
|
|
|
},
|
|
|
|
|
onShowAllMedia: () => {
|
|
|
|
|
this.showAllMedia();
|
|
|
|
|
},
|
|
|
|
|
onShowGroupMembers: () => {
|
|
|
|
|
this.showGV1Members();
|
|
|
|
|
},
|
|
|
|
|
onGoBack: () => {
|
|
|
|
|
this.resetPanel();
|
|
|
|
|
},
|
2021-06-25 16:08:16 +00:00
|
|
|
|
|
2021-10-05 16:47:06 +00:00
|
|
|
|
onArchive: () => {
|
|
|
|
|
this.model.setArchived(true);
|
|
|
|
|
this.model.trigger('unload', 'archive');
|
|
|
|
|
|
2021-10-06 21:00:51 +00:00
|
|
|
|
showToast(ToastConversationArchived, {
|
|
|
|
|
undo: () => {
|
|
|
|
|
this.model.setArchived(false);
|
|
|
|
|
this.openConversation(this.model.get('id'));
|
|
|
|
|
},
|
|
|
|
|
});
|
2021-10-05 16:47:06 +00:00
|
|
|
|
},
|
|
|
|
|
onMarkUnread: () => {
|
|
|
|
|
this.model.setMarkedUnread(true);
|
|
|
|
|
|
|
|
|
|
showToast(ToastConversationMarkedUnread);
|
|
|
|
|
},
|
|
|
|
|
onMoveToInbox: () => {
|
|
|
|
|
this.model.setArchived(false);
|
|
|
|
|
|
|
|
|
|
showToast(ToastConversationUnarchived);
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
window.reduxActions.conversations.setSelectedConversationHeaderTitle();
|
|
|
|
|
|
|
|
|
|
// setupTimeline
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-10-05 16:47:06 +00:00
|
|
|
|
const contactSupport = () => {
|
|
|
|
|
const baseUrl =
|
|
|
|
|
'https://support.signal.org/hc/LOCALE/requests/new?desktop&chat_refreshed';
|
|
|
|
|
const locale = window.getLocale();
|
|
|
|
|
const supportLocale = window.Signal.Util.mapToSupportLocale(locale);
|
|
|
|
|
const url = baseUrl.replace('LOCALE', supportLocale);
|
|
|
|
|
|
|
|
|
|
openLinkInWebBrowser(url);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const learnMoreAboutDeliveryIssue = () => {
|
|
|
|
|
openLinkInWebBrowser(
|
|
|
|
|
'https://support.signal.org/hc/articles/4404859745690'
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const scrollToQuotedMessage = async (
|
|
|
|
|
options: Readonly<{
|
|
|
|
|
authorId: string;
|
|
|
|
|
sentAt: number;
|
|
|
|
|
}>
|
|
|
|
|
) => {
|
|
|
|
|
const { authorId, sentAt } = options;
|
|
|
|
|
|
|
|
|
|
const conversationId = this.model.id;
|
2021-12-10 22:51:54 +00:00
|
|
|
|
const messages = await getMessagesBySentAt(sentAt);
|
2021-10-05 16:47:06 +00:00
|
|
|
|
const message = messages.find(item =>
|
|
|
|
|
Boolean(
|
2021-12-10 22:51:54 +00:00
|
|
|
|
item.conversationId === conversationId &&
|
2021-10-05 16:47:06 +00:00
|
|
|
|
authorId &&
|
2021-12-10 22:51:54 +00:00
|
|
|
|
getContactId(item) === authorId
|
2021-10-05 16:47:06 +00:00
|
|
|
|
)
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (!message) {
|
|
|
|
|
showToast(ToastOriginalMessageNotFound);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.scrollToMessage(message.id);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const markMessageRead = async (messageId: string) => {
|
2022-07-05 16:44:53 +00:00
|
|
|
|
if (!window.SignalContext.activeWindowService.isActive()) {
|
2021-10-05 16:47:06 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-15 23:00:47 +00:00
|
|
|
|
const activeCall = getActiveCallState(window.reduxStore.getState());
|
|
|
|
|
if (activeCall && !activeCall.pip) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-10 22:51:54 +00:00
|
|
|
|
const message = await getMessageById(messageId);
|
2021-10-05 16:47:06 +00:00
|
|
|
|
if (!message) {
|
|
|
|
|
throw new Error(`markMessageRead: failed to load message ${messageId}`);
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-22 18:35:14 +00:00
|
|
|
|
await this.model.markRead(message.get('received_at'), {
|
|
|
|
|
newestSentAt: message.get('sent_at'),
|
|
|
|
|
sendReadReceipts: true,
|
|
|
|
|
});
|
2021-10-05 16:47:06 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const timelineProps = {
|
|
|
|
|
id: this.model.id,
|
|
|
|
|
|
|
|
|
|
...this.getMessageActions(),
|
|
|
|
|
|
|
|
|
|
acknowledgeGroupMemberNameCollisions: (
|
|
|
|
|
groupNameCollisions: Readonly<GroupNameCollisionsWithIdsByTitle>
|
|
|
|
|
): void => {
|
|
|
|
|
this.model.acknowledgeGroupMemberNameCollisions(groupNameCollisions);
|
|
|
|
|
},
|
2022-03-16 00:11:28 +00:00
|
|
|
|
blockGroupLinkRequests: (uuid: UUIDStringType) => {
|
|
|
|
|
this.model.blockGroupLinkRequests(uuid);
|
|
|
|
|
},
|
2021-10-05 16:47:06 +00:00
|
|
|
|
contactSupport,
|
|
|
|
|
learnMoreAboutDeliveryIssue,
|
2021-11-30 11:25:24 +00:00
|
|
|
|
loadNewerMessages: this.model.loadNewerMessages.bind(this.model),
|
|
|
|
|
loadNewestMessages: this.model.loadNewestMessages.bind(this.model),
|
|
|
|
|
loadOlderMessages: this.model.loadOlderMessages.bind(this.model),
|
2021-10-05 16:47:06 +00:00
|
|
|
|
markMessageRead,
|
|
|
|
|
removeMember: (conversationId: string) => {
|
2022-12-06 17:31:44 +00:00
|
|
|
|
longRunningTaskWrapper({
|
|
|
|
|
idForLogging: this.model.idForLogging(),
|
2021-10-05 16:47:06 +00:00
|
|
|
|
name: 'removeMember',
|
|
|
|
|
task: () => this.model.removeFromGroupV2(conversationId),
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
scrollToQuotedMessage,
|
|
|
|
|
unblurAvatar: () => {
|
|
|
|
|
this.model.unblurAvatar();
|
|
|
|
|
},
|
|
|
|
|
updateSharedGroups: () => this.model.throttledUpdateSharedGroups?.(),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// setupCompositionArea
|
|
|
|
|
window.reduxActions.composer.resetComposer();
|
|
|
|
|
|
|
|
|
|
const compositionAreaProps = {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
id: this.model.id,
|
2020-09-24 20:57:54 +00:00
|
|
|
|
onClickAddPack: () => this.showStickerManager(),
|
2021-09-22 20:59:54 +00:00
|
|
|
|
onTextTooLong: () => showToast(ToastMessageBodyTooLong),
|
2021-08-30 21:32:56 +00:00
|
|
|
|
getQuotedMessage: () => this.model.get('quotedMessageId'),
|
2022-12-08 01:26:59 +00:00
|
|
|
|
clearQuotedMessage: () => this.setQuoteMessage(undefined),
|
2021-01-29 22:16:48 +00:00
|
|
|
|
onCancelJoinRequest: async () => {
|
|
|
|
|
await window.showConfirmationDialog({
|
2022-09-27 20:24:21 +00:00
|
|
|
|
dialogName: 'GroupV2CancelRequestToJoin',
|
2021-01-29 22:16:48 +00:00
|
|
|
|
message: window.i18n(
|
|
|
|
|
'GroupV2--join--cancel-request-to-join--confirmation'
|
|
|
|
|
),
|
|
|
|
|
okText: window.i18n('GroupV2--join--cancel-request-to-join--yes'),
|
|
|
|
|
cancelText: window.i18n('GroupV2--join--cancel-request-to-join--no'),
|
|
|
|
|
resolve: () => {
|
2022-12-06 17:31:44 +00:00
|
|
|
|
longRunningTaskWrapper({
|
|
|
|
|
idForLogging: this.model.idForLogging(),
|
2021-01-29 22:16:48 +00:00
|
|
|
|
name: 'onCancelJoinRequest',
|
2021-08-30 21:32:56 +00:00
|
|
|
|
task: async () => this.model.cancelJoinRequest(),
|
2021-01-29 22:16:48 +00:00
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
},
|
2021-06-25 16:08:16 +00:00
|
|
|
|
|
|
|
|
|
onClearAttachments: this.clearAttachments.bind(this),
|
|
|
|
|
onSelectMediaQuality: (isHQ: boolean) => {
|
|
|
|
|
window.reduxActions.composer.setMediaQualitySetting(isHQ);
|
|
|
|
|
},
|
|
|
|
|
|
2021-10-05 16:47:06 +00:00
|
|
|
|
handleClickQuotedMessage: (id: string) => this.scrollToMessage(id),
|
2021-06-25 16:08:16 +00:00
|
|
|
|
|
|
|
|
|
onCloseLinkPreview: () => {
|
2022-06-17 00:48:57 +00:00
|
|
|
|
suspendLinkPreviews();
|
|
|
|
|
removeLinkPreview();
|
2021-06-25 16:08:16 +00:00
|
|
|
|
},
|
2021-07-20 20:18:35 +00:00
|
|
|
|
|
|
|
|
|
openConversation: this.openConversation.bind(this),
|
2020-09-24 20:57:54 +00:00
|
|
|
|
};
|
|
|
|
|
|
2021-10-05 16:47:06 +00:00
|
|
|
|
// createConversationView root
|
|
|
|
|
|
|
|
|
|
const JSX = createConversationView(window.reduxStore, {
|
2022-12-08 01:26:59 +00:00
|
|
|
|
conversationId: this.model.id,
|
2021-10-05 16:47:06 +00:00
|
|
|
|
compositionAreaProps,
|
|
|
|
|
conversationHeaderProps,
|
|
|
|
|
timelineProps,
|
2020-09-24 20:57:54 +00:00
|
|
|
|
});
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.conversationView = new ReactWrapperView({ JSX });
|
2021-10-05 16:47:06 +00:00
|
|
|
|
this.$('.ConversationView__template').append(this.conversationView.el);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
getMessageActions(): MessageActionsType {
|
2021-10-29 23:19:44 +00:00
|
|
|
|
const reactToMessage = async (
|
2021-06-08 14:59:38 +00:00
|
|
|
|
messageId: string,
|
|
|
|
|
reaction: { emoji: string; remove: boolean }
|
|
|
|
|
) => {
|
2021-10-29 23:19:44 +00:00
|
|
|
|
const { emoji, remove } = reaction;
|
|
|
|
|
try {
|
|
|
|
|
await enqueueReactionForSend({
|
|
|
|
|
messageId,
|
|
|
|
|
emoji,
|
|
|
|
|
remove,
|
|
|
|
|
});
|
|
|
|
|
} catch (error) {
|
|
|
|
|
log.error('Error sending reaction', error, messageId, reaction);
|
|
|
|
|
showToast(ToastReactionFailed);
|
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
};
|
2021-06-08 14:59:38 +00:00
|
|
|
|
const replyToMessage = (messageId: string) => {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.setQuoteMessage(messageId);
|
|
|
|
|
};
|
2021-09-27 16:29:00 +00:00
|
|
|
|
const retrySend = retryMessageSend;
|
2021-06-08 14:59:38 +00:00
|
|
|
|
const deleteMessage = (messageId: string) => {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.deleteMessage(messageId);
|
|
|
|
|
};
|
2021-06-08 14:59:38 +00:00
|
|
|
|
const showMessageDetail = (messageId: string) => {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.showMessageDetail(messageId);
|
|
|
|
|
};
|
2020-11-11 17:36:05 +00:00
|
|
|
|
const showContactModal = (contactId: string) => {
|
|
|
|
|
this.showContactModal(contactId);
|
|
|
|
|
};
|
2021-08-30 21:32:56 +00:00
|
|
|
|
const openConversation = (conversationId: string, messageId?: string) => {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.openConversation(conversationId, messageId);
|
|
|
|
|
};
|
2021-08-16 21:35:54 +00:00
|
|
|
|
const showContactDetail = (options: {
|
2021-08-20 01:56:39 +00:00
|
|
|
|
contact: EmbeddedContactType;
|
2022-04-12 00:26:09 +00:00
|
|
|
|
signalAccount?: {
|
|
|
|
|
phoneNumber: string;
|
|
|
|
|
uuid: UUIDStringType;
|
|
|
|
|
};
|
2021-08-16 21:35:54 +00:00
|
|
|
|
}) => {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.showContactDetail(options);
|
|
|
|
|
};
|
2021-08-16 21:35:54 +00:00
|
|
|
|
const kickOffAttachmentDownload = async (
|
|
|
|
|
options: Readonly<{ messageId: string }>
|
|
|
|
|
) => {
|
2021-06-16 00:44:14 +00:00
|
|
|
|
const message = window.MessageController.getById(options.messageId);
|
|
|
|
|
if (!message) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`kickOffAttachmentDownload: Message ${options.messageId} missing!`
|
|
|
|
|
);
|
2021-01-29 22:58:28 +00:00
|
|
|
|
}
|
|
|
|
|
await message.queueAttachmentDownloads();
|
|
|
|
|
};
|
2021-03-22 18:51:53 +00:00
|
|
|
|
const markAttachmentAsCorrupted = (options: AttachmentOptions) => {
|
2021-06-16 00:44:14 +00:00
|
|
|
|
const message = window.MessageController.getById(options.messageId);
|
|
|
|
|
if (!message) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`markAttachmentAsCorrupted: Message ${options.messageId} missing!`
|
|
|
|
|
);
|
|
|
|
|
}
|
2021-03-22 18:51:53 +00:00
|
|
|
|
message.markAttachmentAsCorrupted(options.attachment);
|
|
|
|
|
};
|
2021-08-17 15:43:26 +00:00
|
|
|
|
|
2021-06-08 14:59:38 +00:00
|
|
|
|
const showVisualAttachment = (options: {
|
2021-07-14 23:39:52 +00:00
|
|
|
|
attachment: AttachmentType;
|
2021-06-08 14:59:38 +00:00
|
|
|
|
messageId: string;
|
|
|
|
|
showSingle?: boolean;
|
|
|
|
|
}) => {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.showLightbox(options);
|
|
|
|
|
};
|
2021-08-16 21:35:54 +00:00
|
|
|
|
const downloadAttachment = (options: {
|
|
|
|
|
attachment: AttachmentType;
|
|
|
|
|
timestamp: number;
|
|
|
|
|
isDangerous: boolean;
|
|
|
|
|
}) => {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.downloadAttachment(options);
|
|
|
|
|
};
|
2021-06-08 14:59:38 +00:00
|
|
|
|
const displayTapToViewMessage = (messageId: string) =>
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.displayTapToViewMessage(messageId);
|
2021-06-08 14:59:38 +00:00
|
|
|
|
const showIdentity = (conversationId: string) => {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.showSafetyNumber(conversationId);
|
|
|
|
|
};
|
2022-05-11 20:59:58 +00:00
|
|
|
|
const openGiftBadge = (messageId: string): void => {
|
|
|
|
|
const message = window.MessageController.getById(messageId);
|
|
|
|
|
if (!message) {
|
|
|
|
|
throw new Error(`openGiftBadge: Message ${messageId} missing!`);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
showToast(ToastCannotOpenGiftBadge, {
|
|
|
|
|
isIncoming: isIncoming(message.attributes),
|
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
|
2021-09-21 20:45:25 +00:00
|
|
|
|
const openLink = openLinkInWebBrowser;
|
2020-09-24 20:57:54 +00:00
|
|
|
|
const downloadNewVersion = () => {
|
2021-09-21 20:45:25 +00:00
|
|
|
|
openLinkInWebBrowser('https://signal.org/download');
|
2020-09-24 20:57:54 +00:00
|
|
|
|
};
|
2021-06-16 00:44:14 +00:00
|
|
|
|
const showSafetyNumber = (contactId: string) => {
|
|
|
|
|
this.showSafetyNumber(contactId);
|
|
|
|
|
};
|
2020-09-24 20:57:54 +00:00
|
|
|
|
const showExpiredIncomingTapToViewToast = () => {
|
2021-09-17 18:27:53 +00:00
|
|
|
|
log.info('Showing expired tap-to-view toast for an incoming message');
|
2021-09-22 20:59:54 +00:00
|
|
|
|
showToast(ToastTapToViewExpiredIncoming);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
};
|
|
|
|
|
const showExpiredOutgoingTapToViewToast = () => {
|
2021-09-17 18:27:53 +00:00
|
|
|
|
log.info('Showing expired tap-to-view toast for an outgoing message');
|
2021-09-22 20:59:54 +00:00
|
|
|
|
showToast(ToastTapToViewExpiredOutgoing);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
};
|
2021-08-30 21:32:56 +00:00
|
|
|
|
|
2021-03-24 22:06:12 +00:00
|
|
|
|
return {
|
|
|
|
|
deleteMessage,
|
|
|
|
|
displayTapToViewMessage,
|
|
|
|
|
downloadAttachment,
|
|
|
|
|
downloadNewVersion,
|
|
|
|
|
kickOffAttachmentDownload,
|
|
|
|
|
markAttachmentAsCorrupted,
|
|
|
|
|
openConversation,
|
2022-05-11 20:59:58 +00:00
|
|
|
|
openGiftBadge,
|
2021-03-24 22:06:12 +00:00
|
|
|
|
openLink,
|
|
|
|
|
reactToMessage,
|
|
|
|
|
replyToMessage,
|
|
|
|
|
retrySend,
|
2022-03-04 19:22:31 +00:00
|
|
|
|
retryDeleteForEveryone,
|
2021-03-24 22:06:12 +00:00
|
|
|
|
showContactDetail,
|
|
|
|
|
showContactModal,
|
2021-06-16 00:44:14 +00:00
|
|
|
|
showSafetyNumber,
|
2021-03-24 22:06:12 +00:00
|
|
|
|
showExpiredIncomingTapToViewToast,
|
|
|
|
|
showExpiredOutgoingTapToViewToast,
|
|
|
|
|
showIdentity,
|
|
|
|
|
showMessageDetail,
|
|
|
|
|
showVisualAttachment,
|
2022-04-12 00:26:09 +00:00
|
|
|
|
startConversation,
|
2021-03-24 22:06:12 +00:00
|
|
|
|
};
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2021-03-24 22:06:12 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
async scrollToMessage(messageId: string): Promise<void> {
|
2021-12-10 22:51:54 +00:00
|
|
|
|
const message = await getMessageById(messageId);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (!message) {
|
|
|
|
|
throw new Error(`scrollToMessage: failed to load message ${messageId}`);
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-23 22:57:39 +00:00
|
|
|
|
const state = window.reduxStore.getState();
|
|
|
|
|
|
|
|
|
|
let isInMemory = true;
|
|
|
|
|
|
|
|
|
|
if (!window.MessageController.getById(messageId)) {
|
|
|
|
|
isInMemory = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Message might be in memory, but not in the redux anymore because
|
|
|
|
|
// we call `messageReset()` in `loadAndScroll()`.
|
2021-11-11 22:43:05 +00:00
|
|
|
|
const messagesByConversation =
|
|
|
|
|
getMessagesByConversation(state)[this.model.id];
|
2021-06-23 22:57:39 +00:00
|
|
|
|
if (!messagesByConversation?.messageIds.includes(messageId)) {
|
|
|
|
|
isInMemory = false;
|
|
|
|
|
}
|
2021-06-16 00:44:14 +00:00
|
|
|
|
|
|
|
|
|
if (isInMemory) {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
const { scrollToMessage } = window.reduxActions.conversations;
|
2021-08-30 21:32:56 +00:00
|
|
|
|
scrollToMessage(this.model.id, messageId);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-11-30 11:25:24 +00:00
|
|
|
|
this.model.loadAndScroll(messageId);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
unload(reason: string): void {
|
2021-09-17 18:27:53 +00:00
|
|
|
|
log.info(
|
2020-09-24 20:57:54 +00:00
|
|
|
|
'unloading conversation',
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.model.idForLogging(),
|
2020-09-24 20:57:54 +00:00
|
|
|
|
'due to:',
|
|
|
|
|
reason
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const { conversationUnloaded } = window.reduxActions.conversations;
|
|
|
|
|
if (conversationUnloaded) {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
conversationUnloaded(this.model.id);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (this.model.get('draftChanged')) {
|
|
|
|
|
if (this.model.hasDraft()) {
|
2021-10-19 00:09:55 +00:00
|
|
|
|
const now = Date.now();
|
|
|
|
|
const active_at = this.model.get('active_at') || now;
|
|
|
|
|
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.model.set({
|
2021-10-19 00:09:55 +00:00
|
|
|
|
active_at,
|
2020-09-24 20:57:54 +00:00
|
|
|
|
draftChanged: false,
|
2021-10-19 00:09:55 +00:00
|
|
|
|
draftTimestamp: now,
|
|
|
|
|
timestamp: now,
|
2020-09-24 20:57:54 +00:00
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
this.model.set({
|
|
|
|
|
draftChanged: false,
|
|
|
|
|
draftTimestamp: null,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// We don't wait here; we need to take down the view
|
|
|
|
|
this.saveModel();
|
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.model.updateLastMessage();
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-10-05 16:47:06 +00:00
|
|
|
|
this.conversationView?.remove();
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2020-11-11 17:36:05 +00:00
|
|
|
|
if (this.contactModalView) {
|
|
|
|
|
this.contactModalView.remove();
|
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (this.stickerPreviewModalView) {
|
|
|
|
|
this.stickerPreviewModalView.remove();
|
|
|
|
|
}
|
|
|
|
|
if (this.lightboxView) {
|
|
|
|
|
this.lightboxView.remove();
|
|
|
|
|
}
|
|
|
|
|
if (this.panels && this.panels.length) {
|
|
|
|
|
for (let i = 0, max = this.panels.length; i < max; i += 1) {
|
|
|
|
|
const panel = this.panels[i];
|
2022-06-03 16:33:39 +00:00
|
|
|
|
panel.view.remove();
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
2020-10-30 17:52:21 +00:00
|
|
|
|
window.reduxActions.conversations.setSelectedConversationPanelDepth(0);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2022-06-17 00:48:57 +00:00
|
|
|
|
removeLinkPreview();
|
|
|
|
|
suspendLinkPreviews();
|
2021-08-23 18:40:49 +00:00
|
|
|
|
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.remove();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
async saveModel(): Promise<void> {
|
|
|
|
|
window.Signal.Data.updateConversation(this.model.attributes);
|
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
async onOpened(messageId: string): Promise<void> {
|
2022-01-20 00:40:29 +00:00
|
|
|
|
this.model.onOpenStart();
|
|
|
|
|
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (messageId) {
|
2021-12-10 22:51:54 +00:00
|
|
|
|
const message = await getMessageById(messageId);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
|
|
|
|
if (message) {
|
2021-11-30 11:25:24 +00:00
|
|
|
|
this.model.loadAndScroll(messageId);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
|
log.warn(`onOpened: Did not find message ${messageId}`);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-05-28 19:11:19 +00:00
|
|
|
|
const { retryPlaceholders } = window.Signal.Services;
|
|
|
|
|
if (retryPlaceholders) {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
await retryPlaceholders.findByConversationAndMarkOpened(this.model.id);
|
2021-05-28 19:11:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
2022-01-07 18:20:29 +00:00
|
|
|
|
const loadAndUpdate = async () => {
|
2022-06-20 18:38:52 +00:00
|
|
|
|
Promise.all([
|
|
|
|
|
this.model.loadNewestMessages(undefined, undefined),
|
|
|
|
|
this.model.updateLastMessage(),
|
|
|
|
|
this.model.updateUnread(),
|
|
|
|
|
]);
|
2022-01-07 18:20:29 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
loadAndUpdate();
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2022-12-08 23:56:17 +00:00
|
|
|
|
window.reduxActions.composer.setComposerFocus(this.model.id);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
const quotedMessageId = this.model.get('quotedMessageId');
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (quotedMessageId) {
|
|
|
|
|
this.setQuoteMessage(quotedMessageId);
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.model.fetchLatestGroupV2Data();
|
|
|
|
|
strictAssert(
|
|
|
|
|
this.model.throttledMaybeMigrateV1Group !== undefined,
|
2021-05-28 19:11:19 +00:00
|
|
|
|
'Conversation model should be initialized'
|
|
|
|
|
);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.model.throttledMaybeMigrateV1Group();
|
|
|
|
|
strictAssert(
|
|
|
|
|
this.model.throttledFetchSMSOnlyUUID !== undefined,
|
2021-05-13 20:57:27 +00:00
|
|
|
|
'Conversation model should be initialized'
|
|
|
|
|
);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.model.throttledFetchSMSOnlyUUID();
|
|
|
|
|
|
2022-02-23 01:06:19 +00:00
|
|
|
|
const ourUuid = window.textsecure.storage.user.getUuid(UUIDKind.ACI);
|
|
|
|
|
if (
|
|
|
|
|
!isGroup(this.model.attributes) ||
|
2022-07-08 20:46:25 +00:00
|
|
|
|
(ourUuid && this.model.hasMember(ourUuid))
|
2022-02-23 01:06:19 +00:00
|
|
|
|
) {
|
|
|
|
|
strictAssert(
|
|
|
|
|
this.model.throttledGetProfiles !== undefined,
|
|
|
|
|
'Conversation model should be initialized'
|
|
|
|
|
);
|
|
|
|
|
await this.model.throttledGetProfiles();
|
|
|
|
|
}
|
2021-08-30 21:32:56 +00:00
|
|
|
|
|
|
|
|
|
this.model.updateVerified();
|
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-09-29 20:18:27 +00:00
|
|
|
|
showAllMedia(): void {
|
2021-10-13 17:05:18 +00:00
|
|
|
|
if (document.querySelectorAll('.module-media-gallery').length) {
|
2021-09-29 20:18:27 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-24 20:57:54 +00:00
|
|
|
|
// 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 DEFAULT_MEDIA_FETCH_COUNT = 50;
|
|
|
|
|
const DEFAULT_DOCUMENTS_FETCH_COUNT = 150;
|
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
const conversationId = this.model.get('id');
|
2021-12-20 21:04:02 +00:00
|
|
|
|
const ourUuid = window.textsecure.storage.user.getCheckedUuid().toString();
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
|
|
|
|
const getProps = async () => {
|
2021-11-11 22:43:05 +00:00
|
|
|
|
const rawMedia =
|
|
|
|
|
await window.Signal.Data.getMessagesWithVisualMediaAttachments(
|
|
|
|
|
conversationId,
|
|
|
|
|
{
|
|
|
|
|
limit: DEFAULT_MEDIA_FETCH_COUNT,
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
const rawDocuments =
|
|
|
|
|
await window.Signal.Data.getMessagesWithFileAttachments(
|
|
|
|
|
conversationId,
|
|
|
|
|
{
|
|
|
|
|
limit: DEFAULT_DOCUMENTS_FETCH_COUNT,
|
|
|
|
|
}
|
|
|
|
|
);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
|
|
|
|
// First we upgrade these messages to ensure that they have thumbnails
|
|
|
|
|
for (let max = rawMedia.length, i = 0; i < max; i += 1) {
|
|
|
|
|
const message = rawMedia[i];
|
|
|
|
|
const { schemaVersion } = message;
|
|
|
|
|
|
2022-11-04 20:32:29 +00:00
|
|
|
|
// We want these message to be cached in memory for other operations like
|
|
|
|
|
// listening to 'expired' events when showing the lightbox, and so any other
|
|
|
|
|
// code working with this message has the latest updates.
|
|
|
|
|
const model = window.MessageController.register(message.id, message);
|
|
|
|
|
|
2021-06-17 17:15:10 +00:00
|
|
|
|
if (
|
|
|
|
|
schemaVersion &&
|
|
|
|
|
schemaVersion < Message.VERSION_NEEDED_FOR_DISPLAY
|
|
|
|
|
) {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
// Yep, we really do want to wait for each of these
|
|
|
|
|
// eslint-disable-next-line no-await-in-loop
|
|
|
|
|
rawMedia[i] = await upgradeMessageSchema(message);
|
2022-11-04 20:32:29 +00:00
|
|
|
|
model.set(rawMedia[i]);
|
|
|
|
|
|
2020-09-24 20:57:54 +00:00
|
|
|
|
// eslint-disable-next-line no-await-in-loop
|
2021-12-20 21:04:02 +00:00
|
|
|
|
await window.Signal.Data.saveMessage(rawMedia[i], { ourUuid });
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
const media: Array<MediaType> = flatten(
|
2020-09-24 20:57:54 +00:00
|
|
|
|
rawMedia.map(message => {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
return (message.attachments || []).map(
|
|
|
|
|
(
|
|
|
|
|
attachment: AttachmentType,
|
|
|
|
|
index: number
|
|
|
|
|
): MediaType | undefined => {
|
|
|
|
|
if (
|
|
|
|
|
!attachment.path ||
|
|
|
|
|
!attachment.thumbnail ||
|
|
|
|
|
attachment.pending ||
|
|
|
|
|
attachment.error
|
|
|
|
|
) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
const { thumbnail } = attachment;
|
2020-09-24 20:57:54 +00:00
|
|
|
|
return {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
path: attachment.path,
|
2020-09-24 20:57:54 +00:00
|
|
|
|
objectURL: getAbsoluteAttachmentPath(attachment.path),
|
2022-06-10 01:10:20 +00:00
|
|
|
|
thumbnailObjectUrl: thumbnail?.path
|
2020-09-24 20:57:54 +00:00
|
|
|
|
? getAbsoluteAttachmentPath(thumbnail.path)
|
2021-08-30 21:32:56 +00:00
|
|
|
|
: undefined,
|
2020-09-24 20:57:54 +00:00
|
|
|
|
contentType: attachment.contentType,
|
|
|
|
|
index,
|
|
|
|
|
attachment,
|
2021-08-24 21:47:14 +00:00
|
|
|
|
message: {
|
|
|
|
|
attachments: message.attachments || [],
|
|
|
|
|
conversationId:
|
2022-08-09 21:39:00 +00:00
|
|
|
|
window.ConversationController.lookupOrCreate({
|
|
|
|
|
uuid: message.sourceUuid,
|
|
|
|
|
e164: message.source,
|
2022-12-03 01:05:27 +00:00
|
|
|
|
reason: 'conversation_view.showAllMedia',
|
2022-08-09 21:39:00 +00:00
|
|
|
|
})?.id || message.conversationId,
|
2021-08-24 21:47:14 +00:00
|
|
|
|
id: message.id,
|
|
|
|
|
received_at: message.received_at,
|
|
|
|
|
received_at_ms: Number(message.received_at_ms),
|
2021-08-25 21:08:32 +00:00
|
|
|
|
sent_at: message.sent_at,
|
2021-08-24 21:47:14 +00:00
|
|
|
|
},
|
2020-09-24 20:57:54 +00:00
|
|
|
|
};
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
|
|
|
|
);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
})
|
2021-08-30 21:32:56 +00:00
|
|
|
|
).filter(isNotNil);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
|
|
|
|
// Unlike visual media, only one non-image attachment is supported
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const documents: Array<MediaItemType> = [];
|
|
|
|
|
rawDocuments.forEach(message => {
|
|
|
|
|
const attachments = message.attachments || [];
|
|
|
|
|
const attachment = attachments[0];
|
|
|
|
|
if (!attachment) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
documents.push({
|
|
|
|
|
contentType: attachment.contentType,
|
|
|
|
|
index: 0,
|
|
|
|
|
attachment,
|
|
|
|
|
// We do this cast because we know there attachments (see the checks above).
|
|
|
|
|
message: message as MessageAttributesType & {
|
|
|
|
|
attachments: Array<AttachmentType>;
|
|
|
|
|
},
|
2020-09-24 20:57:54 +00:00
|
|
|
|
});
|
2022-06-03 16:33:39 +00:00
|
|
|
|
});
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
const onItemClick = async ({
|
|
|
|
|
message,
|
|
|
|
|
attachment,
|
|
|
|
|
type,
|
2022-06-03 16:33:39 +00:00
|
|
|
|
}: ItemClickEvent) => {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
switch (type) {
|
|
|
|
|
case 'documents': {
|
2022-07-01 00:52:03 +00:00
|
|
|
|
saveAttachment(attachment, message.sent_at);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case 'media': {
|
2021-08-24 21:47:14 +00:00
|
|
|
|
const selectedMedia =
|
|
|
|
|
media.find(item => attachment.path === item.path) || media[0];
|
|
|
|
|
this.showLightboxForMedia(selectedMedia, media);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
throw new TypeError(`Unknown attachment type: '${type}'`);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
documents,
|
|
|
|
|
media,
|
|
|
|
|
onItemClick,
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
2021-06-16 00:44:14 +00:00
|
|
|
|
function getMessageIds(): Array<string | undefined> | undefined {
|
|
|
|
|
const state = window.reduxStore.getState();
|
|
|
|
|
const byConversation = state?.conversations?.messagesByConversation;
|
|
|
|
|
const messages = byConversation && byConversation[conversationId];
|
|
|
|
|
if (!messages || !messages.messageIds) {
|
|
|
|
|
return undefined;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return messages.messageIds;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Detect message changes in the current conversation
|
|
|
|
|
let previousMessageList: Array<string | undefined> | undefined;
|
|
|
|
|
previousMessageList = getMessageIds();
|
|
|
|
|
|
|
|
|
|
const unsubscribe = window.reduxStore.subscribe(() => {
|
|
|
|
|
const currentMessageList = getMessageIds();
|
|
|
|
|
if (currentMessageList !== previousMessageList) {
|
|
|
|
|
update();
|
|
|
|
|
previousMessageList = currentMessageList;
|
|
|
|
|
}
|
|
|
|
|
});
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const view = new ReactWrapperView({
|
2021-09-29 20:18:27 +00:00
|
|
|
|
className: 'panel',
|
2022-06-03 16:33:39 +00:00
|
|
|
|
// We present an empty panel briefly, while we wait for props to load.
|
2022-11-18 00:45:19 +00:00
|
|
|
|
// eslint-disable-next-line react/jsx-no-useless-fragment
|
2022-06-03 16:33:39 +00:00
|
|
|
|
JSX: <></>,
|
2021-09-29 20:18:27 +00:00
|
|
|
|
onClose: () => {
|
|
|
|
|
unsubscribe();
|
|
|
|
|
},
|
|
|
|
|
});
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const headerTitle = window.i18n('allMedia');
|
2021-09-29 20:18:27 +00:00
|
|
|
|
|
|
|
|
|
const update = async () => {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const props = await getProps();
|
|
|
|
|
view.update(<MediaGallery i18n={window.i18n} {...props} />);
|
2021-09-29 20:18:27 +00:00
|
|
|
|
};
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view, headerTitle });
|
2021-09-29 20:18:27 +00:00
|
|
|
|
|
|
|
|
|
update();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showGV1Members(): void {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const { contactCollection, id } = this.model;
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-06-17 21:15:09 +00:00
|
|
|
|
const memberships =
|
|
|
|
|
contactCollection?.map((conversation: ConversationModel) => {
|
|
|
|
|
return {
|
|
|
|
|
isAdmin: false,
|
|
|
|
|
member: conversation.format(),
|
|
|
|
|
};
|
|
|
|
|
}) || [];
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const reduxState = window.reduxStore.getState();
|
|
|
|
|
const getPreferredBadge = getPreferredBadgeSelector(reduxState);
|
|
|
|
|
const theme = getTheme(reduxState);
|
|
|
|
|
|
|
|
|
|
const view = new ReactWrapperView({
|
2021-06-17 21:15:09 +00:00
|
|
|
|
className: 'group-member-list panel',
|
2022-06-03 16:33:39 +00:00
|
|
|
|
JSX: (
|
|
|
|
|
<ConversationDetailsMembershipList
|
|
|
|
|
canAddNewMembers={false}
|
|
|
|
|
conversationId={id}
|
|
|
|
|
i18n={window.i18n}
|
|
|
|
|
getPreferredBadge={getPreferredBadge}
|
|
|
|
|
maxShownMemberCount={32}
|
|
|
|
|
memberships={memberships}
|
|
|
|
|
showContactModal={contactId => {
|
|
|
|
|
this.showContactModal(contactId);
|
|
|
|
|
}}
|
|
|
|
|
theme={theme}
|
|
|
|
|
/>
|
|
|
|
|
),
|
2020-09-24 20:57:54 +00:00
|
|
|
|
});
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view });
|
2021-06-17 21:15:09 +00:00
|
|
|
|
view.render();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showSafetyNumber(id?: string): void {
|
2021-06-08 14:59:38 +00:00
|
|
|
|
let conversation: undefined | ConversationModel;
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
if (!id && isDirectConversation(this.model.attributes)) {
|
|
|
|
|
conversation = this.model;
|
2020-09-24 20:57:54 +00:00
|
|
|
|
} else {
|
|
|
|
|
conversation = window.ConversationController.get(id);
|
|
|
|
|
}
|
|
|
|
|
if (conversation) {
|
2021-10-06 20:27:14 +00:00
|
|
|
|
window.reduxActions.globalModals.toggleSafetyNumberModal(
|
|
|
|
|
conversation.get('id')
|
|
|
|
|
);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
downloadAttachmentWrapper(
|
|
|
|
|
messageId: string,
|
|
|
|
|
providedAttachment?: AttachmentType
|
|
|
|
|
): void {
|
2021-06-16 00:44:14 +00:00
|
|
|
|
const message = window.MessageController.getById(messageId);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (!message) {
|
|
|
|
|
throw new Error(
|
2021-06-16 00:44:14 +00:00
|
|
|
|
`downloadAttachmentWrapper: Message ${messageId} missing!`
|
2020-09-24 20:57:54 +00:00
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const { attachments, sent_at: timestamp } = message.attributes;
|
|
|
|
|
if (!attachments || attachments.length < 1) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
const attachment =
|
|
|
|
|
providedAttachment && attachments.includes(providedAttachment)
|
|
|
|
|
? providedAttachment
|
|
|
|
|
: attachments[0];
|
2020-09-24 20:57:54 +00:00
|
|
|
|
const { fileName } = attachment;
|
|
|
|
|
|
|
|
|
|
const isDangerous = window.Signal.Util.isFileDangerous(fileName || '');
|
|
|
|
|
|
|
|
|
|
this.downloadAttachment({ attachment, timestamp, isDangerous });
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-06-08 14:59:38 +00:00
|
|
|
|
async downloadAttachment({
|
|
|
|
|
attachment,
|
|
|
|
|
timestamp,
|
|
|
|
|
isDangerous,
|
|
|
|
|
}: {
|
2021-07-14 23:39:52 +00:00
|
|
|
|
attachment: AttachmentType;
|
|
|
|
|
timestamp: number;
|
2021-06-08 14:59:38 +00:00
|
|
|
|
isDangerous: boolean;
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}): Promise<void> {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (isDangerous) {
|
2021-09-22 20:59:54 +00:00
|
|
|
|
showToast(ToastDangerousFileType);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-01 00:52:03 +00:00
|
|
|
|
return saveAttachment(attachment, timestamp);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
async displayTapToViewMessage(messageId: string): Promise<void> {
|
2021-09-17 18:27:53 +00:00
|
|
|
|
log.info('displayTapToViewMessage: attempting to display message');
|
2021-08-12 16:20:22 +00:00
|
|
|
|
|
2021-06-16 00:44:14 +00:00
|
|
|
|
const message = window.MessageController.getById(messageId);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (!message) {
|
2021-06-16 00:44:14 +00:00
|
|
|
|
throw new Error(`displayTapToViewMessage: Message ${messageId} missing!`);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-06-17 17:15:10 +00:00
|
|
|
|
if (!isTapToView(message.attributes)) {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
throw new Error(
|
|
|
|
|
`displayTapToViewMessage: Message ${message.idForLogging()} is not a tap to view message`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (message.isErased()) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`displayTapToViewMessage: Message ${message.idForLogging()} is already erased`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-17 17:15:10 +00:00
|
|
|
|
const firstAttachment = (message.get('attachments') || [])[0];
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (!firstAttachment || !firstAttachment.path) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`displayTapToViewMessage: Message ${message.idForLogging()} had no first attachment with path`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const absolutePath = getAbsoluteAttachmentPath(firstAttachment.path);
|
2021-10-07 17:08:55 +00:00
|
|
|
|
const { path: tempPath } = await copyIntoTempDirectory(absolutePath);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
const tempAttachment = {
|
|
|
|
|
...firstAttachment,
|
|
|
|
|
path: tempPath,
|
|
|
|
|
};
|
|
|
|
|
|
2021-07-22 17:07:53 +00:00
|
|
|
|
await message.markViewOnceMessageViewed();
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2022-06-23 00:21:38 +00:00
|
|
|
|
const close = (): void => {
|
|
|
|
|
try {
|
|
|
|
|
this.stopListening(message);
|
|
|
|
|
closeLightbox();
|
|
|
|
|
} finally {
|
|
|
|
|
deleteTempFile(tempPath);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
2022-06-23 00:21:38 +00:00
|
|
|
|
};
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2022-06-23 00:21:38 +00:00
|
|
|
|
this.listenTo(message, 'expired', close);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.listenTo(message, 'change', () => {
|
2022-06-16 19:12:50 +00:00
|
|
|
|
showLightbox(getProps());
|
2020-09-24 20:57:54 +00:00
|
|
|
|
});
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const getProps = (): ComponentProps<typeof Lightbox> => {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
const { path, contentType } = tempAttachment;
|
|
|
|
|
|
|
|
|
|
return {
|
2022-06-23 00:21:38 +00:00
|
|
|
|
close,
|
2022-06-03 16:33:39 +00:00
|
|
|
|
i18n: window.i18n,
|
2021-08-24 21:47:14 +00:00
|
|
|
|
media: [
|
|
|
|
|
{
|
|
|
|
|
attachment: tempAttachment,
|
|
|
|
|
objectURL: getAbsoluteTempPath(path),
|
|
|
|
|
contentType,
|
|
|
|
|
index: 0,
|
|
|
|
|
message: {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
attachments: message.get('attachments') || [],
|
2021-08-24 21:47:14 +00:00
|
|
|
|
id: message.get('id'),
|
|
|
|
|
conversationId: message.get('conversationId'),
|
|
|
|
|
received_at: message.get('received_at'),
|
2021-08-30 21:32:56 +00:00
|
|
|
|
received_at_ms: Number(message.get('received_at_ms')),
|
2021-08-25 21:08:32 +00:00
|
|
|
|
sent_at: message.get('sent_at'),
|
2021-08-24 21:47:14 +00:00
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
],
|
2020-09-24 20:57:54 +00:00
|
|
|
|
isViewOnce: true,
|
|
|
|
|
};
|
|
|
|
|
};
|
2021-08-24 21:47:14 +00:00
|
|
|
|
|
2022-06-16 19:12:50 +00:00
|
|
|
|
showLightbox(getProps());
|
2021-08-12 16:20:22 +00:00
|
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
|
log.info('displayTapToViewMessage: showed lightbox');
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
deleteMessage(messageId: string): void {
|
2021-06-16 00:44:14 +00:00
|
|
|
|
const message = window.MessageController.getById(messageId);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (!message) {
|
2021-06-16 00:44:14 +00:00
|
|
|
|
throw new Error(`deleteMessage: Message ${messageId} missing!`);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-01-04 18:47:14 +00:00
|
|
|
|
window.showConfirmationDialog({
|
2022-09-27 20:24:21 +00:00
|
|
|
|
dialogName: 'deleteMessage',
|
2021-01-04 18:47:14 +00:00
|
|
|
|
confirmStyle: 'negative',
|
2020-09-24 20:57:54 +00:00
|
|
|
|
message: window.i18n('deleteWarning'),
|
|
|
|
|
okText: window.i18n('delete'),
|
|
|
|
|
resolve: () => {
|
2021-12-10 22:51:54 +00:00
|
|
|
|
window.Signal.Data.removeMessage(message.id);
|
2021-06-17 17:15:10 +00:00
|
|
|
|
if (isOutgoing(message.attributes)) {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.model.decrementSentMessageCount();
|
|
|
|
|
} else {
|
|
|
|
|
this.model.decrementMessageCount();
|
|
|
|
|
}
|
|
|
|
|
this.resetPanel();
|
|
|
|
|
},
|
|
|
|
|
});
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showStickerPackPreview(packId: string, packKey: string): void {
|
2021-07-09 19:36:10 +00:00
|
|
|
|
Stickers.downloadEphemeralPack(packId, packKey);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
|
|
|
|
const props = {
|
|
|
|
|
packId,
|
|
|
|
|
onClose: async () => {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
if (this.stickerPreviewModalView) {
|
|
|
|
|
this.stickerPreviewModalView.remove();
|
|
|
|
|
this.stickerPreviewModalView = undefined;
|
|
|
|
|
}
|
2021-07-09 19:36:10 +00:00
|
|
|
|
await Stickers.removeEphemeralPack(packId);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.stickerPreviewModalView = new ReactWrapperView({
|
2020-09-24 20:57:54 +00:00
|
|
|
|
className: 'sticker-preview-modal-wrapper',
|
|
|
|
|
JSX: window.Signal.State.Roots.createStickerPreviewModal(
|
|
|
|
|
window.reduxStore,
|
|
|
|
|
props
|
|
|
|
|
),
|
|
|
|
|
});
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-16 21:35:54 +00:00
|
|
|
|
showLightboxForMedia(
|
2021-08-23 23:14:53 +00:00
|
|
|
|
selectedMediaItem: MediaItemType,
|
2021-08-24 21:47:14 +00:00
|
|
|
|
media: Array<MediaItemType> = []
|
2021-08-30 21:32:56 +00:00
|
|
|
|
): void {
|
2021-08-25 21:08:32 +00:00
|
|
|
|
const onSave = async ({
|
|
|
|
|
attachment,
|
|
|
|
|
message,
|
|
|
|
|
index,
|
|
|
|
|
}: {
|
|
|
|
|
attachment: AttachmentType;
|
|
|
|
|
message: MediaItemMessageType;
|
|
|
|
|
index: number;
|
|
|
|
|
}) => {
|
2022-07-01 00:52:03 +00:00
|
|
|
|
return saveAttachment(attachment, message.sent_at, index + 1);
|
2021-01-29 21:19:24 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const selectedIndex = media.findIndex(
|
|
|
|
|
mediaItem =>
|
|
|
|
|
mediaItem.attachment.path === selectedMediaItem.attachment.path
|
|
|
|
|
);
|
|
|
|
|
|
2022-10-25 00:56:51 +00:00
|
|
|
|
const mediaMessage = selectedMediaItem.message;
|
|
|
|
|
const message = window.MessageController.getById(mediaMessage.id);
|
|
|
|
|
if (!message) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`showLightboxForMedia: Message ${mediaMessage.id} missing!`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const close = () => {
|
|
|
|
|
closeLightbox();
|
|
|
|
|
this.stopListening(message, 'expired', closeLightbox);
|
|
|
|
|
};
|
|
|
|
|
|
2022-06-16 19:12:50 +00:00
|
|
|
|
showLightbox({
|
2022-10-25 00:56:51 +00:00
|
|
|
|
close,
|
2022-06-16 19:12:50 +00:00
|
|
|
|
i18n: window.i18n,
|
|
|
|
|
getConversation: getConversationSelector(window.reduxStore.getState()),
|
|
|
|
|
media,
|
|
|
|
|
onForward: messageId => {
|
2022-12-09 00:49:54 +00:00
|
|
|
|
window.reduxActions.globalModals.toggleForwardMessageModal(messageId);
|
2022-06-16 19:12:50 +00:00
|
|
|
|
},
|
|
|
|
|
onSave,
|
|
|
|
|
selectedIndex: selectedIndex >= 0 ? selectedIndex : 0,
|
2021-01-29 21:19:24 +00:00
|
|
|
|
});
|
2022-10-25 00:56:51 +00:00
|
|
|
|
|
|
|
|
|
this.listenTo(message, 'expired', close);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2021-01-29 21:19:24 +00:00
|
|
|
|
|
|
|
|
|
showLightbox({
|
|
|
|
|
attachment,
|
|
|
|
|
messageId,
|
|
|
|
|
}: {
|
2021-07-14 23:39:52 +00:00
|
|
|
|
attachment: AttachmentType;
|
2021-01-29 21:19:24 +00:00
|
|
|
|
messageId: string;
|
|
|
|
|
showSingle?: boolean;
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}): void {
|
2021-06-16 00:44:14 +00:00
|
|
|
|
const message = window.MessageController.getById(messageId);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (!message) {
|
2021-06-16 00:44:14 +00:00
|
|
|
|
throw new Error(`showLightbox: Message ${messageId} missing!`);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
const sticker = message.get('sticker');
|
|
|
|
|
if (sticker) {
|
|
|
|
|
const { packId, packKey } = sticker;
|
|
|
|
|
this.showStickerPackPreview(packId, packKey);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-23 23:14:53 +00:00
|
|
|
|
const { contentType } = attachment;
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
!window.Signal.Util.GoogleChrome.isImageTypeSupported(contentType) &&
|
|
|
|
|
!window.Signal.Util.GoogleChrome.isVideoTypeSupported(contentType)
|
|
|
|
|
) {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.downloadAttachmentWrapper(messageId, attachment);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-14 23:39:52 +00:00
|
|
|
|
const attachments: Array<AttachmentType> = message.get('attachments') || [];
|
|
|
|
|
|
|
|
|
|
const loop = isGIF(attachments);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
|
|
|
|
const media = attachments
|
2021-07-14 23:39:52 +00:00
|
|
|
|
.filter(item => item.thumbnail && !item.pending && !item.error)
|
|
|
|
|
.map((item, index) => ({
|
|
|
|
|
objectURL: getAbsoluteAttachmentPath(item.path ?? ''),
|
2020-09-24 20:57:54 +00:00
|
|
|
|
path: item.path,
|
|
|
|
|
contentType: item.contentType,
|
2021-07-14 23:39:52 +00:00
|
|
|
|
loop,
|
2020-09-24 20:57:54 +00:00
|
|
|
|
index,
|
2021-08-23 23:14:53 +00:00
|
|
|
|
message: {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
attachments: message.get('attachments') || [],
|
2021-08-23 23:14:53 +00:00
|
|
|
|
id: message.get('id'),
|
2021-08-24 21:47:14 +00:00
|
|
|
|
conversationId:
|
2022-08-09 21:39:00 +00:00
|
|
|
|
window.ConversationController.lookupOrCreate({
|
|
|
|
|
uuid: message.get('sourceUuid'),
|
|
|
|
|
e164: message.get('source'),
|
2022-12-03 01:05:27 +00:00
|
|
|
|
reason: 'conversation_view.showLightBox',
|
2022-08-09 21:39:00 +00:00
|
|
|
|
})?.id || message.get('conversationId'),
|
2021-08-23 23:14:53 +00:00
|
|
|
|
received_at: message.get('received_at'),
|
2021-08-30 21:32:56 +00:00
|
|
|
|
received_at_ms: Number(message.get('received_at_ms')),
|
2021-08-25 21:08:32 +00:00
|
|
|
|
sent_at: message.get('sent_at'),
|
2021-08-23 23:14:53 +00:00
|
|
|
|
},
|
2020-09-24 20:57:54 +00:00
|
|
|
|
attachment: item,
|
2021-08-23 23:14:53 +00:00
|
|
|
|
thumbnailObjectUrl:
|
|
|
|
|
item.thumbnail?.objectUrl ||
|
|
|
|
|
getAbsoluteAttachmentPath(item.thumbnail?.path ?? ''),
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}));
|
|
|
|
|
|
2022-04-29 18:48:26 +00:00
|
|
|
|
if (!media.length) {
|
|
|
|
|
log.error(
|
|
|
|
|
'showLightbox: unable to load attachment',
|
|
|
|
|
attachments.map(x => ({
|
|
|
|
|
contentType: x.contentType,
|
|
|
|
|
error: x.error,
|
|
|
|
|
flags: x.flags,
|
|
|
|
|
path: x.path,
|
|
|
|
|
size: x.size,
|
|
|
|
|
}))
|
|
|
|
|
);
|
|
|
|
|
showToast(ToastUnableToLoadAttachment);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-23 23:14:53 +00:00
|
|
|
|
const selectedMedia =
|
|
|
|
|
media.find(item => attachment.path === item.path) || media[0];
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-24 21:47:14 +00:00
|
|
|
|
this.showLightboxForMedia(selectedMedia, media);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2021-06-08 14:59:38 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showContactModal(contactId: string): void {
|
2021-09-21 22:37:10 +00:00
|
|
|
|
window.reduxActions.globalModals.showContactModal(contactId, this.model.id);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2021-06-08 14:59:38 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showGroupLinkManagement(): void {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const view = new ReactWrapperView({
|
2021-01-29 21:19:24 +00:00
|
|
|
|
className: 'panel',
|
|
|
|
|
JSX: window.Signal.State.Roots.createGroupLinkManagement(
|
|
|
|
|
window.reduxStore,
|
|
|
|
|
{
|
2021-08-30 21:32:56 +00:00
|
|
|
|
conversationId: this.model.id,
|
2021-01-29 21:19:24 +00:00
|
|
|
|
}
|
|
|
|
|
),
|
|
|
|
|
});
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const headerTitle = window.i18n('ConversationDetails--group-link');
|
2021-01-29 21:19:24 +00:00
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view, headerTitle });
|
2021-01-29 21:19:24 +00:00
|
|
|
|
view.render();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2021-06-08 14:59:38 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showGroupV2Permissions(): void {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const view = new ReactWrapperView({
|
2021-01-29 21:19:24 +00:00
|
|
|
|
className: 'panel',
|
|
|
|
|
JSX: window.Signal.State.Roots.createGroupV2Permissions(
|
|
|
|
|
window.reduxStore,
|
|
|
|
|
{
|
2021-08-30 21:32:56 +00:00
|
|
|
|
conversationId: this.model.id,
|
2021-01-29 21:19:24 +00:00
|
|
|
|
}
|
|
|
|
|
),
|
|
|
|
|
});
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const headerTitle = window.i18n('permissions');
|
2021-01-29 21:19:24 +00:00
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view, headerTitle });
|
2021-01-29 21:19:24 +00:00
|
|
|
|
view.render();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2021-06-08 14:59:38 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showPendingInvites(): void {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const view = new ReactWrapperView({
|
2021-01-29 21:19:24 +00:00
|
|
|
|
className: 'panel',
|
|
|
|
|
JSX: window.Signal.State.Roots.createPendingInvites(window.reduxStore, {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
conversationId: this.model.id,
|
2021-11-10 23:01:06 +00:00
|
|
|
|
ourUuid: window.textsecure.storage.user.getCheckedUuid().toString(),
|
2021-01-29 21:19:24 +00:00
|
|
|
|
}),
|
|
|
|
|
});
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const headerTitle = window.i18n(
|
|
|
|
|
'ConversationDetails--requests-and-invites'
|
|
|
|
|
);
|
2021-01-29 21:19:24 +00:00
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view, headerTitle });
|
2021-01-29 21:19:24 +00:00
|
|
|
|
view.render();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2021-08-05 12:35:33 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showConversationNotificationsSettings(): void {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const view = new ReactWrapperView({
|
2021-08-05 12:35:33 +00:00
|
|
|
|
className: 'panel',
|
|
|
|
|
JSX: window.Signal.State.Roots.createConversationNotificationsSettings(
|
|
|
|
|
window.reduxStore,
|
|
|
|
|
{
|
2021-08-30 21:32:56 +00:00
|
|
|
|
conversationId: this.model.id,
|
2021-08-05 12:35:33 +00:00
|
|
|
|
}
|
|
|
|
|
),
|
|
|
|
|
});
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const headerTitle = window.i18n('ConversationDetails--notifications');
|
2021-08-05 12:35:33 +00:00
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view, headerTitle });
|
2021-08-05 12:35:33 +00:00
|
|
|
|
view.render();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2021-05-28 16:15:17 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showChatColorEditor(): void {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const view = new ReactWrapperView({
|
2021-05-28 16:15:17 +00:00
|
|
|
|
className: 'panel',
|
|
|
|
|
JSX: window.Signal.State.Roots.createChatColorPicker(window.reduxStore, {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
conversationId: this.model.get('id'),
|
2021-05-28 16:15:17 +00:00
|
|
|
|
}),
|
|
|
|
|
});
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const headerTitle = window.i18n('ChatColorPicker__menu-title');
|
2021-05-28 16:15:17 +00:00
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view, headerTitle });
|
2021-05-28 16:15:17 +00:00
|
|
|
|
view.render();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2021-05-28 16:15:17 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showConversationDetails(): void {
|
2021-07-20 20:18:35 +00:00
|
|
|
|
// Run a getProfiles in case member's capabilities have changed
|
|
|
|
|
// Redux should cover us on the return here so no need to await this.
|
2021-08-30 21:32:56 +00:00
|
|
|
|
if (this.model.throttledGetProfiles) {
|
|
|
|
|
this.model.throttledGetProfiles();
|
|
|
|
|
}
|
2021-01-29 21:19:24 +00:00
|
|
|
|
|
|
|
|
|
// these methods are used in more than one place and should probably be
|
|
|
|
|
// dried up and hoisted to methods on ConversationView
|
|
|
|
|
|
2021-04-28 20:27:16 +00:00
|
|
|
|
const onLeave = () => {
|
2022-12-06 17:31:44 +00:00
|
|
|
|
longRunningTaskWrapper({
|
|
|
|
|
idForLogging: this.model.idForLogging(),
|
2021-04-28 20:27:16 +00:00
|
|
|
|
name: 'onLeave',
|
2021-08-30 21:32:56 +00:00
|
|
|
|
task: () => this.model.leaveGroupV2(),
|
2021-01-29 21:19:24 +00:00
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const props = {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
addMembers: this.model.addMembersV2.bind(this.model),
|
|
|
|
|
conversationId: this.model.get('id'),
|
2021-01-29 21:19:24 +00:00
|
|
|
|
showAllMedia: this.showAllMedia.bind(this),
|
|
|
|
|
showContactModal: this.showContactModal.bind(this),
|
2021-10-20 23:46:41 +00:00
|
|
|
|
showChatColorEditor: this.showChatColorEditor.bind(this),
|
2021-01-29 21:19:24 +00:00
|
|
|
|
showGroupLinkManagement: this.showGroupLinkManagement.bind(this),
|
|
|
|
|
showGroupV2Permissions: this.showGroupV2Permissions.bind(this),
|
2021-11-11 22:43:05 +00:00
|
|
|
|
showConversationNotificationsSettings:
|
|
|
|
|
this.showConversationNotificationsSettings.bind(this),
|
2021-01-29 21:19:24 +00:00
|
|
|
|
showPendingInvites: this.showPendingInvites.bind(this),
|
|
|
|
|
showLightboxForMedia: this.showLightboxForMedia.bind(this),
|
2021-08-30 21:32:56 +00:00
|
|
|
|
updateGroupAttributes: this.model.updateGroupAttributesV2.bind(
|
|
|
|
|
this.model
|
|
|
|
|
),
|
2021-04-28 20:27:16 +00:00
|
|
|
|
onLeave,
|
2021-01-29 21:19:24 +00:00
|
|
|
|
};
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const view = new ReactWrapperView({
|
2021-01-29 21:19:24 +00:00
|
|
|
|
className: 'conversation-details-pane panel',
|
|
|
|
|
JSX: window.Signal.State.Roots.createConversationDetails(
|
|
|
|
|
window.reduxStore,
|
|
|
|
|
props
|
|
|
|
|
),
|
|
|
|
|
});
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const headerTitle = '';
|
2021-01-29 21:19:24 +00:00
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view, headerTitle });
|
2021-01-29 21:19:24 +00:00
|
|
|
|
view.render();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2021-01-29 21:19:24 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showMessageDetail(messageId: string): void {
|
2021-06-16 00:44:14 +00:00
|
|
|
|
const message = window.MessageController.getById(messageId);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (!message) {
|
2021-06-16 00:44:14 +00:00
|
|
|
|
throw new Error(`showMessageDetail: Message ${messageId} missing!`);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!message.isNormalBubble()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-24 22:06:12 +00:00
|
|
|
|
const getProps = () => ({
|
2021-07-19 22:44:49 +00:00
|
|
|
|
...message.getPropsForMessageDetail(
|
|
|
|
|
window.ConversationController.getOurConversationIdOrThrow()
|
|
|
|
|
),
|
2021-03-24 22:06:12 +00:00
|
|
|
|
...this.getMessageActions(),
|
|
|
|
|
});
|
|
|
|
|
|
2020-09-24 20:57:54 +00:00
|
|
|
|
const onClose = () => {
|
|
|
|
|
this.stopListening(message, 'change', update);
|
|
|
|
|
this.resetPanel();
|
|
|
|
|
};
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const view = new ReactWrapperView({
|
2020-09-24 20:57:54 +00:00
|
|
|
|
className: 'panel message-detail-wrapper',
|
2021-03-24 22:06:12 +00:00
|
|
|
|
JSX: window.Signal.State.Roots.createMessageDetail(
|
|
|
|
|
window.reduxStore,
|
|
|
|
|
getProps()
|
|
|
|
|
),
|
2020-09-24 20:57:54 +00:00
|
|
|
|
onClose,
|
|
|
|
|
});
|
|
|
|
|
|
2021-09-29 20:18:27 +00:00
|
|
|
|
const update = () =>
|
|
|
|
|
view.update(
|
|
|
|
|
window.Signal.State.Roots.createMessageDetail(
|
|
|
|
|
window.reduxStore,
|
|
|
|
|
getProps()
|
|
|
|
|
)
|
|
|
|
|
);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.listenTo(message, 'change', update);
|
|
|
|
|
this.listenTo(message, 'expired', onClose);
|
|
|
|
|
// We could listen to all involved contacts, but we'll call that overkill
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view });
|
2020-09-24 20:57:54 +00:00
|
|
|
|
view.render();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
showStickerManager(): void {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const view = new ReactWrapperView({
|
2020-09-24 20:57:54 +00:00
|
|
|
|
className: ['sticker-manager-wrapper', 'panel'].join(' '),
|
|
|
|
|
JSX: window.Signal.State.Roots.createStickerManager(window.reduxStore),
|
|
|
|
|
onClose: () => {
|
|
|
|
|
this.resetPanel();
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view });
|
2020-09-24 20:57:54 +00:00
|
|
|
|
view.render();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-06-08 14:59:38 +00:00
|
|
|
|
showContactDetail({
|
|
|
|
|
contact,
|
|
|
|
|
signalAccount,
|
|
|
|
|
}: {
|
2021-08-20 01:56:39 +00:00
|
|
|
|
contact: EmbeddedContactType;
|
2022-04-12 00:26:09 +00:00
|
|
|
|
signalAccount?: {
|
|
|
|
|
phoneNumber: string;
|
|
|
|
|
uuid: UUIDStringType;
|
|
|
|
|
};
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}): void {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const view = new ReactWrapperView({
|
2020-09-24 20:57:54 +00:00
|
|
|
|
className: 'contact-detail-pane panel',
|
2022-06-03 16:33:39 +00:00
|
|
|
|
JSX: (
|
|
|
|
|
<ContactDetail
|
|
|
|
|
i18n={window.i18n}
|
|
|
|
|
contact={contact}
|
|
|
|
|
hasSignalAccount={Boolean(signalAccount)}
|
|
|
|
|
onSendMessage={() => {
|
|
|
|
|
if (signalAccount) {
|
2022-12-05 22:56:23 +00:00
|
|
|
|
startConversation(signalAccount.phoneNumber, signalAccount.uuid);
|
2022-06-03 16:33:39 +00:00
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
),
|
2020-09-24 20:57:54 +00:00
|
|
|
|
onClose: () => {
|
|
|
|
|
this.resetPanel();
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.addPanel({ view });
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
|
async openConversation(
|
|
|
|
|
conversationId: string,
|
|
|
|
|
messageId?: string
|
|
|
|
|
): Promise<void> {
|
|
|
|
|
window.Whisper.events.trigger(
|
|
|
|
|
'showConversation',
|
|
|
|
|
conversationId,
|
|
|
|
|
messageId
|
|
|
|
|
);
|
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
addPanel(panel: PanelType): void {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
this.panels = this.panels || [];
|
|
|
|
|
|
|
|
|
|
if (this.panels.length === 0) {
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.previousFocus = document.activeElement as HTMLElement;
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.panels.unshift(panel);
|
|
|
|
|
panel.view.$el.insertAfter(this.$('.panel').last());
|
|
|
|
|
panel.view.$el.one('animationend', () => {
|
|
|
|
|
panel.view.$el.addClass('panel--static');
|
2020-09-24 20:57:54 +00:00
|
|
|
|
});
|
2020-10-30 17:52:21 +00:00
|
|
|
|
|
|
|
|
|
window.reduxActions.conversations.setSelectedConversationPanelDepth(
|
|
|
|
|
this.panels.length
|
|
|
|
|
);
|
2021-01-29 21:19:24 +00:00
|
|
|
|
window.reduxActions.conversations.setSelectedConversationHeaderTitle(
|
2022-06-03 16:33:39 +00:00
|
|
|
|
panel.headerTitle
|
2021-01-29 21:19:24 +00:00
|
|
|
|
);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
|
|
|
|
resetPanel(): void {
|
2020-09-24 20:57:54 +00:00
|
|
|
|
if (!this.panels || !this.panels.length) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
const panel = this.panels.shift();
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
this.panels.length === 0 &&
|
|
|
|
|
this.previousFocus &&
|
|
|
|
|
this.previousFocus.focus
|
|
|
|
|
) {
|
|
|
|
|
this.previousFocus.focus();
|
2021-08-30 21:32:56 +00:00
|
|
|
|
this.previousFocus = undefined;
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (this.panels.length > 0) {
|
2022-06-03 16:33:39 +00:00
|
|
|
|
this.panels[0].view.$el.fadeIn(250);
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
if (panel) {
|
2022-07-05 16:30:55 +00:00
|
|
|
|
let timeout: ReturnType<typeof setTimeout> | undefined;
|
|
|
|
|
const removePanel = () => {
|
|
|
|
|
if (!timeout) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
clearTimeout(timeout);
|
|
|
|
|
timeout = undefined;
|
|
|
|
|
|
2022-06-03 16:33:39 +00:00
|
|
|
|
panel.view.remove();
|
2022-07-05 16:30:55 +00:00
|
|
|
|
};
|
|
|
|
|
panel.view.$el
|
|
|
|
|
.addClass('panel--remove')
|
|
|
|
|
.one('transitionend', removePanel);
|
|
|
|
|
|
|
|
|
|
// Backup, in case things go wrong with the transitionend event
|
|
|
|
|
timeout = setTimeout(removePanel, SECOND);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-10-30 17:52:21 +00:00
|
|
|
|
|
|
|
|
|
window.reduxActions.conversations.setSelectedConversationPanelDepth(
|
|
|
|
|
this.panels.length
|
|
|
|
|
);
|
2021-01-29 21:19:24 +00:00
|
|
|
|
window.reduxActions.conversations.setSelectedConversationHeaderTitle(
|
|
|
|
|
this.panels[0]?.headerTitle
|
|
|
|
|
);
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
2020-09-24 20:57:54 +00:00
|
|
|
|
|
2022-12-08 01:26:59 +00:00
|
|
|
|
async setQuoteMessage(messageId: string | undefined): Promise<void> {
|
|
|
|
|
const { model } = this;
|
|
|
|
|
const message = messageId ? await getMessageById(messageId) : undefined;
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
message &&
|
|
|
|
|
!canReply(
|
|
|
|
|
message.attributes,
|
|
|
|
|
window.ConversationController.getOurConversationIdOrThrow(),
|
|
|
|
|
findAndFormatContact
|
|
|
|
|
)
|
|
|
|
|
) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (message && !message.isNormalBubble()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const existing = model.get('quotedMessageId');
|
|
|
|
|
if (existing !== messageId) {
|
|
|
|
|
const now = Date.now();
|
|
|
|
|
let active_at = this.model.get('active_at');
|
|
|
|
|
let timestamp = this.model.get('timestamp');
|
|
|
|
|
|
|
|
|
|
if (!active_at && messageId) {
|
|
|
|
|
active_at = now;
|
|
|
|
|
timestamp = now;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.model.set({
|
|
|
|
|
active_at,
|
|
|
|
|
draftChanged: true,
|
|
|
|
|
quotedMessageId: messageId,
|
|
|
|
|
timestamp,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await this.saveModel();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (message) {
|
|
|
|
|
const quote = await model.makeQuote(message);
|
|
|
|
|
window.reduxActions.composer.setQuotedMessage({
|
|
|
|
|
conversationId: model.id,
|
|
|
|
|
quote,
|
|
|
|
|
});
|
|
|
|
|
|
2022-12-08 23:56:17 +00:00
|
|
|
|
window.reduxActions.composer.setComposerFocus(this.model.id);
|
2022-12-08 07:43:48 +00:00
|
|
|
|
window.reduxActions.composer.setComposerDisabledState(false);
|
2022-12-08 01:26:59 +00:00
|
|
|
|
} else {
|
|
|
|
|
window.reduxActions.composer.setQuotedMessage(undefined);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async clearAttachments(): Promise<void> {
|
|
|
|
|
const draftAttachments = this.model.get('draftAttachments') || [];
|
|
|
|
|
this.model.set({
|
|
|
|
|
draftAttachments: [],
|
|
|
|
|
draftChanged: true,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
this.updateAttachmentsView();
|
|
|
|
|
|
|
|
|
|
// We're fine doing this all at once; at most it should be 32 attachments
|
|
|
|
|
await Promise.all([
|
|
|
|
|
this.saveModel(),
|
|
|
|
|
Promise.all(
|
|
|
|
|
draftAttachments.map(attachment => deleteDraftAttachment(attachment))
|
|
|
|
|
),
|
|
|
|
|
]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
updateAttachmentsView(): void {
|
|
|
|
|
const draftAttachments = this.model.get('draftAttachments') || [];
|
|
|
|
|
window.reduxActions.composer.replaceAttachments(
|
|
|
|
|
this.model.get('id'),
|
|
|
|
|
draftAttachments
|
|
|
|
|
);
|
2022-12-08 23:56:17 +00:00
|
|
|
|
if (hasDraftAttachments(this.model.attributes, { includePending: true })) {
|
2022-12-08 01:26:59 +00:00
|
|
|
|
removeLinkPreview();
|
2020-09-24 20:57:54 +00:00
|
|
|
|
}
|
2021-08-30 21:32:56 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
window.Whisper.ConversationView = ConversationView;
|