|
|
|
@ -8,24 +8,20 @@ import SessionUIKit
|
|
|
|
|
import SessionUtilitiesKit
|
|
|
|
|
|
|
|
|
|
protocol SwipeActionOptimisticCell {
|
|
|
|
|
func optimisticUpdate(isMuted: Bool?, isBlocked: Bool?, isPinned: Bool?, hasUnread: Bool?)
|
|
|
|
|
func optimisticUpdate(isMuted: Bool?, isPinned: Bool?, hasUnread: Bool?)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
extension SwipeActionOptimisticCell {
|
|
|
|
|
public func optimisticUpdate(isMuted: Bool) {
|
|
|
|
|
optimisticUpdate(isMuted: isMuted, isBlocked: nil, isPinned: nil, hasUnread: nil)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public func optimisticUpdate(isBlocked: Bool) {
|
|
|
|
|
optimisticUpdate(isMuted: nil, isBlocked: isBlocked, isPinned: nil, hasUnread: nil)
|
|
|
|
|
optimisticUpdate(isMuted: isMuted, isPinned: nil, hasUnread: nil)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public func optimisticUpdate(isPinned: Bool) {
|
|
|
|
|
optimisticUpdate(isMuted: nil, isBlocked: nil, isPinned: isPinned, hasUnread: nil)
|
|
|
|
|
optimisticUpdate(isMuted: nil, isPinned: isPinned, hasUnread: nil)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public func optimisticUpdate(hasUnread: Bool) {
|
|
|
|
|
optimisticUpdate(isMuted: nil, isBlocked: nil, isPinned: nil, hasUnread: hasUnread)
|
|
|
|
|
optimisticUpdate(isMuted: nil, isPinned: nil, hasUnread: hasUnread)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -38,6 +34,7 @@ public extension UIContextualAction {
|
|
|
|
|
case block
|
|
|
|
|
case leave
|
|
|
|
|
case delete
|
|
|
|
|
case deleteContact
|
|
|
|
|
case clear
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -370,102 +367,110 @@ public extension UIContextualAction {
|
|
|
|
|
(!threadIsContactMessageRequest ? nil : Contact.Columns.didApproveMe.set(to: true)),
|
|
|
|
|
(!threadIsContactMessageRequest ? nil : Contact.Columns.isApproved.set(to: false))
|
|
|
|
|
].compactMap { $0 }
|
|
|
|
|
let nameToUse: String = {
|
|
|
|
|
switch threadViewModel.threadVariant {
|
|
|
|
|
case .group:
|
|
|
|
|
return Profile.displayName(
|
|
|
|
|
for: .contact,
|
|
|
|
|
id: profileInfo.id,
|
|
|
|
|
name: profileInfo.profile?.name,
|
|
|
|
|
nickname: profileInfo.profile?.nickname,
|
|
|
|
|
suppressId: false
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
default: return threadViewModel.displayName
|
|
|
|
|
}
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
|
|
let performBlock: (UIViewController?) -> () = { viewController in
|
|
|
|
|
(tableView.cellForRow(at: indexPath) as? SwipeActionOptimisticCell)?
|
|
|
|
|
.optimisticUpdate(
|
|
|
|
|
isBlocked: !threadIsBlocked
|
|
|
|
|
)
|
|
|
|
|
completionHandler(true)
|
|
|
|
|
|
|
|
|
|
// Delay the change to give the cell "unswipe" animation some time to complete
|
|
|
|
|
DispatchQueue.global(qos: .default).asyncAfter(deadline: .now() + unswipeAnimationDelay) {
|
|
|
|
|
dependencies[singleton: .storage]
|
|
|
|
|
.writePublisher { db in
|
|
|
|
|
// Create the contact if it doesn't exist
|
|
|
|
|
switch threadViewModel.threadVariant {
|
|
|
|
|
case .contact:
|
|
|
|
|
try Contact
|
|
|
|
|
.fetchOrCreate(db, id: threadViewModel.threadId, using: dependencies)
|
|
|
|
|
.upsert(db)
|
|
|
|
|
try Contact
|
|
|
|
|
.filter(id: threadViewModel.threadId)
|
|
|
|
|
.updateAllAndConfig(
|
|
|
|
|
db,
|
|
|
|
|
contactChanges,
|
|
|
|
|
using: dependencies
|
|
|
|
|
)
|
|
|
|
|
let confirmationModal: ConfirmationModal = ConfirmationModal(
|
|
|
|
|
info: ConfirmationModal.Info(
|
|
|
|
|
title: (threadIsBlocked ?
|
|
|
|
|
"blockUnblock".localized() :
|
|
|
|
|
"block".localized()
|
|
|
|
|
),
|
|
|
|
|
body: (threadIsBlocked ?
|
|
|
|
|
.attributedText(
|
|
|
|
|
"blockUnblockName"
|
|
|
|
|
.put(key: "name", value: nameToUse)
|
|
|
|
|
.localizedFormatted(baseFont: ConfirmationModal.explanationFont)
|
|
|
|
|
) :
|
|
|
|
|
.attributedText(
|
|
|
|
|
"blockDescription"
|
|
|
|
|
.put(key: "name", value: nameToUse)
|
|
|
|
|
.localizedFormatted(baseFont: ConfirmationModal.explanationFont)
|
|
|
|
|
)
|
|
|
|
|
),
|
|
|
|
|
confirmTitle: (threadIsBlocked ?
|
|
|
|
|
"blockUnblock".localized() :
|
|
|
|
|
"block".localized()
|
|
|
|
|
),
|
|
|
|
|
confirmStyle: .danger,
|
|
|
|
|
cancelStyle: .alert_text,
|
|
|
|
|
dismissOnConfirm: true,
|
|
|
|
|
onConfirm: { _ in
|
|
|
|
|
completionHandler(true)
|
|
|
|
|
|
|
|
|
|
// Delay the change to give the cell "unswipe" animation some time to complete
|
|
|
|
|
DispatchQueue.global(qos: .default).asyncAfter(deadline: .now() + unswipeAnimationDelay) {
|
|
|
|
|
dependencies[singleton: .storage]
|
|
|
|
|
.writePublisher { db in
|
|
|
|
|
// Create the contact if it doesn't exist
|
|
|
|
|
switch threadViewModel.threadVariant {
|
|
|
|
|
case .contact:
|
|
|
|
|
try Contact
|
|
|
|
|
.fetchOrCreate(
|
|
|
|
|
db,
|
|
|
|
|
id: threadViewModel.threadId,
|
|
|
|
|
using: dependencies
|
|
|
|
|
)
|
|
|
|
|
.upsert(db)
|
|
|
|
|
try Contact
|
|
|
|
|
.filter(id: threadViewModel.threadId)
|
|
|
|
|
.updateAllAndConfig(
|
|
|
|
|
db,
|
|
|
|
|
contactChanges,
|
|
|
|
|
using: dependencies
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
case .group:
|
|
|
|
|
try Contact
|
|
|
|
|
.fetchOrCreate(
|
|
|
|
|
db,
|
|
|
|
|
id: profileInfo.id,
|
|
|
|
|
using: dependencies
|
|
|
|
|
)
|
|
|
|
|
.upsert(db)
|
|
|
|
|
try Contact
|
|
|
|
|
.filter(id: profileInfo.id)
|
|
|
|
|
.updateAllAndConfig(
|
|
|
|
|
db,
|
|
|
|
|
contactChanges,
|
|
|
|
|
using: dependencies
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
default: break
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case .group:
|
|
|
|
|
try Contact
|
|
|
|
|
.fetchOrCreate(db, id: profileInfo.id, using: dependencies)
|
|
|
|
|
.upsert(db)
|
|
|
|
|
try Contact
|
|
|
|
|
.filter(id: profileInfo.id)
|
|
|
|
|
.updateAllAndConfig(
|
|
|
|
|
// Blocked message requests should be deleted
|
|
|
|
|
if threadViewModel.threadIsMessageRequest == true {
|
|
|
|
|
try SessionThread.deleteOrLeave(
|
|
|
|
|
db,
|
|
|
|
|
contactChanges,
|
|
|
|
|
type: .deleteContactConversationAndMarkHidden,
|
|
|
|
|
threadId: threadViewModel.threadId,
|
|
|
|
|
threadVariant: threadViewModel.threadVariant,
|
|
|
|
|
using: dependencies
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
default: break
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Blocked message requests should be deleted
|
|
|
|
|
if threadViewModel.threadIsMessageRequest == true {
|
|
|
|
|
try SessionThread.deleteOrLeave(
|
|
|
|
|
db,
|
|
|
|
|
type: .deleteContactConversationAndMarkHidden,
|
|
|
|
|
threadId: threadViewModel.threadId,
|
|
|
|
|
threadVariant: threadViewModel.threadVariant,
|
|
|
|
|
using: dependencies
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
|
|
|
.sinkUntilComplete()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch threadViewModel.threadIsMessageRequest == true {
|
|
|
|
|
case false: performBlock(nil)
|
|
|
|
|
case true:
|
|
|
|
|
let nameToUse: String = {
|
|
|
|
|
switch threadViewModel.threadVariant {
|
|
|
|
|
case .group:
|
|
|
|
|
return Profile.displayName(
|
|
|
|
|
for: .contact,
|
|
|
|
|
id: profileInfo.id,
|
|
|
|
|
name: profileInfo.profile?.name,
|
|
|
|
|
nickname: profileInfo.profile?.nickname,
|
|
|
|
|
suppressId: false
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
default: return threadViewModel.displayName
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
.subscribe(on: DispatchQueue.global(qos: .userInitiated))
|
|
|
|
|
.sinkUntilComplete()
|
|
|
|
|
}
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
|
|
let confirmationModal: ConfirmationModal = ConfirmationModal(
|
|
|
|
|
info: ConfirmationModal.Info(
|
|
|
|
|
title: "block".localized(),
|
|
|
|
|
body: .attributedText(
|
|
|
|
|
"blockDescription"
|
|
|
|
|
.put(key: "name", value: nameToUse)
|
|
|
|
|
.localizedFormatted(baseFont: .systemFont(ofSize: Values.smallFontSize))
|
|
|
|
|
),
|
|
|
|
|
confirmTitle: "block".localized(),
|
|
|
|
|
confirmStyle: .danger,
|
|
|
|
|
cancelStyle: .alert_text,
|
|
|
|
|
dismissOnConfirm: true,
|
|
|
|
|
onConfirm: { _ in
|
|
|
|
|
performBlock(viewController)
|
|
|
|
|
},
|
|
|
|
|
afterClosed: { completionHandler(false) }
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
viewController?.present(confirmationModal, animated: true, completion: nil)
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
afterClosed: { completionHandler(false) }
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
viewController?.present(confirmationModal, animated: true, completion: nil)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// MARK: -- leave
|
|
|
|
@ -671,6 +676,52 @@ public extension UIContextualAction {
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
viewController?.present(confirmationModal, animated: true, completion: nil)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// MARK: -- deleteContact
|
|
|
|
|
|
|
|
|
|
case .deleteContact:
|
|
|
|
|
return UIContextualAction(
|
|
|
|
|
title: "contactDelete".localized(),
|
|
|
|
|
icon: Lucide.image(icon: .trash2, size: 24, color: .white),
|
|
|
|
|
themeTintColor: .white,
|
|
|
|
|
themeBackgroundColor: themeBackgroundColor,
|
|
|
|
|
accessibility: Accessibility(identifier: "Delete button"),
|
|
|
|
|
side: side,
|
|
|
|
|
actionIndex: targetIndex,
|
|
|
|
|
indexPath: indexPath,
|
|
|
|
|
tableView: tableView
|
|
|
|
|
) { [weak viewController] _, _, completionHandler in
|
|
|
|
|
let confirmationModal: ConfirmationModal = ConfirmationModal(
|
|
|
|
|
info: ConfirmationModal.Info(
|
|
|
|
|
title: "contactDelete".localized(),
|
|
|
|
|
body: .attributedText(
|
|
|
|
|
"contactDeleteDescription"
|
|
|
|
|
.put(key: "name", value: threadViewModel.displayName)
|
|
|
|
|
.localizedFormatted(baseFont: .boldSystemFont(ofSize: Values.smallFontSize))
|
|
|
|
|
),
|
|
|
|
|
confirmTitle: "delete".localized(),
|
|
|
|
|
confirmStyle: .danger,
|
|
|
|
|
cancelStyle: .alert_text,
|
|
|
|
|
dismissOnConfirm: true,
|
|
|
|
|
onConfirm: { _ in
|
|
|
|
|
dependencies[singleton: .storage].writeAsync { db in
|
|
|
|
|
try SessionThread.deleteOrLeave(
|
|
|
|
|
db,
|
|
|
|
|
type: .deleteContactConversationAndContact,
|
|
|
|
|
threadId: threadViewModel.threadId,
|
|
|
|
|
threadVariant: threadViewModel.threadVariant,
|
|
|
|
|
using: dependencies
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
completionHandler(true)
|
|
|
|
|
},
|
|
|
|
|
afterClosed: { completionHandler(false) }
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
viewController?.present(confirmationModal, animated: true, completion: nil)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|