mirror of https://github.com/oxen-io/session-ios
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2607 lines
115 KiB
Swift
2607 lines
115 KiB
Swift
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import UIKit
|
|
import Combine
|
|
import CoreServices
|
|
import Photos
|
|
import PhotosUI
|
|
import Sodium
|
|
import GRDB
|
|
import SessionUIKit
|
|
import SessionMessagingKit
|
|
import SessionUtilitiesKit
|
|
import SignalUtilitiesKit
|
|
|
|
extension ConversationVC:
|
|
InputViewDelegate,
|
|
MessageCellDelegate,
|
|
ContextMenuActionDelegate,
|
|
ScrollToBottomButtonDelegate,
|
|
SendMediaNavDelegate,
|
|
UIDocumentPickerDelegate,
|
|
AttachmentApprovalViewControllerDelegate,
|
|
GifPickerViewControllerDelegate
|
|
{
|
|
@objc func handleTitleViewTapped() {
|
|
// Don't take the user to settings for unapproved threads
|
|
guard viewModel.threadData.threadRequiresApproval == false else { return }
|
|
|
|
openSettings()
|
|
}
|
|
|
|
@objc func openSettings() {
|
|
let viewController: SessionTableViewController = SessionTableViewController(
|
|
viewModel: ThreadSettingsViewModel(
|
|
threadId: self.viewModel.threadData.threadId,
|
|
threadVariant: self.viewModel.threadData.threadVariant,
|
|
didTriggerSearch: { [weak self] in
|
|
DispatchQueue.main.async {
|
|
self?.showSearchUI()
|
|
self?.popAllConversationSettingsViews {
|
|
// Note: Without this delay the search bar doesn't show
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) {
|
|
self?.searchController.uiSearchController.searchBar.becomeFirstResponder()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
)
|
|
)
|
|
|
|
navigationController?.pushViewController(viewController, animated: true)
|
|
}
|
|
|
|
// MARK: - ScrollToBottomButtonDelegate
|
|
|
|
func handleScrollToBottomButtonTapped() {
|
|
// The table view's content size is calculated by the estimated height of cells,
|
|
// so the result may be inaccurate before all the cells are loaded. Use this
|
|
// to scroll to the last row instead.
|
|
scrollToBottom(isAnimated: true)
|
|
}
|
|
|
|
// MARK: - Call
|
|
|
|
@objc func startCall(_ sender: Any?) {
|
|
guard SessionCall.isEnabled else { return }
|
|
guard viewModel.threadData.threadIsBlocked == false else { return }
|
|
guard Storage.shared[.areCallsEnabled] else {
|
|
let confirmationModal: ConfirmationModal = ConfirmationModal(
|
|
info: ConfirmationModal.Info(
|
|
title: "modal_call_permission_request_title".localized(),
|
|
explanation: "modal_call_permission_request_explanation".localized(),
|
|
confirmTitle: "vc_settings_title".localized(),
|
|
confirmAccessibility: Accessibility(identifier: "Settings"),
|
|
dismissOnConfirm: false // Custom dismissal logic
|
|
) { [weak self] _ in
|
|
self?.dismiss(animated: true) {
|
|
let navController: UINavigationController = StyledNavigationController(
|
|
rootViewController: SessionTableViewController(
|
|
viewModel: PrivacySettingsViewModel(
|
|
shouldShowCloseButton: true
|
|
)
|
|
)
|
|
)
|
|
navController.modalPresentationStyle = .fullScreen
|
|
self?.present(navController, animated: true, completion: nil)
|
|
}
|
|
}
|
|
)
|
|
|
|
self.navigationController?.present(confirmationModal, animated: true, completion: nil)
|
|
return
|
|
}
|
|
|
|
Permissions.requestMicrophonePermissionIfNeeded()
|
|
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
|
|
guard AVAudioSession.sharedInstance().recordPermission == .granted else { return }
|
|
guard self.viewModel.threadData.threadVariant == .contact else { return }
|
|
guard AppEnvironment.shared.callManager.currentCall == nil else { return }
|
|
guard let call: SessionCall = Storage.shared.read({ db in SessionCall(db, for: threadId, uuid: UUID().uuidString.lowercased(), mode: .offer, outgoing: true) }) else {
|
|
return
|
|
}
|
|
|
|
let callVC = CallVC(for: call)
|
|
callVC.conversationVC = self
|
|
hideInputAccessoryView()
|
|
|
|
present(callVC, animated: true, completion: nil)
|
|
}
|
|
|
|
// MARK: - Blocking
|
|
|
|
@objc func unblock() {
|
|
self.showBlockedModalIfNeeded()
|
|
}
|
|
|
|
@discardableResult func showBlockedModalIfNeeded() -> Bool {
|
|
guard
|
|
self.viewModel.threadData.threadVariant == .contact &&
|
|
self.viewModel.threadData.threadIsBlocked == true
|
|
else { return false }
|
|
|
|
let message = String(
|
|
format: "modal_blocked_explanation".localized(),
|
|
self.viewModel.threadData.displayName
|
|
)
|
|
let confirmationModal: ConfirmationModal = ConfirmationModal(
|
|
info: ConfirmationModal.Info(
|
|
title: String(
|
|
format: "modal_blocked_title".localized(),
|
|
self.viewModel.threadData.displayName
|
|
),
|
|
attributedExplanation: NSAttributedString(string: message)
|
|
.adding(
|
|
attributes: [ .font: UIFont.boldSystemFont(ofSize: Values.smallFontSize) ],
|
|
range: (message as NSString).range(of: self.viewModel.threadData.displayName)
|
|
),
|
|
confirmTitle: "modal_blocked_button_title".localized(),
|
|
confirmAccessibility: Accessibility(identifier: "Confirm block"),
|
|
cancelAccessibility: Accessibility(identifier: "Cancel block"),
|
|
dismissOnConfirm: false // Custom dismissal logic
|
|
) { [weak self] _ in
|
|
self?.viewModel.unblockContact()
|
|
self?.dismiss(animated: true, completion: nil)
|
|
}
|
|
)
|
|
present(confirmationModal, animated: true, completion: nil)
|
|
|
|
return true
|
|
}
|
|
|
|
// MARK: - SendMediaNavDelegate
|
|
|
|
func sendMediaNavDidCancel(_ sendMediaNavigationController: SendMediaNavigationController?) {
|
|
dismiss(animated: true, completion: nil)
|
|
}
|
|
|
|
func sendMediaNav(_ sendMediaNavigationController: SendMediaNavigationController, didApproveAttachments attachments: [SignalAttachment], forThreadId threadId: String, messageText: String?) {
|
|
sendAttachments(attachments, with: messageText ?? "")
|
|
self.snInputView.text = ""
|
|
resetMentions()
|
|
dismiss(animated: true) { }
|
|
}
|
|
|
|
func sendMediaNavInitialMessageText(_ sendMediaNavigationController: SendMediaNavigationController) -> String? {
|
|
return snInputView.text
|
|
}
|
|
|
|
func sendMediaNav(_ sendMediaNavigationController: SendMediaNavigationController, didChangeMessageText newMessageText: String?) {
|
|
snInputView.text = (newMessageText ?? "")
|
|
}
|
|
|
|
// MARK: - AttachmentApprovalViewControllerDelegate
|
|
|
|
func attachmentApproval(_ attachmentApproval: AttachmentApprovalViewController, didApproveAttachments attachments: [SignalAttachment], forThreadId threadId: String, messageText: String?) {
|
|
sendAttachments(attachments, with: messageText ?? "") { [weak self] in
|
|
self?.dismiss(animated: true, completion: nil)
|
|
}
|
|
|
|
scrollToBottom(isAnimated: false)
|
|
self.snInputView.text = ""
|
|
resetMentions()
|
|
}
|
|
|
|
func attachmentApprovalDidCancel(_ attachmentApproval: AttachmentApprovalViewController) {
|
|
dismiss(animated: true, completion: nil)
|
|
}
|
|
|
|
func attachmentApproval(_ attachmentApproval: AttachmentApprovalViewController, didChangeMessageText newMessageText: String?) {
|
|
snInputView.text = newMessageText ?? ""
|
|
}
|
|
|
|
func attachmentApproval(_ attachmentApproval: AttachmentApprovalViewController, didRemoveAttachment attachment: SignalAttachment) {
|
|
}
|
|
|
|
func attachmentApprovalDidTapAddMore(_ attachmentApproval: AttachmentApprovalViewController) {
|
|
}
|
|
|
|
// MARK: - ExpandingAttachmentsButtonDelegate
|
|
|
|
func handleGIFButtonTapped() {
|
|
guard Storage.shared[.isGiphyEnabled] else {
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
info: ConfirmationModal.Info(
|
|
title: "GIPHY_PERMISSION_TITLE".localized(),
|
|
explanation: "GIPHY_PERMISSION_MESSAGE".localized(),
|
|
confirmTitle: "continue_2".localized()
|
|
) { [weak self] _ in
|
|
Storage.shared.writeAsync(
|
|
updates: { db in
|
|
db[.isGiphyEnabled] = true
|
|
},
|
|
completion: { _, _ in
|
|
DispatchQueue.main.async {
|
|
self?.handleGIFButtonTapped()
|
|
}
|
|
}
|
|
)
|
|
}
|
|
)
|
|
|
|
present(modal, animated: true, completion: nil)
|
|
return
|
|
}
|
|
|
|
let gifVC = GifPickerViewController()
|
|
gifVC.delegate = self
|
|
|
|
let navController = StyledNavigationController(rootViewController: gifVC)
|
|
navController.modalPresentationStyle = .fullScreen
|
|
present(navController, animated: true) { }
|
|
}
|
|
|
|
func handleDocumentButtonTapped() {
|
|
// UIDocumentPickerModeImport copies to a temp file within our container.
|
|
// It uses more memory than "open" but lets us avoid working with security scoped URLs.
|
|
let documentPickerVC = UIDocumentPickerViewController(documentTypes: [ kUTTypeItem as String ], in: UIDocumentPickerMode.import)
|
|
documentPickerVC.delegate = self
|
|
documentPickerVC.modalPresentationStyle = .fullScreen
|
|
|
|
present(documentPickerVC, animated: true, completion: nil)
|
|
}
|
|
|
|
func handleLibraryButtonTapped() {
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
|
|
Permissions.requestLibraryPermissionIfNeeded { [weak self] in
|
|
DispatchQueue.main.async {
|
|
let sendMediaNavController = SendMediaNavigationController.showingMediaLibraryFirst(
|
|
threadId: threadId
|
|
)
|
|
sendMediaNavController.sendMediaNavDelegate = self
|
|
sendMediaNavController.modalPresentationStyle = .fullScreen
|
|
self?.present(sendMediaNavController, animated: true, completion: nil)
|
|
}
|
|
}
|
|
}
|
|
|
|
func handleCameraButtonTapped() {
|
|
guard Permissions.requestCameraPermissionIfNeeded(presentingViewController: self) else { return }
|
|
|
|
Permissions.requestMicrophonePermissionIfNeeded()
|
|
|
|
if AVAudioSession.sharedInstance().recordPermission != .granted {
|
|
SNLog("Proceeding without microphone access. Any recorded video will be silent.")
|
|
}
|
|
|
|
let sendMediaNavController = SendMediaNavigationController.showingCameraFirst(threadId: self.viewModel.threadData.threadId)
|
|
sendMediaNavController.sendMediaNavDelegate = self
|
|
sendMediaNavController.modalPresentationStyle = .fullScreen
|
|
|
|
present(sendMediaNavController, animated: true, completion: nil)
|
|
}
|
|
|
|
// MARK: - GifPickerViewControllerDelegate
|
|
|
|
func gifPickerDidSelect(attachment: SignalAttachment) {
|
|
showAttachmentApprovalDialog(for: [ attachment ])
|
|
}
|
|
|
|
// MARK: - UIDocumentPickerDelegate
|
|
|
|
func documentPicker(_ controller: UIDocumentPickerViewController, didPickDocumentsAt urls: [URL]) {
|
|
guard let url = urls.first else { return } // TODO: Handle multiple?
|
|
|
|
let urlResourceValues: URLResourceValues
|
|
do {
|
|
urlResourceValues = try url.resourceValues(forKeys: [ .typeIdentifierKey, .isDirectoryKey, .nameKey ])
|
|
}
|
|
catch {
|
|
DispatchQueue.main.async { [weak self] in
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
targetView: self?.view,
|
|
info: ConfirmationModal.Info(
|
|
title: "Session",
|
|
explanation: "An error occurred.",
|
|
cancelTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text
|
|
)
|
|
)
|
|
self?.present(modal, animated: true)
|
|
}
|
|
return
|
|
}
|
|
|
|
let type = urlResourceValues.typeIdentifier ?? (kUTTypeData as String)
|
|
guard urlResourceValues.isDirectory != true else {
|
|
DispatchQueue.main.async { [weak self] in
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
targetView: self?.view,
|
|
info: ConfirmationModal.Info(
|
|
title: "ATTACHMENT_PICKER_DOCUMENTS_PICKED_DIRECTORY_FAILED_ALERT_TITLE".localized(),
|
|
explanation: "ATTACHMENT_PICKER_DOCUMENTS_PICKED_DIRECTORY_FAILED_ALERT_BODY".localized(),
|
|
cancelTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text
|
|
)
|
|
)
|
|
self?.present(modal, animated: true)
|
|
}
|
|
return
|
|
}
|
|
|
|
let fileName = urlResourceValues.name ?? NSLocalizedString("ATTACHMENT_DEFAULT_FILENAME", comment: "")
|
|
guard let dataSource = DataSourcePath.dataSource(with: url, shouldDeleteOnDeallocation: false) else {
|
|
DispatchQueue.main.async { [weak self] in
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
targetView: self?.view,
|
|
info: ConfirmationModal.Info(
|
|
title: "ATTACHMENT_PICKER_DOCUMENTS_FAILED_ALERT_TITLE".localized(),
|
|
cancelTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text
|
|
)
|
|
)
|
|
self?.present(modal, animated: true)
|
|
}
|
|
return
|
|
}
|
|
dataSource.sourceFilename = fileName
|
|
|
|
// Although we want to be able to send higher quality attachments through the document picker
|
|
// it's more imporant that we ensure the sent format is one all clients can accept (e.g. *not* quicktime .mov)
|
|
guard !SignalAttachment.isInvalidVideo(dataSource: dataSource, dataUTI: type) else {
|
|
return showAttachmentApprovalDialogAfterProcessingVideo(at: url, with: fileName)
|
|
}
|
|
|
|
// "Document picker" attachments _SHOULD NOT_ be resized
|
|
let attachment = SignalAttachment.attachment(dataSource: dataSource, dataUTI: type, imageQuality: .original)
|
|
showAttachmentApprovalDialog(for: [ attachment ])
|
|
}
|
|
|
|
func showAttachmentApprovalDialog(for attachments: [SignalAttachment]) {
|
|
let navController = AttachmentApprovalViewController.wrappedInNavController(
|
|
threadId: self.viewModel.threadData.threadId,
|
|
attachments: attachments,
|
|
approvalDelegate: self
|
|
)
|
|
|
|
present(navController, animated: true, completion: nil)
|
|
}
|
|
|
|
func showAttachmentApprovalDialogAfterProcessingVideo(at url: URL, with fileName: String) {
|
|
ModalActivityIndicatorViewController.present(fromViewController: self, canCancel: true, message: nil) { [weak self] modalActivityIndicator in
|
|
let dataSource = DataSourcePath.dataSource(with: url, shouldDeleteOnDeallocation: false)!
|
|
dataSource.sourceFilename = fileName
|
|
|
|
SignalAttachment
|
|
.compressVideoAsMp4(
|
|
dataSource: dataSource,
|
|
dataUTI: kUTTypeMPEG4 as String
|
|
)
|
|
.attachmentPublisher
|
|
.sinkUntilComplete(
|
|
receiveValue: { [weak self] attachment in
|
|
guard !modalActivityIndicator.wasCancelled else { return }
|
|
|
|
modalActivityIndicator.dismiss {
|
|
guard !attachment.hasError else {
|
|
self?.showErrorAlert(for: attachment, onDismiss: nil)
|
|
return
|
|
}
|
|
|
|
self?.showAttachmentApprovalDialog(for: [ attachment ])
|
|
}
|
|
}
|
|
)
|
|
}
|
|
}
|
|
|
|
// MARK: - InputViewDelegate
|
|
|
|
// MARK: --Message Sending
|
|
|
|
func handleSendButtonTapped() {
|
|
sendMessage()
|
|
}
|
|
|
|
func sendMessage(hasPermissionToSendSeed: Bool = false) {
|
|
guard !showBlockedModalIfNeeded() else { return }
|
|
|
|
let text = replaceMentions(in: snInputView.text.trimmingCharacters(in: .whitespacesAndNewlines))
|
|
|
|
guard !text.isEmpty else { return }
|
|
|
|
if text.contains(mnemonic) && !viewModel.threadData.threadIsNoteToSelf && !hasPermissionToSendSeed {
|
|
// Warn the user if they're about to send their seed to someone
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
info: ConfirmationModal.Info(
|
|
title: "modal_send_seed_title".localized(),
|
|
explanation: "modal_send_seed_explanation".localized(),
|
|
confirmTitle: "modal_send_seed_send_button_title".localized(),
|
|
confirmStyle: .danger,
|
|
cancelStyle: .alert_text,
|
|
onConfirm: { [weak self] _ in self?.sendMessage(hasPermissionToSendSeed: true) }
|
|
)
|
|
)
|
|
|
|
return present(modal, animated: true, completion: nil)
|
|
}
|
|
|
|
// Clearing this out immediately to make this appear more snappy
|
|
DispatchQueue.main.async { [weak self] in
|
|
self?.snInputView.text = ""
|
|
self?.snInputView.quoteDraftInfo = nil
|
|
|
|
self?.resetMentions()
|
|
}
|
|
|
|
// Note: 'shouldBeVisible' is set to true the first time a thread is saved so we can
|
|
// use it to determine if the user is creating a new thread and update the 'isApproved'
|
|
// flags appropriately
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
let threadVariant: SessionThread.Variant = self.viewModel.threadData.threadVariant
|
|
let oldThreadShouldBeVisible: Bool = (self.viewModel.threadData.threadShouldBeVisible == true)
|
|
let sentTimestampMs: Int64 = SnodeAPI.currentOffsetTimestampMs()
|
|
let linkPreviewDraft: LinkPreviewDraft? = snInputView.linkPreviewInfo?.draft
|
|
let quoteModel: QuotedReplyModel? = snInputView.quoteDraftInfo?.model
|
|
|
|
// If this was a message request then approve it
|
|
approveMessageRequestIfNeeded(
|
|
for: threadId,
|
|
threadVariant: self.viewModel.threadData.threadVariant,
|
|
isNewThread: !oldThreadShouldBeVisible,
|
|
timestampMs: (sentTimestampMs - 1) // Set 1ms earlier as this is used for sorting
|
|
)
|
|
|
|
// Send the message
|
|
Storage.shared
|
|
.writePublisher { [weak self] db in
|
|
// Let the viewModel know we are about to send a message
|
|
self?.viewModel.sentMessageBeforeUpdate = true
|
|
|
|
// Update the thread to be visible (if it isn't already)
|
|
if self?.viewModel.threadData.threadShouldBeVisible == false {
|
|
_ = try SessionThread
|
|
.filter(id: threadId)
|
|
.updateAllAndConfig(db, SessionThread.Columns.shouldBeVisible.set(to: true))
|
|
}
|
|
|
|
let authorId: String = {
|
|
if let blindedId = self?.viewModel.threadData.currentUserBlindedPublicKey {
|
|
return blindedId
|
|
}
|
|
return self?.viewModel.threadData.currentUserPublicKey ?? getUserHexEncodedPublicKey(db)
|
|
}()
|
|
|
|
// Create the interaction
|
|
let interaction: Interaction = try Interaction(
|
|
threadId: threadId,
|
|
authorId: authorId,
|
|
variant: .standardOutgoing,
|
|
body: text,
|
|
timestampMs: sentTimestampMs,
|
|
hasMention: Interaction.isUserMentioned(db, threadId: threadId, body: text),
|
|
expiresInSeconds: try? DisappearingMessagesConfiguration
|
|
.select(.durationSeconds)
|
|
.filter(id: threadId)
|
|
.filter(DisappearingMessagesConfiguration.Columns.isEnabled == true)
|
|
.asRequest(of: TimeInterval.self)
|
|
.fetchOne(db),
|
|
linkPreviewUrl: linkPreviewDraft?.urlString
|
|
).inserted(db)
|
|
|
|
// If there is a LinkPreview and it doesn't match an existing one then add it now
|
|
if
|
|
let linkPreviewDraft: LinkPreviewDraft = linkPreviewDraft,
|
|
(try? interaction.linkPreview.isEmpty(db)) == true
|
|
{
|
|
try LinkPreview(
|
|
url: linkPreviewDraft.urlString,
|
|
title: linkPreviewDraft.title,
|
|
attachmentId: LinkPreview.saveAttachmentIfPossible(
|
|
db,
|
|
imageData: linkPreviewDraft.jpegImageData,
|
|
mimeType: OWSMimeTypeImageJpeg
|
|
)
|
|
).insert(db)
|
|
}
|
|
|
|
// If there is a Quote the insert it now
|
|
if let interactionId: Int64 = interaction.id, let quoteModel: QuotedReplyModel = quoteModel {
|
|
try Quote(
|
|
interactionId: interactionId,
|
|
authorId: quoteModel.authorId,
|
|
timestampMs: quoteModel.timestampMs,
|
|
body: quoteModel.body,
|
|
attachmentId: quoteModel.generateAttachmentThumbnailIfNeeded(db)
|
|
).insert(db)
|
|
}
|
|
|
|
try MessageSender.send(
|
|
db,
|
|
interaction: interaction,
|
|
threadId: threadId,
|
|
threadVariant: threadVariant
|
|
)
|
|
}
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
.sinkUntilComplete(
|
|
receiveCompletion: { [weak self] _ in
|
|
self?.handleMessageSent()
|
|
}
|
|
)
|
|
}
|
|
|
|
func sendAttachments(_ attachments: [SignalAttachment], with text: String, hasPermissionToSendSeed: Bool = false, onComplete: (() -> ())? = nil) {
|
|
guard !showBlockedModalIfNeeded() else { return }
|
|
|
|
for attachment in attachments {
|
|
if attachment.hasError {
|
|
return showErrorAlert(for: attachment, onDismiss: onComplete)
|
|
}
|
|
}
|
|
|
|
let text = replaceMentions(in: snInputView.text.trimmingCharacters(in: .whitespacesAndNewlines))
|
|
|
|
if text.contains(mnemonic) && !viewModel.threadData.threadIsNoteToSelf && !hasPermissionToSendSeed {
|
|
// Warn the user if they're about to send their seed to someone
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
info: ConfirmationModal.Info(
|
|
title: "modal_send_seed_title".localized(),
|
|
explanation: "modal_send_seed_explanation".localized(),
|
|
confirmTitle: "modal_send_seed_send_button_title".localized(),
|
|
confirmStyle: .danger,
|
|
cancelStyle: .alert_text,
|
|
onConfirm: { [weak self] _ in
|
|
self?.sendAttachments(attachments, with: text, hasPermissionToSendSeed: true, onComplete: onComplete)
|
|
}
|
|
)
|
|
)
|
|
|
|
return present(modal, animated: true, completion: nil)
|
|
}
|
|
|
|
// Clearing this out immediately to make this appear more snappy
|
|
DispatchQueue.main.async { [weak self] in
|
|
self?.snInputView.text = ""
|
|
self?.snInputView.quoteDraftInfo = nil
|
|
|
|
self?.resetMentions()
|
|
}
|
|
|
|
// Note: 'shouldBeVisible' is set to true the first time a thread is saved so we can
|
|
// use it to determine if the user is creating a new thread and update the 'isApproved'
|
|
// flags appropriately
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
let threadVariant: SessionThread.Variant = self.viewModel.threadData.threadVariant
|
|
let oldThreadShouldBeVisible: Bool = (self.viewModel.threadData.threadShouldBeVisible == true)
|
|
let sentTimestampMs: Int64 = SnodeAPI.currentOffsetTimestampMs()
|
|
|
|
// If this was a message request then approve it
|
|
approveMessageRequestIfNeeded(
|
|
for: threadId,
|
|
threadVariant: self.viewModel.threadData.threadVariant,
|
|
isNewThread: !oldThreadShouldBeVisible,
|
|
timestampMs: (sentTimestampMs - 1) // Set 1ms earlier as this is used for sorting
|
|
)
|
|
|
|
// Send the message
|
|
Storage.shared
|
|
.writePublisher { [weak self] db in
|
|
// Let the viewModel know we are about to send a message
|
|
self?.viewModel.sentMessageBeforeUpdate = true
|
|
|
|
// Update the thread to be visible (if it isn't already)
|
|
if self?.viewModel.threadData.threadShouldBeVisible == false {
|
|
_ = try SessionThread
|
|
.filter(id: threadId)
|
|
.updateAllAndConfig(db, SessionThread.Columns.shouldBeVisible.set(to: true))
|
|
}
|
|
|
|
// Create the interaction
|
|
let interaction: Interaction = try Interaction(
|
|
threadId: threadId,
|
|
authorId: getUserHexEncodedPublicKey(db),
|
|
variant: .standardOutgoing,
|
|
body: text,
|
|
timestampMs: sentTimestampMs,
|
|
hasMention: Interaction.isUserMentioned(db, threadId: threadId, body: text),
|
|
expiresInSeconds: try? DisappearingMessagesConfiguration
|
|
.select(.durationSeconds)
|
|
.filter(id: threadId)
|
|
.filter(DisappearingMessagesConfiguration.Columns.isEnabled == true)
|
|
.asRequest(of: TimeInterval.self)
|
|
.fetchOne(db)
|
|
).inserted(db)
|
|
|
|
guard let interactionId: Int64 = interaction.id else {
|
|
return
|
|
}
|
|
|
|
// Prepare any attachments
|
|
try Attachment.prepare(
|
|
db,
|
|
attachments: attachments,
|
|
for: interactionId
|
|
)
|
|
|
|
// Send the message
|
|
try MessageSender.send(
|
|
db,
|
|
interaction: interaction,
|
|
threadId: threadId,
|
|
threadVariant: threadVariant
|
|
)
|
|
}
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
.sinkUntilComplete(
|
|
receiveCompletion: { [weak self] _ in
|
|
self?.handleMessageSent()
|
|
|
|
// Attachment successfully sent - dismiss the screen
|
|
DispatchQueue.main.async {
|
|
onComplete?()
|
|
}
|
|
}
|
|
)
|
|
}
|
|
|
|
func handleMessageSent() {
|
|
if Storage.shared[.playNotificationSoundInForeground] {
|
|
let soundID = Preferences.Sound.systemSoundId(for: .messageSent, quiet: true)
|
|
AudioServicesPlaySystemSound(soundID)
|
|
}
|
|
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
|
|
Storage.shared.writeAsync { db in
|
|
TypingIndicators.didStopTyping(db, threadId: threadId, direction: .outgoing)
|
|
|
|
_ = try SessionThread
|
|
.filter(id: threadId)
|
|
.updateAll(db, SessionThread.Columns.messageDraft.set(to: ""))
|
|
}
|
|
}
|
|
|
|
func showLinkPreviewSuggestionModal() {
|
|
let linkPreviewModal: ConfirmationModal = ConfirmationModal(
|
|
info: ConfirmationModal.Info(
|
|
title: "modal_link_previews_title".localized(),
|
|
explanation: "modal_link_previews_explanation".localized(),
|
|
confirmTitle: "modal_link_previews_button_title".localized()
|
|
) { [weak self] _ in
|
|
Storage.shared.writeAsync { db in
|
|
db[.areLinkPreviewsEnabled] = true
|
|
}
|
|
|
|
self?.snInputView.autoGenerateLinkPreview()
|
|
}
|
|
)
|
|
|
|
present(linkPreviewModal, animated: true, completion: nil)
|
|
}
|
|
|
|
func inputTextViewDidChangeContent(_ inputTextView: InputTextView) {
|
|
// Note: If there is a 'draft' message then we don't want it to trigger the typing indicator to
|
|
// appear (as that is not expected/correct behaviour)
|
|
guard !viewIsAppearing else { return }
|
|
|
|
let newText: String = (inputTextView.text ?? "")
|
|
|
|
if !newText.isEmpty {
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
let threadVariant: SessionThread.Variant = self.viewModel.threadData.threadVariant
|
|
let threadIsMessageRequest: Bool = (self.viewModel.threadData.threadIsMessageRequest == true)
|
|
let threadIsBlocked: Bool = (self.viewModel.threadData.threadIsBlocked == true)
|
|
let needsToStartTypingIndicator: Bool = TypingIndicators.didStartTypingNeedsToStart(
|
|
threadId: threadId,
|
|
threadVariant: threadVariant,
|
|
threadIsBlocked: threadIsBlocked,
|
|
threadIsMessageRequest: threadIsMessageRequest,
|
|
direction: .outgoing,
|
|
timestampMs: SnodeAPI.currentOffsetTimestampMs()
|
|
)
|
|
|
|
if needsToStartTypingIndicator {
|
|
Storage.shared.writeAsync { db in
|
|
TypingIndicators.start(db, threadId: threadId, direction: .outgoing)
|
|
}
|
|
}
|
|
}
|
|
|
|
updateMentions(for: newText)
|
|
}
|
|
|
|
// MARK: --Attachments
|
|
|
|
func didPasteImageFromPasteboard(_ image: UIImage) {
|
|
guard let imageData = image.jpegData(compressionQuality: 1.0) else { return }
|
|
|
|
let dataSource = DataSourceValue.dataSource(with: imageData, utiType: kUTTypeJPEG as String)
|
|
let attachment = SignalAttachment.attachment(dataSource: dataSource, dataUTI: kUTTypeJPEG as String, imageQuality: .medium)
|
|
|
|
let approvalVC = AttachmentApprovalViewController.wrappedInNavController(
|
|
threadId: self.viewModel.threadData.threadId,
|
|
attachments: [ attachment ],
|
|
approvalDelegate: self
|
|
)
|
|
approvalVC.modalPresentationStyle = .fullScreen
|
|
|
|
self.present(approvalVC, animated: true, completion: nil)
|
|
}
|
|
|
|
// MARK: --Mentions
|
|
|
|
func handleMentionSelected(_ mentionInfo: MentionInfo, from view: MentionSelectionView) {
|
|
guard let currentMentionStartIndex = currentMentionStartIndex else { return }
|
|
|
|
mentions.append(mentionInfo)
|
|
|
|
let newText: String = snInputView.text.replacingCharacters(
|
|
in: currentMentionStartIndex...,
|
|
with: "@\(mentionInfo.profile.displayName(for: self.viewModel.threadData.threadVariant)) "
|
|
)
|
|
|
|
snInputView.text = newText
|
|
self.currentMentionStartIndex = nil
|
|
snInputView.hideMentionsUI()
|
|
|
|
mentions = mentions.filter { mentionInfo -> Bool in
|
|
newText.contains(mentionInfo.profile.displayName(for: self.viewModel.threadData.threadVariant))
|
|
}
|
|
}
|
|
|
|
func updateMentions(for newText: String) {
|
|
guard !newText.isEmpty else {
|
|
if currentMentionStartIndex != nil {
|
|
snInputView.hideMentionsUI()
|
|
}
|
|
|
|
resetMentions()
|
|
return
|
|
}
|
|
|
|
let lastCharacterIndex = newText.index(before: newText.endIndex)
|
|
let lastCharacter = newText[lastCharacterIndex]
|
|
|
|
// Check if there is whitespace before the '@' or the '@' is the first character
|
|
let isCharacterBeforeLastWhiteSpaceOrStartOfLine: Bool
|
|
if newText.count == 1 {
|
|
isCharacterBeforeLastWhiteSpaceOrStartOfLine = true // Start of line
|
|
}
|
|
else {
|
|
let characterBeforeLast = newText[newText.index(before: lastCharacterIndex)]
|
|
isCharacterBeforeLastWhiteSpaceOrStartOfLine = characterBeforeLast.isWhitespace
|
|
}
|
|
|
|
if lastCharacter == "@" && isCharacterBeforeLastWhiteSpaceOrStartOfLine {
|
|
currentMentionStartIndex = lastCharacterIndex
|
|
snInputView.showMentionsUI(for: self.viewModel.mentions())
|
|
}
|
|
else if lastCharacter.isWhitespace || lastCharacter == "@" { // the lastCharacter == "@" is to check for @@
|
|
currentMentionStartIndex = nil
|
|
snInputView.hideMentionsUI()
|
|
}
|
|
else {
|
|
if let currentMentionStartIndex = currentMentionStartIndex {
|
|
let query = String(newText[newText.index(after: currentMentionStartIndex)...]) // + 1 to get rid of the @
|
|
snInputView.showMentionsUI(for: self.viewModel.mentions(for: query))
|
|
}
|
|
}
|
|
}
|
|
|
|
func resetMentions() {
|
|
currentMentionStartIndex = nil
|
|
mentions = []
|
|
}
|
|
|
|
func replaceMentions(in text: String) -> String {
|
|
var result = text
|
|
for mention in mentions {
|
|
guard let range = result.range(of: "@\(mention.profile.displayName(for: mention.threadVariant))") else { continue }
|
|
result = result.replacingCharacters(in: range, with: "@\(mention.profile.id)")
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func hideInputAccessoryView() {
|
|
self.inputAccessoryView?.isHidden = true
|
|
self.inputAccessoryView?.alpha = 0
|
|
}
|
|
|
|
func showInputAccessoryView() {
|
|
UIView.animate(withDuration: 0.25, animations: {
|
|
self.inputAccessoryView?.isHidden = false
|
|
self.inputAccessoryView?.alpha = 1
|
|
})
|
|
}
|
|
|
|
// MARK: MessageCellDelegate
|
|
|
|
func handleItemLongPressed(_ cellViewModel: MessageViewModel) {
|
|
// Show the context menu if applicable
|
|
guard
|
|
// FIXME: Need to update this when an appropriate replacement is added (see https://teng.pub/technical/2021/11/9/uiapplication-key-window-replacement)
|
|
let keyWindow: UIWindow = UIApplication.shared.keyWindow,
|
|
let sectionIndex: Int = self.viewModel.interactionData
|
|
.firstIndex(where: { $0.model == .messages }),
|
|
let index = self.viewModel.interactionData[sectionIndex]
|
|
.elements
|
|
.firstIndex(of: cellViewModel),
|
|
let cell = tableView.cellForRow(at: IndexPath(row: index, section: sectionIndex)) as? MessageCell,
|
|
let contextSnapshotView: UIView = cell.contextSnapshotView,
|
|
let snapshot = contextSnapshotView.snapshotView(afterScreenUpdates: false),
|
|
contextMenuWindow == nil,
|
|
let actions: [ContextMenuVC.Action] = ContextMenuVC.actions(
|
|
for: cellViewModel,
|
|
recentEmojis: (self.viewModel.threadData.recentReactionEmoji ?? []).compactMap { EmojiWithSkinTones(rawValue: $0) },
|
|
currentUserPublicKey: self.viewModel.threadData.currentUserPublicKey,
|
|
currentUserBlindedPublicKey: self.viewModel.threadData.currentUserBlindedPublicKey,
|
|
currentUserIsOpenGroupModerator: OpenGroupManager.isUserModeratorOrAdmin(
|
|
self.viewModel.threadData.currentUserPublicKey,
|
|
for: self.viewModel.threadData.openGroupRoomToken,
|
|
on: self.viewModel.threadData.openGroupServer
|
|
),
|
|
currentThreadIsMessageRequest: (self.viewModel.threadData.threadIsMessageRequest == true),
|
|
delegate: self
|
|
)
|
|
else { return }
|
|
|
|
/// Lock the contentOffset of the tableView so the transition doesn't look buggy
|
|
self.tableView.lockContentOffset = true
|
|
|
|
UIImpactFeedbackGenerator(style: .heavy).impactOccurred()
|
|
self.contextMenuWindow = ContextMenuWindow()
|
|
self.contextMenuVC = ContextMenuVC(
|
|
snapshot: snapshot,
|
|
frame: contextSnapshotView.convert(contextSnapshotView.bounds, to: keyWindow),
|
|
cellViewModel: cellViewModel,
|
|
actions: actions
|
|
) { [weak self] in
|
|
self?.contextMenuWindow?.isHidden = true
|
|
self?.contextMenuVC = nil
|
|
self?.contextMenuWindow = nil
|
|
self?.scrollButton.alpha = 0
|
|
|
|
UIView.animate(
|
|
withDuration: 0.25,
|
|
animations: {
|
|
self?.scrollButton.alpha = (self?.getScrollButtonOpacity() ?? 0)
|
|
self?.unreadCountView.alpha = (self?.scrollButton.alpha ?? 0)
|
|
},
|
|
completion: { _ in
|
|
guard let contentOffset: CGPoint = self?.tableView.contentOffset else { return }
|
|
|
|
// Unlock the contentOffset so everything will be in the right
|
|
// place when we return
|
|
self?.tableView.lockContentOffset = false
|
|
self?.tableView.setContentOffset(contentOffset, animated: false)
|
|
}
|
|
)
|
|
}
|
|
|
|
self.contextMenuWindow?.themeBackgroundColor = .clear
|
|
self.contextMenuWindow?.rootViewController = self.contextMenuVC
|
|
self.contextMenuWindow?.overrideUserInterfaceStyle = ThemeManager.currentTheme.interfaceStyle
|
|
self.contextMenuWindow?.makeKeyAndVisible()
|
|
}
|
|
|
|
func handleItemTapped(_ cellViewModel: MessageViewModel, gestureRecognizer: UITapGestureRecognizer) {
|
|
guard cellViewModel.variant != .standardOutgoing || (cellViewModel.state != .failed && cellViewModel.state != .failedToSync) else {
|
|
// Show the failed message sheet
|
|
showFailedMessageSheet(for: cellViewModel)
|
|
return
|
|
}
|
|
|
|
// For call info messages show the "call missed" modal
|
|
guard cellViewModel.variant != .infoCall else {
|
|
let callMissedTipsModal: CallMissedTipsModal = CallMissedTipsModal(caller: cellViewModel.authorName)
|
|
present(callMissedTipsModal, animated: true, completion: nil)
|
|
return
|
|
}
|
|
|
|
// If it's an incoming media message and the thread isn't trusted then show the placeholder view
|
|
if cellViewModel.cellType != .textOnlyMessage && cellViewModel.variant == .standardIncoming && !cellViewModel.threadIsTrusted {
|
|
let message: String = String(
|
|
format: "modal_download_attachment_explanation".localized(),
|
|
cellViewModel.authorName
|
|
)
|
|
let confirmationModal: ConfirmationModal = ConfirmationModal(
|
|
info: ConfirmationModal.Info(
|
|
title: String(
|
|
format: "modal_download_attachment_title".localized(),
|
|
cellViewModel.authorName
|
|
),
|
|
attributedExplanation: NSAttributedString(string: message)
|
|
.adding(
|
|
attributes: [ .font: UIFont.boldSystemFont(ofSize: Values.smallFontSize) ],
|
|
range: (message as NSString).range(of: cellViewModel.authorName)
|
|
),
|
|
confirmTitle: "modal_download_button_title".localized(),
|
|
confirmAccessibility: Accessibility(identifier: "Download media"),
|
|
cancelAccessibility: Accessibility(identifier: "Don't download media"),
|
|
dismissOnConfirm: false // Custom dismissal logic
|
|
) { [weak self] _ in
|
|
self?.viewModel.trustContact()
|
|
self?.dismiss(animated: true, completion: nil)
|
|
}
|
|
)
|
|
|
|
present(confirmationModal, animated: true, completion: nil)
|
|
return
|
|
}
|
|
|
|
switch cellViewModel.cellType {
|
|
case .audio: viewModel.playOrPauseAudio(for: cellViewModel)
|
|
|
|
case .mediaMessage:
|
|
guard
|
|
let sectionIndex: Int = self.viewModel.interactionData
|
|
.firstIndex(where: { $0.model == .messages }),
|
|
let messageIndex: Int = self.viewModel.interactionData[sectionIndex]
|
|
.elements
|
|
.firstIndex(where: { $0.id == cellViewModel.id }),
|
|
let cell = tableView.cellForRow(at: IndexPath(row: messageIndex, section: sectionIndex)) as? VisibleMessageCell,
|
|
let albumView: MediaAlbumView = cell.albumView
|
|
else { return }
|
|
|
|
let locationInCell: CGPoint = gestureRecognizer.location(in: cell)
|
|
|
|
// Figure out which of the media views was tapped
|
|
let locationInAlbumView: CGPoint = cell.convert(locationInCell, to: albumView)
|
|
guard let mediaView = albumView.mediaView(forLocation: locationInAlbumView) else { return }
|
|
|
|
switch mediaView.attachment.state {
|
|
case .pendingDownload, .downloading, .uploading, .invalid: break
|
|
|
|
// Failed uploads should be handled via the "resend" process instead
|
|
case .failedUpload: break
|
|
|
|
case .failedDownload:
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
|
|
// Retry downloading the failed attachment
|
|
Storage.shared.writeAsync { db in
|
|
JobRunner.add(
|
|
db,
|
|
job: Job(
|
|
variant: .attachmentDownload,
|
|
threadId: threadId,
|
|
interactionId: cellViewModel.id,
|
|
details: AttachmentDownloadJob.Details(
|
|
attachmentId: mediaView.attachment.id
|
|
)
|
|
)
|
|
)
|
|
}
|
|
break
|
|
|
|
default:
|
|
// Ignore invalid media
|
|
guard mediaView.attachment.isValid else { return }
|
|
|
|
let viewController: UIViewController? = MediaGalleryViewModel.createDetailViewController(
|
|
for: self.viewModel.threadData.threadId,
|
|
threadVariant: self.viewModel.threadData.threadVariant,
|
|
interactionId: cellViewModel.id,
|
|
selectedAttachmentId: mediaView.attachment.id,
|
|
options: [ .sliderEnabled, .showAllMediaButton ]
|
|
)
|
|
|
|
if let viewController: UIViewController = viewController {
|
|
/// Delay becoming the first responder to make the return transition a little nicer (allows
|
|
/// for the footer on the detail view to slide out rather than instantly vanish)
|
|
self.delayFirstResponder = true
|
|
|
|
/// Dismiss the input before starting the presentation to make everything look smoother
|
|
self.resignFirstResponder()
|
|
|
|
/// Delay the actual presentation to give the 'resignFirstResponder' call the chance to complete
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + .milliseconds(250)) { [weak self] in
|
|
/// Lock the contentOffset of the tableView so the transition doesn't look buggy
|
|
self?.tableView.lockContentOffset = true
|
|
|
|
self?.present(viewController, animated: true) { [weak self] in
|
|
// Unlock the contentOffset so everything will be in the right
|
|
// place when we return
|
|
self?.tableView.lockContentOffset = false
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
case .genericAttachment:
|
|
guard
|
|
let attachment: Attachment = cellViewModel.attachments?.first,
|
|
let originalFilePath: String = attachment.originalFilePath
|
|
else { return }
|
|
|
|
let fileUrl: URL = URL(fileURLWithPath: originalFilePath)
|
|
|
|
// Open a preview of the document for text, pdf or microsoft files
|
|
if
|
|
attachment.isText ||
|
|
attachment.isMicrosoftDoc ||
|
|
attachment.contentType == OWSMimeTypeApplicationPdf
|
|
{
|
|
|
|
let interactionController: UIDocumentInteractionController = UIDocumentInteractionController(url: fileUrl)
|
|
interactionController.delegate = self
|
|
interactionController.presentPreview(animated: true)
|
|
return
|
|
}
|
|
|
|
// Otherwise share the file
|
|
let shareVC = UIActivityViewController(activityItems: [ fileUrl ], applicationActivities: nil)
|
|
|
|
if UIDevice.current.isIPad {
|
|
shareVC.excludedActivityTypes = []
|
|
shareVC.popoverPresentationController?.permittedArrowDirections = []
|
|
shareVC.popoverPresentationController?.sourceView = self.view
|
|
shareVC.popoverPresentationController?.sourceRect = self.view.bounds
|
|
}
|
|
|
|
navigationController?.present(shareVC, animated: true, completion: nil)
|
|
|
|
case .textOnlyMessage:
|
|
if let quote: Quote = cellViewModel.quote {
|
|
// Scroll to the original quoted message
|
|
let maybeOriginalInteractionInfo: Interaction.TimestampInfo? = Storage.shared.read { db in
|
|
try quote.originalInteraction
|
|
.select(.id, .timestampMs)
|
|
.asRequest(of: Interaction.TimestampInfo.self)
|
|
.fetchOne(db)
|
|
}
|
|
|
|
guard let interactionInfo: Interaction.TimestampInfo = maybeOriginalInteractionInfo else {
|
|
return
|
|
}
|
|
|
|
self.scrollToInteractionIfNeeded(with: interactionInfo, highlight: true)
|
|
}
|
|
else if let linkPreview: LinkPreview = cellViewModel.linkPreview {
|
|
switch linkPreview.variant {
|
|
case .standard: openUrl(linkPreview.url)
|
|
case .openGroupInvitation: joinOpenGroup(name: linkPreview.title, url: linkPreview.url)
|
|
}
|
|
}
|
|
|
|
default: break
|
|
}
|
|
}
|
|
|
|
func handleItemDoubleTapped(_ cellViewModel: MessageViewModel) {
|
|
switch cellViewModel.cellType {
|
|
// The user can double tap a voice message when it's playing to speed it up
|
|
case .audio: self.viewModel.speedUpAudio(for: cellViewModel)
|
|
default: break
|
|
}
|
|
}
|
|
|
|
func handleItemSwiped(_ cellViewModel: MessageViewModel, state: SwipeState) {
|
|
switch state {
|
|
case .began: tableView.isScrollEnabled = false
|
|
case .ended, .cancelled: tableView.isScrollEnabled = true
|
|
}
|
|
}
|
|
|
|
func openUrl(_ urlString: String) {
|
|
guard let url: URL = URL(string: urlString) else { return }
|
|
|
|
// URLs can be unsafe, so always ask the user whether they want to open one
|
|
let actionSheet: UIAlertController = UIAlertController(
|
|
title: "modal_open_url_title".localized(),
|
|
message: String(format: "modal_open_url_explanation".localized(), url.absoluteString),
|
|
preferredStyle: .actionSheet
|
|
)
|
|
actionSheet.addAction(UIAlertAction(title: "modal_open_url_button_title".localized(), style: .default) { [weak self] _ in
|
|
UIApplication.shared.open(url, options: [:], completionHandler: nil)
|
|
self?.showInputAccessoryView()
|
|
})
|
|
actionSheet.addAction(UIAlertAction(title: "modal_copy_url_button_title".localized(), style: .default) { [weak self] _ in
|
|
UIPasteboard.general.string = url.absoluteString
|
|
self?.showInputAccessoryView()
|
|
})
|
|
actionSheet.addAction(UIAlertAction(title: "cancel".localized(), style: .cancel) { [weak self] _ in
|
|
self?.showInputAccessoryView()
|
|
})
|
|
|
|
Modal.setupForIPadIfNeeded(actionSheet, targetView: self.view)
|
|
self.present(actionSheet, animated: true)
|
|
}
|
|
|
|
func handleReplyButtonTapped(for cellViewModel: MessageViewModel) {
|
|
reply(cellViewModel)
|
|
}
|
|
|
|
func startThread(with sessionId: String, openGroupServer: String?, openGroupPublicKey: String?) {
|
|
guard viewModel.threadData.canWrite else { return }
|
|
guard SessionId.Prefix(from: sessionId) == .blinded else {
|
|
Storage.shared.write { db in
|
|
try SessionThread
|
|
.fetchOrCreate(db, id: sessionId, variant: .contact, shouldBeVisible: nil)
|
|
}
|
|
|
|
let conversationVC: ConversationVC = ConversationVC(threadId: sessionId, threadVariant: .contact)
|
|
|
|
self.navigationController?.pushViewController(conversationVC, animated: true)
|
|
return
|
|
}
|
|
|
|
// If the sessionId is blinded then check if there is an existing un-blinded thread with the contact
|
|
// and use that, otherwise just use the blinded id
|
|
guard let openGroupServer: String = openGroupServer, let openGroupPublicKey: String = openGroupPublicKey else {
|
|
return
|
|
}
|
|
|
|
let targetThreadId: String? = Storage.shared.write { db in
|
|
let lookup: BlindedIdLookup = try BlindedIdLookup
|
|
.fetchOrCreate(
|
|
db,
|
|
blindedId: sessionId,
|
|
openGroupServer: openGroupServer,
|
|
openGroupPublicKey: openGroupPublicKey,
|
|
isCheckingForOutbox: false
|
|
)
|
|
|
|
return try SessionThread
|
|
.fetchOrCreate(
|
|
db,
|
|
id: (lookup.sessionId ?? lookup.blindedId),
|
|
variant: .contact,
|
|
shouldBeVisible: nil
|
|
)
|
|
.id
|
|
}
|
|
|
|
guard let threadId: String = targetThreadId else { return }
|
|
|
|
let conversationVC: ConversationVC = ConversationVC(threadId: threadId, threadVariant: .contact)
|
|
self.navigationController?.pushViewController(conversationVC, animated: true)
|
|
}
|
|
|
|
func showReactionList(_ cellViewModel: MessageViewModel, selectedReaction: EmojiWithSkinTones?) {
|
|
guard
|
|
cellViewModel.reactionInfo?.isEmpty == false &&
|
|
(
|
|
self.viewModel.threadData.threadVariant == .legacyGroup ||
|
|
self.viewModel.threadData.threadVariant == .group ||
|
|
self.viewModel.threadData.threadVariant == .community
|
|
),
|
|
let allMessages: [MessageViewModel] = self.viewModel.interactionData
|
|
.first(where: { $0.model == .messages })?
|
|
.elements
|
|
else { return }
|
|
|
|
let reactionListSheet: ReactionListSheet = ReactionListSheet(for: cellViewModel.id) { [weak self] in
|
|
self?.currentReactionListSheet = nil
|
|
}
|
|
reactionListSheet.delegate = self
|
|
reactionListSheet.handleInteractionUpdates(
|
|
allMessages,
|
|
selectedReaction: selectedReaction,
|
|
initialLoad: true,
|
|
shouldShowClearAllButton: OpenGroupManager.isUserModeratorOrAdmin(
|
|
self.viewModel.threadData.currentUserPublicKey,
|
|
for: self.viewModel.threadData.openGroupRoomToken,
|
|
on: self.viewModel.threadData.openGroupServer
|
|
)
|
|
)
|
|
reactionListSheet.modalPresentationStyle = .overFullScreen
|
|
present(reactionListSheet, animated: true, completion: nil)
|
|
|
|
// Store so we can updated the content based on the current VC
|
|
self.currentReactionListSheet = reactionListSheet
|
|
}
|
|
|
|
func needsLayout(for cellViewModel: MessageViewModel, expandingReactions: Bool) {
|
|
guard
|
|
let messageSectionIndex: Int = self.viewModel.interactionData
|
|
.firstIndex(where: { $0.model == .messages }),
|
|
let targetMessageIndex = self.viewModel.interactionData[messageSectionIndex]
|
|
.elements
|
|
.firstIndex(where: { $0.id == cellViewModel.id })
|
|
else { return }
|
|
|
|
if expandingReactions {
|
|
self.viewModel.expandReactions(for: cellViewModel.id)
|
|
}
|
|
else {
|
|
self.viewModel.collapseReactions(for: cellViewModel.id)
|
|
}
|
|
|
|
UIView.setAnimationsEnabled(false)
|
|
tableView.reloadRows(
|
|
at: [IndexPath(row: targetMessageIndex, section: messageSectionIndex)],
|
|
with: .none
|
|
)
|
|
UIView.setAnimationsEnabled(true)
|
|
}
|
|
|
|
func react(_ cellViewModel: MessageViewModel, with emoji: EmojiWithSkinTones) {
|
|
react(cellViewModel, with: emoji.rawValue, remove: false)
|
|
}
|
|
|
|
func removeReact(_ cellViewModel: MessageViewModel, for emoji: EmojiWithSkinTones) {
|
|
react(cellViewModel, with: emoji.rawValue, remove: true)
|
|
}
|
|
|
|
func removeAllReactions(_ cellViewModel: MessageViewModel, for emoji: String) {
|
|
guard cellViewModel.threadVariant == .community else { return }
|
|
|
|
Storage.shared
|
|
.readPublisherFlatMap { db -> AnyPublisher<(OpenGroupAPI.ReactionRemoveAllResponse, OpenGroupAPI.PendingChange), Error> in
|
|
guard
|
|
let openGroup: OpenGroup = try? OpenGroup
|
|
.fetchOne(db, id: cellViewModel.threadId),
|
|
let openGroupServerMessageId: Int64 = try? Interaction
|
|
.select(.openGroupServerMessageId)
|
|
.filter(id: cellViewModel.id)
|
|
.asRequest(of: Int64.self)
|
|
.fetchOne(db)
|
|
else { throw StorageError.objectNotFound }
|
|
|
|
let pendingChange: OpenGroupAPI.PendingChange = OpenGroupManager
|
|
.addPendingReaction(
|
|
emoji: emoji,
|
|
id: openGroupServerMessageId,
|
|
in: openGroup.roomToken,
|
|
on: openGroup.server,
|
|
type: .removeAll
|
|
)
|
|
|
|
return OpenGroupAPI
|
|
.reactionDeleteAll(
|
|
db,
|
|
emoji: emoji,
|
|
id: openGroupServerMessageId,
|
|
in: openGroup.roomToken,
|
|
on: openGroup.server
|
|
)
|
|
.map { _, response in (response, pendingChange) }
|
|
.eraseToAnyPublisher()
|
|
}
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
.handleEvents(
|
|
receiveOutput: { response, pendingChange in
|
|
OpenGroupManager
|
|
.updatePendingChange(
|
|
pendingChange,
|
|
seqNo: response.seqNo
|
|
)
|
|
}
|
|
)
|
|
.sinkUntilComplete(
|
|
receiveCompletion: { _ in
|
|
Storage.shared.writeAsync { db in
|
|
_ = try Reaction
|
|
.filter(Reaction.Columns.interactionId == cellViewModel.id)
|
|
.filter(Reaction.Columns.emoji == emoji)
|
|
.deleteAll(db)
|
|
}
|
|
}
|
|
)
|
|
}
|
|
|
|
func react(_ cellViewModel: MessageViewModel, with emoji: String, remove: Bool) {
|
|
guard cellViewModel.variant == .standardIncoming || cellViewModel.variant == .standardOutgoing else {
|
|
return
|
|
}
|
|
|
|
let threadIsMessageRequest: Bool = (self.viewModel.threadData.threadIsMessageRequest == true)
|
|
guard !threadIsMessageRequest else { return }
|
|
|
|
// Perform local rate limiting (don't allow more than 20 reactions within 60 seconds)
|
|
let sentTimestamp: Int64 = SnodeAPI.currentOffsetTimestampMs()
|
|
let recentReactionTimestamps: [Int64] = General.cache.wrappedValue.recentReactionTimestamps
|
|
|
|
guard
|
|
recentReactionTimestamps.count < 20 ||
|
|
(sentTimestamp - (recentReactionTimestamps.first ?? sentTimestamp)) > (60 * 1000)
|
|
else {
|
|
let toastController: ToastController = ToastController(
|
|
text: "EMOJI_REACTS_RATE_LIMIT_TOAST".localized(),
|
|
background: .backgroundSecondary
|
|
)
|
|
toastController.presentToastView(
|
|
fromBottomOfView: self.view,
|
|
inset: (snInputView.bounds.height + Values.largeSpacing),
|
|
duration: .milliseconds(2500)
|
|
)
|
|
return
|
|
}
|
|
|
|
General.cache.mutate {
|
|
$0.recentReactionTimestamps = Array($0.recentReactionTimestamps
|
|
.suffix(19))
|
|
.appending(sentTimestamp)
|
|
}
|
|
|
|
// Perform the sending logic
|
|
Storage.shared
|
|
.writePublisherFlatMap { [weak self] db -> AnyPublisher<MessageSender.PreparedSendData?, Error> in
|
|
// Update the thread to be visible (if it isn't already)
|
|
if self?.viewModel.threadData.threadShouldBeVisible == false {
|
|
_ = try SessionThread
|
|
.filter(id: cellViewModel.threadId)
|
|
.updateAllAndConfig(db, SessionThread.Columns.shouldBeVisible.set(to: true))
|
|
}
|
|
|
|
let pendingReaction: Reaction? = {
|
|
if remove {
|
|
return try? Reaction
|
|
.filter(Reaction.Columns.interactionId == cellViewModel.id)
|
|
.filter(Reaction.Columns.authorId == cellViewModel.currentUserPublicKey)
|
|
.filter(Reaction.Columns.emoji == emoji)
|
|
.fetchOne(db)
|
|
} else {
|
|
let sortId = Reaction.getSortId(
|
|
db,
|
|
interactionId: cellViewModel.id,
|
|
emoji: emoji
|
|
)
|
|
|
|
return Reaction(
|
|
interactionId: cellViewModel.id,
|
|
serverHash: nil,
|
|
timestampMs: sentTimestamp,
|
|
authorId: cellViewModel.currentUserPublicKey,
|
|
emoji: emoji,
|
|
count: 1,
|
|
sortId: sortId
|
|
)
|
|
}
|
|
}()
|
|
|
|
// Update the database
|
|
if remove {
|
|
try Reaction
|
|
.filter(Reaction.Columns.interactionId == cellViewModel.id)
|
|
.filter(Reaction.Columns.authorId == cellViewModel.currentUserPublicKey)
|
|
.filter(Reaction.Columns.emoji == emoji)
|
|
.deleteAll(db)
|
|
}
|
|
else {
|
|
try pendingReaction?.insert(db)
|
|
|
|
// Add it to the recent list
|
|
Emoji.addRecent(db, emoji: emoji)
|
|
}
|
|
|
|
// If it's not an OpenGroup then send the message directly to the thread
|
|
guard
|
|
let openGroup: OpenGroup = try? OpenGroup.fetchOne(db, id: cellViewModel.threadId),
|
|
OpenGroupManager.isOpenGroupSupport(.reactions, on: openGroup.server)
|
|
else {
|
|
let sendData: MessageSender.PreparedSendData = try MessageSender.preparedSendData(
|
|
db,
|
|
message: VisibleMessage(
|
|
sentTimestamp: UInt64(sentTimestamp),
|
|
text: nil,
|
|
reaction: VisibleMessage.VMReaction(
|
|
timestamp: UInt64(cellViewModel.timestampMs),
|
|
publicKey: {
|
|
guard cellViewModel.variant == .standardIncoming else {
|
|
return cellViewModel.currentUserPublicKey
|
|
}
|
|
|
|
return cellViewModel.authorId
|
|
}(),
|
|
emoji: emoji,
|
|
kind: (remove ? .remove : .react)
|
|
)
|
|
),
|
|
to: try Message.Destination
|
|
.from(db, threadId: cellViewModel.threadId, threadVariant: cellViewModel.threadVariant),
|
|
namespace: try Message.Destination
|
|
.from(db, threadId: cellViewModel.threadId, threadVariant: cellViewModel.threadVariant)
|
|
.defaultNamespace,
|
|
interactionId: cellViewModel.id
|
|
)
|
|
|
|
return Just(sendData)
|
|
.setFailureType(to: Error.self)
|
|
.eraseToAnyPublisher()
|
|
}
|
|
|
|
// Otherwise we need to make an API call to the OpenGroup
|
|
// Send reaction to open groups
|
|
guard
|
|
let openGroupServerMessageId: Int64 = try? Interaction
|
|
.select(.openGroupServerMessageId)
|
|
.filter(id: cellViewModel.id)
|
|
.asRequest(of: Int64.self)
|
|
.fetchOne(db)
|
|
else { throw MessageSenderError.invalidMessage }
|
|
|
|
let pendingChange: OpenGroupAPI.PendingChange = OpenGroupManager
|
|
.addPendingReaction(
|
|
emoji: emoji,
|
|
id: openGroupServerMessageId,
|
|
in: openGroup.roomToken,
|
|
on: openGroup.server,
|
|
type: (remove ? .remove : .add)
|
|
)
|
|
|
|
let request: AnyPublisher<Int64?, Error> = {
|
|
switch remove {
|
|
case true:
|
|
return OpenGroupAPI
|
|
.reactionDelete(
|
|
db,
|
|
emoji: emoji,
|
|
id: openGroupServerMessageId,
|
|
in: openGroup.roomToken,
|
|
on: openGroup.server
|
|
)
|
|
.map { _, response in response.seqNo }
|
|
.eraseToAnyPublisher()
|
|
|
|
case false:
|
|
return OpenGroupAPI
|
|
.reactionAdd(
|
|
db,
|
|
emoji: emoji,
|
|
id: openGroupServerMessageId,
|
|
in: openGroup.roomToken,
|
|
on: openGroup.server
|
|
)
|
|
.map { _, response in response.seqNo }
|
|
.eraseToAnyPublisher()
|
|
}
|
|
}()
|
|
|
|
return request
|
|
.handleEvents(
|
|
receiveOutput: { seqNo in
|
|
OpenGroupManager
|
|
.updatePendingChange(
|
|
pendingChange,
|
|
seqNo: seqNo
|
|
)
|
|
},
|
|
receiveCompletion: { [weak self] result in
|
|
switch result {
|
|
case .finished: break
|
|
case .failure:
|
|
OpenGroupManager.removePendingChange(pendingChange)
|
|
|
|
self?.handleReactionSentFailure(
|
|
pendingReaction,
|
|
remove: remove
|
|
)
|
|
}
|
|
}
|
|
)
|
|
.map { _ in nil }
|
|
.eraseToAnyPublisher()
|
|
}
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
.flatMap { maybeSendData -> AnyPublisher<Void, Error> in
|
|
guard let sendData: MessageSender.PreparedSendData = maybeSendData else {
|
|
return Just(())
|
|
.setFailureType(to: Error.self)
|
|
.eraseToAnyPublisher()
|
|
}
|
|
|
|
return MessageSender.sendImmediate(preparedSendData: sendData)
|
|
}
|
|
.sinkUntilComplete()
|
|
}
|
|
|
|
func handleReactionSentFailure(_ pendingReaction: Reaction?, remove: Bool) {
|
|
guard let pendingReaction = pendingReaction else { return }
|
|
Storage.shared.writeAsync { db in
|
|
// Reverse the database
|
|
if remove {
|
|
try pendingReaction.insert(db)
|
|
}
|
|
else {
|
|
try Reaction
|
|
.filter(Reaction.Columns.interactionId == pendingReaction.interactionId)
|
|
.filter(Reaction.Columns.authorId == pendingReaction.authorId)
|
|
.filter(Reaction.Columns.emoji == pendingReaction.emoji)
|
|
.deleteAll(db)
|
|
}
|
|
}
|
|
}
|
|
|
|
func showFullEmojiKeyboard(_ cellViewModel: MessageViewModel) {
|
|
hideInputAccessoryView()
|
|
|
|
let emojiPicker = EmojiPickerSheet(
|
|
completionHandler: { [weak self] emoji in
|
|
guard let emoji: EmojiWithSkinTones = emoji else { return }
|
|
|
|
self?.react(cellViewModel, with: emoji)
|
|
},
|
|
dismissHandler: { [weak self] in
|
|
self?.showInputAccessoryView()
|
|
}
|
|
)
|
|
|
|
present(emojiPicker, animated: true, completion: nil)
|
|
}
|
|
|
|
func contextMenuDismissed() {
|
|
recoverInputView()
|
|
}
|
|
|
|
// MARK: --action handling
|
|
|
|
func showFailedMessageSheet(for cellViewModel: MessageViewModel) {
|
|
let sheet = UIAlertController(
|
|
title: (cellViewModel.state == .failedToSync ?
|
|
"MESSAGE_DELIVERY_FAILED_SYNC_TITLE".localized() :
|
|
"MESSAGE_DELIVERY_FAILED_TITLE".localized()
|
|
),
|
|
message: cellViewModel.mostRecentFailureText,
|
|
preferredStyle: .actionSheet
|
|
)
|
|
sheet.addAction(UIAlertAction(title: "TXT_CANCEL_TITLE".localized(), style: .cancel, handler: nil))
|
|
|
|
if cellViewModel.state != .failedToSync {
|
|
sheet.addAction(UIAlertAction(title: "TXT_DELETE_TITLE".localized(), style: .destructive, handler: { _ in
|
|
Storage.shared.writeAsync { db in
|
|
try Interaction
|
|
.filter(id: cellViewModel.id)
|
|
.deleteAll(db)
|
|
}
|
|
}))
|
|
}
|
|
|
|
sheet.addAction(UIAlertAction(
|
|
title: (cellViewModel.state == .failedToSync ?
|
|
"context_menu_resync".localized() :
|
|
"context_menu_resend".localized()
|
|
),
|
|
style: .default,
|
|
handler: { [weak self] _ in self?.retry(cellViewModel) }
|
|
))
|
|
|
|
// HACK: Extracting this info from the error string is pretty dodgy
|
|
let prefix: String = "HTTP request failed at destination (Service node "
|
|
if let mostRecentFailureText: String = cellViewModel.mostRecentFailureText, mostRecentFailureText.hasPrefix(prefix) {
|
|
let rest = mostRecentFailureText.substring(from: prefix.count)
|
|
|
|
if let index = rest.firstIndex(of: ")") {
|
|
let snodeAddress = String(rest[rest.startIndex..<index])
|
|
|
|
sheet.addAction(UIAlertAction(title: "Copy Service Node Info", style: .default) { _ in
|
|
UIPasteboard.general.string = snodeAddress
|
|
})
|
|
}
|
|
}
|
|
|
|
Modal.setupForIPadIfNeeded(sheet, targetView: self.view)
|
|
present(sheet, animated: true, completion: nil)
|
|
}
|
|
|
|
func joinOpenGroup(name: String?, url: String) {
|
|
// Open groups can be unsafe, so always ask the user whether they want to join one
|
|
let finalName: String = (name ?? "Open Group")
|
|
let message: String = "Are you sure you want to join the \(finalName) open group?";
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
info: ConfirmationModal.Info(
|
|
title: "Join \(finalName)?",
|
|
attributedExplanation: NSMutableAttributedString(string: message)
|
|
.adding(
|
|
attributes: [ .font: UIFont.boldSystemFont(ofSize: Values.smallFontSize) ],
|
|
range: (message as NSString).range(of: finalName)
|
|
),
|
|
confirmTitle: "JOIN_COMMUNITY_BUTTON_TITLE".localized(),
|
|
onConfirm: { modal in
|
|
guard let presentingViewController: UIViewController = modal.presentingViewController else {
|
|
return
|
|
}
|
|
|
|
guard let (room, server, publicKey) = SessionUtil.parseCommunity(url: url) else {
|
|
let errorModal: ConfirmationModal = ConfirmationModal(
|
|
info: ConfirmationModal.Info(
|
|
title: "COMMUNITY_ERROR_GENERIC".localized(),
|
|
cancelTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text
|
|
)
|
|
)
|
|
|
|
return presentingViewController.present(errorModal, animated: true, completion: nil)
|
|
}
|
|
|
|
Storage.shared
|
|
.writePublisherFlatMap { db in
|
|
OpenGroupManager.shared.add(
|
|
db,
|
|
roomToken: room,
|
|
server: server,
|
|
publicKey: publicKey,
|
|
calledFromConfigHandling: false
|
|
)
|
|
}
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
.receive(on: DispatchQueue.main)
|
|
.sinkUntilComplete(
|
|
receiveCompletion: { result in
|
|
switch result {
|
|
case .finished: break
|
|
case .failure(let error):
|
|
let errorModal: ConfirmationModal = ConfirmationModal(
|
|
info: ConfirmationModal.Info(
|
|
title: "COMMUNITY_ERROR_GENERIC".localized(),
|
|
explanation: error.localizedDescription,
|
|
cancelTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text
|
|
)
|
|
)
|
|
|
|
presentingViewController.present(errorModal, animated: true, completion: nil)
|
|
}
|
|
}
|
|
)
|
|
}
|
|
)
|
|
)
|
|
|
|
present(modal, animated: true, completion: nil)
|
|
}
|
|
|
|
// MARK: - ContextMenuActionDelegate
|
|
|
|
func info(_ cellViewModel: MessageViewModel) {
|
|
let mediaInfoVC = MediaInfoVC(
|
|
attachments: (cellViewModel.attachments ?? []),
|
|
isOutgoing: (cellViewModel.variant == .standardOutgoing),
|
|
threadId: self.viewModel.threadData.threadId,
|
|
threadVariant: self.viewModel.threadData.threadVariant,
|
|
interactionId: cellViewModel.id
|
|
)
|
|
navigationController?.pushViewController(mediaInfoVC, animated: true)
|
|
}
|
|
|
|
func retry(_ cellViewModel: MessageViewModel) {
|
|
Storage.shared.writeAsync { [weak self] db in
|
|
guard
|
|
let threadId: String = self?.viewModel.threadData.threadId,
|
|
let threadVariant: SessionThread.Variant = self?.viewModel.threadData.threadVariant,
|
|
let interaction: Interaction = try? Interaction.fetchOne(db, id: cellViewModel.id)
|
|
else { return }
|
|
|
|
if
|
|
let quote = try? interaction.quote.fetchOne(db),
|
|
let quotedAttachment = try? quote.attachment.fetchOne(db),
|
|
quotedAttachment.isVisualMedia,
|
|
quotedAttachment.downloadUrl == Attachment.nonMediaQuoteFileId,
|
|
let quotedInteraction = try? quote.originalInteraction.fetchOne(db)
|
|
{
|
|
let attachment: Attachment? = {
|
|
if let attachment = try? quotedInteraction.attachments.fetchOne(db) {
|
|
return attachment
|
|
}
|
|
if
|
|
let linkPreview = try? quotedInteraction.linkPreview.fetchOne(db),
|
|
let linkPreviewAttachment = try? linkPreview.attachment.fetchOne(db)
|
|
{
|
|
return linkPreviewAttachment
|
|
}
|
|
|
|
return nil
|
|
}()
|
|
try quote.with(
|
|
attachmentId: attachment?.cloneAsQuoteThumbnail()?.inserted(db).id
|
|
).update(db)
|
|
}
|
|
|
|
// Remove message sending jobs for the same interaction in database
|
|
// Prevent the same message being sent twice
|
|
try Job.filter(Job.Columns.interactionId == interaction.id).deleteAll(db)
|
|
|
|
try MessageSender.send(
|
|
db,
|
|
interaction: interaction,
|
|
threadId: threadId,
|
|
threadVariant: threadVariant,
|
|
isSyncMessage: (cellViewModel.state == .failedToSync)
|
|
)
|
|
}
|
|
}
|
|
|
|
func reply(_ cellViewModel: MessageViewModel) {
|
|
let maybeQuoteDraft: QuotedReplyModel? = QuotedReplyModel.quotedReplyForSending(
|
|
threadId: self.viewModel.threadData.threadId,
|
|
authorId: cellViewModel.authorId,
|
|
variant: cellViewModel.variant,
|
|
body: cellViewModel.body,
|
|
timestampMs: cellViewModel.timestampMs,
|
|
attachments: cellViewModel.attachments,
|
|
linkPreviewAttachment: cellViewModel.linkPreviewAttachment,
|
|
currentUserPublicKey: cellViewModel.currentUserPublicKey,
|
|
currentUserBlindedPublicKey: cellViewModel.currentUserBlindedPublicKey
|
|
)
|
|
|
|
guard let quoteDraft: QuotedReplyModel = maybeQuoteDraft else { return }
|
|
|
|
snInputView.quoteDraftInfo = (
|
|
model: quoteDraft,
|
|
isOutgoing: (cellViewModel.variant == .standardOutgoing)
|
|
)
|
|
snInputView.becomeFirstResponder()
|
|
}
|
|
|
|
func copy(_ cellViewModel: MessageViewModel) {
|
|
switch cellViewModel.cellType {
|
|
case .typingIndicator, .dateHeader: break
|
|
|
|
case .textOnlyMessage:
|
|
if cellViewModel.body == nil, let linkPreview: LinkPreview = cellViewModel.linkPreview {
|
|
UIPasteboard.general.string = linkPreview.url
|
|
return
|
|
}
|
|
|
|
UIPasteboard.general.string = cellViewModel.body
|
|
|
|
case .audio, .genericAttachment, .mediaMessage:
|
|
guard
|
|
cellViewModel.attachments?.count == 1,
|
|
let attachment: Attachment = cellViewModel.attachments?.first,
|
|
attachment.isValid,
|
|
(
|
|
attachment.state == .downloaded ||
|
|
attachment.state == .uploaded
|
|
),
|
|
let utiType: String = MIMETypeUtil.utiType(forMIMEType: attachment.contentType),
|
|
let originalFilePath: String = attachment.originalFilePath,
|
|
let data: Data = try? Data(contentsOf: URL(fileURLWithPath: originalFilePath))
|
|
else { return }
|
|
|
|
UIPasteboard.general.setData(data, forPasteboardType: utiType)
|
|
}
|
|
}
|
|
|
|
func copySessionID(_ cellViewModel: MessageViewModel) {
|
|
guard cellViewModel.variant == .standardIncoming || cellViewModel.variant == .standardIncomingDeleted else {
|
|
return
|
|
}
|
|
|
|
UIPasteboard.general.string = cellViewModel.authorId
|
|
}
|
|
|
|
func delete(_ cellViewModel: MessageViewModel) {
|
|
switch cellViewModel.variant {
|
|
case .standardIncomingDeleted, .infoCall,
|
|
.infoScreenshotNotification, .infoMediaSavedNotification,
|
|
.infoClosedGroupCreated, .infoClosedGroupUpdated,
|
|
.infoClosedGroupCurrentUserLeft, .infoClosedGroupCurrentUserLeaving, .infoClosedGroupCurrentUserErrorLeaving,
|
|
.infoMessageRequestAccepted, .infoDisappearingMessagesUpdate:
|
|
// Info messages and unsent messages should just trigger a local
|
|
// deletion (they are created as side effects so we wouldn't be
|
|
// able to delete them for all participants anyway)
|
|
Storage.shared.writeAsync { db in
|
|
_ = try Interaction
|
|
.filter(id: cellViewModel.id)
|
|
.deleteAll(db)
|
|
}
|
|
return
|
|
|
|
case .standardOutgoing, .standardIncoming: break
|
|
}
|
|
|
|
let threadName: String = self.viewModel.threadData.displayName
|
|
let userPublicKey: String = getUserHexEncodedPublicKey()
|
|
|
|
// Remote deletion logic
|
|
func deleteRemotely(from viewController: UIViewController?, request: AnyPublisher<Void, Error>, onComplete: (() -> ())?) {
|
|
// Show a loading indicator
|
|
Deferred {
|
|
Future<Void, Error> { resolver in
|
|
ModalActivityIndicatorViewController.present(fromViewController: viewController, canCancel: false) { _ in
|
|
resolver(Result.success(()))
|
|
}
|
|
}
|
|
}
|
|
.flatMap { _ in request }
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
.receive(on: DispatchQueue.main)
|
|
.sinkUntilComplete(
|
|
receiveCompletion: { [weak self] result in
|
|
switch result {
|
|
case .failure: break
|
|
case .finished:
|
|
// Delete the interaction (and associated data) from the database
|
|
Storage.shared.writeAsync { db in
|
|
_ = try Interaction
|
|
.filter(id: cellViewModel.id)
|
|
.deleteAll(db)
|
|
}
|
|
}
|
|
|
|
// Regardless of success we should dismiss and callback
|
|
if self?.presentedViewController is ModalActivityIndicatorViewController {
|
|
self?.dismiss(animated: true, completion: nil) // Dismiss the loader
|
|
}
|
|
|
|
onComplete?()
|
|
}
|
|
)
|
|
}
|
|
|
|
// How we delete the message differs depending on the type of thread
|
|
switch cellViewModel.threadVariant {
|
|
// Handle open group messages the old way
|
|
case .community:
|
|
// If it's an incoming message the user must have moderator status
|
|
let result: (openGroupServerMessageId: Int64?, openGroup: OpenGroup?)? = Storage.shared.read { db -> (Int64?, OpenGroup?) in
|
|
(
|
|
try Interaction
|
|
.select(.openGroupServerMessageId)
|
|
.filter(id: cellViewModel.id)
|
|
.asRequest(of: Int64.self)
|
|
.fetchOne(db),
|
|
try OpenGroup.fetchOne(db, id: cellViewModel.threadId)
|
|
)
|
|
}
|
|
|
|
guard
|
|
let openGroup: OpenGroup = result?.openGroup,
|
|
let openGroupServerMessageId: Int64 = result?.openGroupServerMessageId, (
|
|
cellViewModel.variant != .standardIncoming ||
|
|
OpenGroupManager.isUserModeratorOrAdmin(
|
|
userPublicKey,
|
|
for: openGroup.roomToken,
|
|
on: openGroup.server
|
|
)
|
|
)
|
|
else {
|
|
// If the message hasn't been sent yet then just delete locally
|
|
guard cellViewModel.state == .sending || cellViewModel.state == .failed else { return }
|
|
|
|
// Retrieve any message send jobs for this interaction
|
|
let jobs: [Job] = Storage.shared
|
|
.read { db in
|
|
try? Job
|
|
.filter(Job.Columns.variant == Job.Variant.messageSend)
|
|
.filter(Job.Columns.interactionId == cellViewModel.id)
|
|
.fetchAll(db)
|
|
}
|
|
.defaulting(to: [])
|
|
|
|
// If the job is currently running then wait until it's done before triggering
|
|
// the deletion
|
|
let targetJob: Job? = jobs.first(where: { JobRunner.isCurrentlyRunning($0) })
|
|
|
|
guard targetJob == nil else {
|
|
JobRunner.afterCurrentlyRunningJob(targetJob) { [weak self] result in
|
|
switch result {
|
|
// If it succeeded then we'll need to delete from the server so re-run
|
|
// this function (if we still don't have the server id for some reason
|
|
// then this would result in a local-only deletion which should be fine
|
|
case .succeeded: self?.delete(cellViewModel)
|
|
|
|
// Otherwise we just need to cancel the pending job (in case it retries)
|
|
// and delete the interaction
|
|
default:
|
|
JobRunner.removePendingJob(targetJob)
|
|
|
|
Storage.shared.writeAsync { db in
|
|
_ = try Interaction
|
|
.filter(id: cellViewModel.id)
|
|
.deleteAll(db)
|
|
}
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// If it's not currently running then remove any pending jobs (just to be safe) and
|
|
// delete the interaction locally
|
|
jobs.forEach { JobRunner.removePendingJob($0) }
|
|
|
|
Storage.shared.writeAsync { db in
|
|
_ = try Interaction
|
|
.filter(id: cellViewModel.id)
|
|
.deleteAll(db)
|
|
}
|
|
return
|
|
}
|
|
|
|
// Delete the message from the open group
|
|
deleteRemotely(
|
|
from: self,
|
|
request: Storage.shared.readPublisherFlatMap { db in
|
|
OpenGroupAPI.messageDelete(
|
|
db,
|
|
id: openGroupServerMessageId,
|
|
in: openGroup.roomToken,
|
|
on: openGroup.server
|
|
)
|
|
.map { _ in () }
|
|
.eraseToAnyPublisher()
|
|
}
|
|
) { [weak self] in
|
|
self?.showInputAccessoryView()
|
|
}
|
|
|
|
case .contact, .legacyGroup, .group:
|
|
let serverHash: String? = Storage.shared.read { db -> String? in
|
|
try Interaction
|
|
.select(.serverHash)
|
|
.filter(id: cellViewModel.id)
|
|
.asRequest(of: String.self)
|
|
.fetchOne(db)
|
|
}
|
|
let unsendRequest: UnsendRequest = UnsendRequest(
|
|
timestamp: UInt64(cellViewModel.timestampMs),
|
|
author: (cellViewModel.variant == .standardOutgoing ?
|
|
userPublicKey :
|
|
cellViewModel.authorId
|
|
)
|
|
)
|
|
|
|
// For incoming interactions or interactions with no serverHash just delete them locally
|
|
guard cellViewModel.variant == .standardOutgoing, let serverHash: String = serverHash else {
|
|
Storage.shared.writeAsync { db in
|
|
_ = try Interaction
|
|
.filter(id: cellViewModel.id)
|
|
.deleteAll(db)
|
|
|
|
// No need to send the unsendRequest if there is no serverHash (ie. the message
|
|
// was outgoing but never got to the server)
|
|
guard serverHash != nil else { return }
|
|
|
|
MessageSender
|
|
.send(
|
|
db,
|
|
message: unsendRequest,
|
|
threadId: cellViewModel.threadId,
|
|
interactionId: nil,
|
|
to: .contact(publicKey: userPublicKey)
|
|
)
|
|
}
|
|
return
|
|
}
|
|
|
|
let actionSheet: UIAlertController = UIAlertController(title: nil, message: nil, preferredStyle: .actionSheet)
|
|
actionSheet.addAction(UIAlertAction(
|
|
title: "delete_message_for_me".localized(),
|
|
accessibilityIdentifier: "Delete for me",
|
|
style: .destructive
|
|
) { [weak self] _ in
|
|
Storage.shared.writeAsync { db in
|
|
_ = try Interaction
|
|
.filter(id: cellViewModel.id)
|
|
.deleteAll(db)
|
|
|
|
MessageSender
|
|
.send(
|
|
db,
|
|
message: unsendRequest,
|
|
threadId: cellViewModel.threadId,
|
|
interactionId: nil,
|
|
to: .contact(publicKey: userPublicKey)
|
|
)
|
|
}
|
|
self?.showInputAccessoryView()
|
|
})
|
|
|
|
actionSheet.addAction(UIAlertAction(
|
|
title: {
|
|
switch cellViewModel.threadVariant {
|
|
case .legacyGroup, .group: return "delete_message_for_everyone".localized()
|
|
default:
|
|
return (cellViewModel.threadId == userPublicKey ?
|
|
"delete_message_for_me_and_my_devices".localized() :
|
|
String(format: "delete_message_for_me_and_recipient".localized(), threadName)
|
|
)
|
|
}
|
|
}(),
|
|
accessibilityIdentifier: "Delete for everyone",
|
|
style: .destructive
|
|
) { [weak self] _ in
|
|
deleteRemotely(
|
|
from: self,
|
|
request: SnodeAPI
|
|
.deleteMessages(
|
|
publicKey: cellViewModel.threadId,
|
|
serverHashes: [serverHash]
|
|
)
|
|
.map { _ in () }
|
|
.eraseToAnyPublisher()
|
|
) { [weak self] in
|
|
Storage.shared.writeAsync { db in
|
|
try MessageSender
|
|
.send(
|
|
db,
|
|
message: unsendRequest,
|
|
interactionId: nil,
|
|
threadId: cellViewModel.threadId,
|
|
threadVariant: cellViewModel.threadVariant
|
|
)
|
|
}
|
|
|
|
self?.showInputAccessoryView()
|
|
}
|
|
})
|
|
|
|
actionSheet.addAction(UIAlertAction.init(title: "TXT_CANCEL_TITLE".localized(), style: .cancel) { [weak self] _ in
|
|
self?.showInputAccessoryView()
|
|
})
|
|
|
|
self.inputAccessoryView?.isHidden = true
|
|
self.inputAccessoryView?.alpha = 0
|
|
Modal.setupForIPadIfNeeded(actionSheet, targetView: self.view)
|
|
self.present(actionSheet, animated: true)
|
|
}
|
|
}
|
|
|
|
func save(_ cellViewModel: MessageViewModel) {
|
|
guard cellViewModel.cellType == .mediaMessage else { return }
|
|
|
|
let mediaAttachments: [(Attachment, String)] = (cellViewModel.attachments ?? [])
|
|
.filter { attachment in
|
|
attachment.isValid &&
|
|
attachment.isVisualMedia && (
|
|
attachment.state == .downloaded ||
|
|
attachment.state == .uploaded
|
|
)
|
|
}
|
|
.compactMap { attachment in
|
|
guard let originalFilePath: String = attachment.originalFilePath else { return nil }
|
|
|
|
return (attachment, originalFilePath)
|
|
}
|
|
|
|
guard !mediaAttachments.isEmpty else { return }
|
|
|
|
mediaAttachments.forEach { attachment, originalFilePath in
|
|
PHPhotoLibrary.shared().performChanges(
|
|
{
|
|
if attachment.isImage || attachment.isAnimated {
|
|
PHAssetChangeRequest.creationRequestForAssetFromImage(
|
|
atFileURL: URL(fileURLWithPath: originalFilePath)
|
|
)
|
|
}
|
|
else if attachment.isVideo {
|
|
PHAssetChangeRequest.creationRequestForAssetFromVideo(
|
|
atFileURL: URL(fileURLWithPath: originalFilePath)
|
|
)
|
|
}
|
|
},
|
|
completionHandler: { _, _ in }
|
|
)
|
|
}
|
|
|
|
// Send a 'media saved' notification if needed
|
|
guard self.viewModel.threadData.threadVariant == .contact, cellViewModel.variant == .standardIncoming else {
|
|
return
|
|
}
|
|
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
let threadVariant: SessionThread.Variant = self.viewModel.threadData.threadVariant
|
|
|
|
Storage.shared.writeAsync { db in
|
|
try MessageSender.send(
|
|
db,
|
|
message: DataExtractionNotification(
|
|
kind: .mediaSaved(timestamp: UInt64(cellViewModel.timestampMs)),
|
|
sentTimestamp: UInt64(SnodeAPI.currentOffsetTimestampMs())
|
|
),
|
|
interactionId: nil,
|
|
threadId: threadId,
|
|
threadVariant: threadVariant
|
|
)
|
|
}
|
|
}
|
|
|
|
func ban(_ cellViewModel: MessageViewModel) {
|
|
guard cellViewModel.threadVariant == .community else { return }
|
|
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
targetView: self.view,
|
|
info: ConfirmationModal.Info(
|
|
title: "Session",
|
|
explanation: "This will ban the selected user from this room. It won't ban them from other rooms.",
|
|
confirmTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text,
|
|
onConfirm: { [weak self] _ in
|
|
Storage.shared
|
|
.readPublisherFlatMap { db -> AnyPublisher<Void, Error> in
|
|
guard let openGroup: OpenGroup = try OpenGroup.fetchOne(db, id: threadId) else {
|
|
throw StorageError.objectNotFound
|
|
}
|
|
|
|
return OpenGroupAPI
|
|
.userBan(
|
|
db,
|
|
sessionId: cellViewModel.authorId,
|
|
from: [openGroup.roomToken],
|
|
on: openGroup.server
|
|
)
|
|
.map { _ in () }
|
|
.eraseToAnyPublisher()
|
|
}
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
.receive(on: DispatchQueue.main)
|
|
.sinkUntilComplete(
|
|
receiveCompletion: { result in
|
|
switch result {
|
|
case .finished: break
|
|
case .failure:
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
targetView: self?.view,
|
|
info: ConfirmationModal.Info(
|
|
title: CommonStrings.errorAlertTitle,
|
|
explanation: "context_menu_ban_user_error_alert_message".localized(),
|
|
cancelTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text
|
|
)
|
|
)
|
|
self?.present(modal, animated: true)
|
|
}
|
|
}
|
|
)
|
|
|
|
self?.becomeFirstResponder()
|
|
},
|
|
afterClosed: { [weak self] in self?.becomeFirstResponder() }
|
|
)
|
|
)
|
|
self.present(modal, animated: true)
|
|
}
|
|
|
|
func banAndDeleteAllMessages(_ cellViewModel: MessageViewModel) {
|
|
guard cellViewModel.threadVariant == .community else { return }
|
|
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
targetView: self.view,
|
|
info: ConfirmationModal.Info(
|
|
title: "Session",
|
|
explanation: "This will ban the selected user from this room and delete all messages sent by them. It won't ban them from other rooms or delete the messages they sent there.",
|
|
confirmTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text,
|
|
onConfirm: { [weak self] _ in
|
|
Storage.shared
|
|
.readPublisherFlatMap { db -> AnyPublisher<Void, Error> in
|
|
guard let openGroup: OpenGroup = try OpenGroup.fetchOne(db, id: threadId) else {
|
|
throw StorageError.objectNotFound
|
|
}
|
|
|
|
return OpenGroupAPI
|
|
.userBanAndDeleteAllMessages(
|
|
db,
|
|
sessionId: cellViewModel.authorId,
|
|
in: openGroup.roomToken,
|
|
on: openGroup.server
|
|
)
|
|
.map { _ in () }
|
|
.eraseToAnyPublisher()
|
|
}
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
.receive(on: DispatchQueue.main)
|
|
.sinkUntilComplete(
|
|
receiveCompletion: { result in
|
|
switch result {
|
|
case .finished: break
|
|
case .failure:
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
targetView: self?.view,
|
|
info: ConfirmationModal.Info(
|
|
title: CommonStrings.errorAlertTitle,
|
|
explanation: "context_menu_ban_user_error_alert_message".localized(),
|
|
cancelTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text
|
|
)
|
|
)
|
|
self?.present(modal, animated: true)
|
|
}
|
|
}
|
|
)
|
|
|
|
self?.becomeFirstResponder()
|
|
},
|
|
afterClosed: { [weak self] in self?.becomeFirstResponder() }
|
|
)
|
|
)
|
|
self.present(modal, animated: true)
|
|
}
|
|
|
|
// MARK: - VoiceMessageRecordingViewDelegate
|
|
|
|
func startVoiceMessageRecording() {
|
|
// Request permission if needed
|
|
Permissions.requestMicrophonePermissionIfNeeded() { [weak self] in
|
|
DispatchQueue.main.async {
|
|
self?.cancelVoiceMessageRecording()
|
|
}
|
|
}
|
|
|
|
// Keep screen on
|
|
UIApplication.shared.isIdleTimerDisabled = false
|
|
guard AVAudioSession.sharedInstance().recordPermission == .granted else { return }
|
|
|
|
// Cancel any current audio playback
|
|
self.viewModel.stopAudio()
|
|
|
|
// Create URL
|
|
let directory: String = OWSTemporaryDirectory()
|
|
let fileName: String = "\(SnodeAPI.currentOffsetTimestampMs()).m4a"
|
|
let url: URL = URL(fileURLWithPath: directory).appendingPathComponent(fileName)
|
|
|
|
// Set up audio session
|
|
let isConfigured = (Environment.shared?.audioSession.startAudioActivity(recordVoiceMessageActivity) == true)
|
|
guard isConfigured else {
|
|
return cancelVoiceMessageRecording()
|
|
}
|
|
|
|
// Set up audio recorder
|
|
let audioRecorder: AVAudioRecorder
|
|
do {
|
|
audioRecorder = try AVAudioRecorder(
|
|
url: url,
|
|
settings: [
|
|
AVFormatIDKey: NSNumber(value: kAudioFormatMPEG4AAC),
|
|
AVSampleRateKey: NSNumber(value: 44100),
|
|
AVNumberOfChannelsKey: NSNumber(value: 2),
|
|
AVEncoderBitRateKey: NSNumber(value: 128 * 1024)
|
|
]
|
|
)
|
|
audioRecorder.isMeteringEnabled = true
|
|
self.audioRecorder = audioRecorder
|
|
}
|
|
catch {
|
|
SNLog("Couldn't start audio recording due to error: \(error).")
|
|
return cancelVoiceMessageRecording()
|
|
}
|
|
|
|
// Limit voice messages to a minute
|
|
audioTimer = Timer.scheduledTimer(withTimeInterval: 180, repeats: false, block: { [weak self] _ in
|
|
self?.snInputView.hideVoiceMessageUI()
|
|
self?.endVoiceMessageRecording()
|
|
})
|
|
|
|
// Prepare audio recorder
|
|
guard audioRecorder.prepareToRecord() else {
|
|
SNLog("Couldn't prepare audio recorder.")
|
|
return cancelVoiceMessageRecording()
|
|
}
|
|
|
|
// Start recording
|
|
guard audioRecorder.record() else {
|
|
SNLog("Couldn't record audio.")
|
|
return cancelVoiceMessageRecording()
|
|
}
|
|
}
|
|
|
|
func endVoiceMessageRecording() {
|
|
UIApplication.shared.isIdleTimerDisabled = true
|
|
|
|
// Hide the UI
|
|
snInputView.hideVoiceMessageUI()
|
|
|
|
// Cancel the timer
|
|
audioTimer?.invalidate()
|
|
|
|
// Check preconditions
|
|
guard let audioRecorder = audioRecorder else { return }
|
|
|
|
// Get duration
|
|
let duration = audioRecorder.currentTime
|
|
|
|
// Stop the recording
|
|
stopVoiceMessageRecording()
|
|
|
|
// Check for user misunderstanding
|
|
guard duration > 1 else {
|
|
self.audioRecorder = nil
|
|
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
targetView: self.view,
|
|
info: ConfirmationModal.Info(
|
|
title: "VOICE_MESSAGE_TOO_SHORT_ALERT_TITLE".localized(),
|
|
explanation: "VOICE_MESSAGE_TOO_SHORT_ALERT_MESSAGE".localized(),
|
|
cancelTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text
|
|
)
|
|
)
|
|
self.present(modal, animated: true)
|
|
return
|
|
}
|
|
|
|
// Get data
|
|
let dataSourceOrNil = DataSourcePath.dataSource(with: audioRecorder.url, shouldDeleteOnDeallocation: true)
|
|
self.audioRecorder = nil
|
|
|
|
guard let dataSource = dataSourceOrNil else { return SNLog("Couldn't load recorded data.") }
|
|
|
|
// Create attachment
|
|
let fileName = ("VOICE_MESSAGE_FILE_NAME".localized() as NSString).appendingPathExtension("m4a")
|
|
dataSource.sourceFilename = fileName
|
|
|
|
let attachment = SignalAttachment.voiceMessageAttachment(dataSource: dataSource, dataUTI: kUTTypeMPEG4Audio as String)
|
|
|
|
guard !attachment.hasError else {
|
|
return showErrorAlert(for: attachment, onDismiss: nil)
|
|
}
|
|
|
|
// Send attachment
|
|
sendAttachments([ attachment ], with: "")
|
|
}
|
|
|
|
func cancelVoiceMessageRecording() {
|
|
snInputView.hideVoiceMessageUI()
|
|
audioTimer?.invalidate()
|
|
stopVoiceMessageRecording()
|
|
audioRecorder = nil
|
|
}
|
|
|
|
func stopVoiceMessageRecording() {
|
|
audioRecorder?.stop()
|
|
Environment.shared?.audioSession.endAudioActivity(recordVoiceMessageActivity)
|
|
}
|
|
|
|
// MARK: - Data Extraction Notifications
|
|
|
|
@objc func sendScreenshotNotification() {
|
|
// Only send screenshot notifications to one-to-one conversations
|
|
guard self.viewModel.threadData.threadVariant == .contact else { return }
|
|
|
|
let threadId: String = self.viewModel.threadData.threadId
|
|
let threadVariant: SessionThread.Variant = self.viewModel.threadData.threadVariant
|
|
|
|
Storage.shared.writeAsync { db in
|
|
try MessageSender.send(
|
|
db,
|
|
message: DataExtractionNotification(
|
|
kind: .screenshot,
|
|
sentTimestamp: UInt64(SnodeAPI.currentOffsetTimestampMs())
|
|
),
|
|
interactionId: nil,
|
|
threadId: threadId,
|
|
threadVariant: threadVariant
|
|
)
|
|
}
|
|
}
|
|
|
|
// MARK: - Convenience
|
|
|
|
func showErrorAlert(for attachment: SignalAttachment, onDismiss: (() -> ())?) {
|
|
let modal: ConfirmationModal = ConfirmationModal(
|
|
targetView: self.view,
|
|
info: ConfirmationModal.Info(
|
|
title: "ATTACHMENT_ERROR_ALERT_TITLE".localized(),
|
|
explanation: (attachment.localizedErrorDescription ?? SignalAttachment.missingDataErrorMessage),
|
|
cancelTitle: "BUTTON_OK".localized(),
|
|
cancelStyle: .alert_text,
|
|
afterClosed: onDismiss
|
|
)
|
|
)
|
|
self.present(modal, animated: true)
|
|
}
|
|
}
|
|
|
|
// MARK: - UIDocumentInteractionControllerDelegate
|
|
|
|
extension ConversationVC: UIDocumentInteractionControllerDelegate {
|
|
func documentInteractionControllerViewControllerForPreview(_ controller: UIDocumentInteractionController) -> UIViewController {
|
|
return self
|
|
}
|
|
}
|
|
|
|
// MARK: - Message Request Actions
|
|
|
|
extension ConversationVC {
|
|
fileprivate func approveMessageRequestIfNeeded(
|
|
for threadId: String,
|
|
threadVariant: SessionThread.Variant,
|
|
isNewThread: Bool,
|
|
timestampMs: Int64
|
|
) {
|
|
guard threadVariant == .contact else { return }
|
|
|
|
let updateNavigationBackStack: () -> Void = {
|
|
// Remove the 'MessageRequestsViewController' from the nav hierarchy if present
|
|
DispatchQueue.main.async { [weak self] in
|
|
if
|
|
let viewControllers: [UIViewController] = self?.navigationController?.viewControllers,
|
|
let messageRequestsIndex = viewControllers.firstIndex(where: { $0 is MessageRequestsViewController }),
|
|
messageRequestsIndex > 0
|
|
{
|
|
var newViewControllers = viewControllers
|
|
newViewControllers.remove(at: messageRequestsIndex)
|
|
self?.navigationController?.viewControllers = newViewControllers
|
|
}
|
|
}
|
|
}
|
|
|
|
// If the contact doesn't exist then we should create it so we can store the 'isApproved' state
|
|
// (it'll be updated with correct profile info if they accept the message request so this
|
|
// shouldn't cause weird behaviours)
|
|
guard
|
|
let contact: Contact = Storage.shared.read({ db in Contact.fetchOrCreate(db, id: threadId) }),
|
|
!contact.isApproved
|
|
else { return }
|
|
|
|
Storage.shared
|
|
.writePublisher { db in
|
|
// If we aren't creating a new thread (ie. sending a message request) then send a
|
|
// messageRequestResponse back to the sender (this allows the sender to know that
|
|
// they have been approved and can now use this contact in closed groups)
|
|
if !isNewThread {
|
|
try MessageSender.send(
|
|
db,
|
|
message: MessageRequestResponse(
|
|
isApproved: true,
|
|
sentTimestampMs: UInt64(timestampMs)
|
|
),
|
|
interactionId: nil,
|
|
threadId: threadId,
|
|
threadVariant: threadVariant
|
|
)
|
|
}
|
|
|
|
// Default 'didApproveMe' to true for the person approving the message request
|
|
try contact.save(db)
|
|
try Contact
|
|
.filter(id: contact.id)
|
|
.updateAllAndConfig(
|
|
db,
|
|
Contact.Columns.isApproved.set(to: true),
|
|
Contact.Columns.didApproveMe
|
|
.set(to: contact.didApproveMe || !isNewThread)
|
|
)
|
|
}
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
.receive(on: DispatchQueue.main)
|
|
.sinkUntilComplete(
|
|
receiveCompletion: { _ in
|
|
// Update the UI
|
|
updateNavigationBackStack()
|
|
}
|
|
)
|
|
}
|
|
|
|
@objc func acceptMessageRequest() {
|
|
self.approveMessageRequestIfNeeded(
|
|
for: self.viewModel.threadData.threadId,
|
|
threadVariant: self.viewModel.threadData.threadVariant,
|
|
isNewThread: false,
|
|
timestampMs: SnodeAPI.currentOffsetTimestampMs()
|
|
)
|
|
}
|
|
|
|
@objc func deleteMessageRequest() {
|
|
let actions: [UIContextualAction]? = UIContextualAction.generateSwipeActions(
|
|
[.delete],
|
|
for: .trailing,
|
|
indexPath: IndexPath(row: 0, section: 0),
|
|
tableView: self.tableView,
|
|
threadViewModel: self.viewModel.threadData,
|
|
viewController: self
|
|
)
|
|
|
|
guard let action: UIContextualAction = actions?.first else { return }
|
|
|
|
action.handler(action, self.view, { [weak self] didConfirm in
|
|
guard didConfirm else { return }
|
|
|
|
self?.stopObservingChanges()
|
|
|
|
DispatchQueue.main.async {
|
|
self?.navigationController?.popViewController(animated: true)
|
|
}
|
|
})
|
|
}
|
|
|
|
@objc func blockMessageRequest() {
|
|
let actions: [UIContextualAction]? = UIContextualAction.generateSwipeActions(
|
|
[.block],
|
|
for: .trailing,
|
|
indexPath: IndexPath(row: 0, section: 0),
|
|
tableView: self.tableView,
|
|
threadViewModel: self.viewModel.threadData,
|
|
viewController: self
|
|
)
|
|
|
|
guard let action: UIContextualAction = actions?.first else { return }
|
|
|
|
action.handler(action, self.view, { [weak self] didConfirm in
|
|
guard didConfirm else { return }
|
|
|
|
self?.stopObservingChanges()
|
|
|
|
DispatchQueue.main.async {
|
|
self?.navigationController?.popViewController(animated: true)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
// MARK: - MediaPresentationContextProvider
|
|
|
|
extension ConversationVC: MediaPresentationContextProvider {
|
|
func mediaPresentationContext(mediaItem: Media, in coordinateSpace: UICoordinateSpace) -> MediaPresentationContext? {
|
|
guard case let .gallery(galleryItem) = mediaItem else { return nil }
|
|
|
|
// Note: According to Apple's docs the 'indexPathsForVisibleRows' method returns an
|
|
// unsorted array which means we can't use it to determine the desired 'visibleCell'
|
|
// we are after, due to this we will need to iterate all of the visible cells to find
|
|
// the one we want
|
|
let maybeMessageCell: VisibleMessageCell? = tableView.visibleCells
|
|
.first { cell -> Bool in
|
|
((cell as? VisibleMessageCell)?
|
|
.albumView?
|
|
.itemViews
|
|
.contains(where: { mediaView in
|
|
mediaView.attachment.id == galleryItem.attachment.id
|
|
}))
|
|
.defaulting(to: false)
|
|
}
|
|
.map { $0 as? VisibleMessageCell }
|
|
let maybeTargetView: MediaView? = maybeMessageCell?
|
|
.albumView?
|
|
.itemViews
|
|
.first(where: { $0.attachment.id == galleryItem.attachment.id })
|
|
|
|
guard
|
|
let messageCell: VisibleMessageCell = maybeMessageCell,
|
|
let targetView: MediaView = maybeTargetView,
|
|
let mediaSuperview: UIView = targetView.superview
|
|
else { return nil }
|
|
|
|
let cornerRadius: CGFloat
|
|
let cornerMask: CACornerMask
|
|
let presentationFrame: CGRect = coordinateSpace.convert(targetView.frame, from: mediaSuperview)
|
|
let frameInBubble: CGRect = messageCell.bubbleView.convert(targetView.frame, from: mediaSuperview)
|
|
|
|
if messageCell.bubbleView.bounds == targetView.bounds {
|
|
cornerRadius = messageCell.bubbleView.layer.cornerRadius
|
|
cornerMask = messageCell.bubbleView.layer.maskedCorners
|
|
}
|
|
else {
|
|
// If the frames don't match then assume it's either multiple images or there is a caption
|
|
// and determine which corners need to be rounded
|
|
cornerRadius = messageCell.bubbleView.layer.cornerRadius
|
|
|
|
var newCornerMask = CACornerMask()
|
|
let cellMaskedCorners: CACornerMask = messageCell.bubbleView.layer.maskedCorners
|
|
|
|
if
|
|
cellMaskedCorners.contains(.layerMinXMinYCorner) &&
|
|
frameInBubble.minX < CGFloat.leastNonzeroMagnitude &&
|
|
frameInBubble.minY < CGFloat.leastNonzeroMagnitude
|
|
{
|
|
newCornerMask.insert(.layerMinXMinYCorner)
|
|
}
|
|
|
|
if
|
|
cellMaskedCorners.contains(.layerMaxXMinYCorner) &&
|
|
abs(frameInBubble.maxX - messageCell.bubbleView.bounds.width) < CGFloat.leastNonzeroMagnitude &&
|
|
frameInBubble.minY < CGFloat.leastNonzeroMagnitude
|
|
{
|
|
newCornerMask.insert(.layerMaxXMinYCorner)
|
|
}
|
|
|
|
if
|
|
cellMaskedCorners.contains(.layerMinXMaxYCorner) &&
|
|
frameInBubble.minX < CGFloat.leastNonzeroMagnitude &&
|
|
abs(frameInBubble.maxY - messageCell.bubbleView.bounds.height) < CGFloat.leastNonzeroMagnitude
|
|
{
|
|
newCornerMask.insert(.layerMinXMaxYCorner)
|
|
}
|
|
|
|
if
|
|
cellMaskedCorners.contains(.layerMaxXMaxYCorner) &&
|
|
abs(frameInBubble.maxX - messageCell.bubbleView.bounds.width) < CGFloat.leastNonzeroMagnitude &&
|
|
abs(frameInBubble.maxY - messageCell.bubbleView.bounds.height) < CGFloat.leastNonzeroMagnitude
|
|
{
|
|
newCornerMask.insert(.layerMaxXMaxYCorner)
|
|
}
|
|
|
|
cornerMask = newCornerMask
|
|
}
|
|
|
|
return MediaPresentationContext(
|
|
mediaView: targetView,
|
|
presentationFrame: presentationFrame,
|
|
cornerRadius: cornerRadius,
|
|
cornerMask: cornerMask
|
|
)
|
|
}
|
|
|
|
func snapshotOverlayView(in coordinateSpace: UICoordinateSpace) -> (UIView, CGRect)? {
|
|
return self.navigationController?.navigationBar.generateSnapshot(in: coordinateSpace)
|
|
}
|
|
}
|