2022-01-24 16:07:17 +00:00
|
|
|
//
|
|
|
|
// ChatModel.swift
|
|
|
|
// SimpleX
|
|
|
|
//
|
|
|
|
// Created by Evgeny Poberezkin on 22/01/2022.
|
|
|
|
// Copyright © 2022 SimpleX Chat. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import Combine
|
2022-01-31 21:28:07 +00:00
|
|
|
import SwiftUI
|
2022-05-24 19:34:27 +01:00
|
|
|
import WebKit
|
2022-05-31 07:55:13 +01:00
|
|
|
import SimpleXChat
|
2022-01-24 16:07:17 +00:00
|
|
|
|
|
|
|
final class ChatModel: ObservableObject {
|
2022-05-09 09:52:09 +01:00
|
|
|
@Published var onboardingStage: OnboardingStage?
|
2022-07-01 22:45:58 +01:00
|
|
|
@Published var v3DBMigration: V3DBMigrationState = v3DBMigrationDefault.get()
|
2022-01-24 16:07:17 +00:00
|
|
|
@Published var currentUser: User?
|
2022-06-24 13:52:20 +01:00
|
|
|
@Published var chatRunning: Bool?
|
|
|
|
@Published var chatDbChanged = false
|
2022-02-02 12:51:39 +00:00
|
|
|
// list of chat "previews"
|
|
|
|
@Published var chats: [Chat] = []
|
|
|
|
// current chat
|
|
|
|
@Published var chatId: String?
|
2022-01-29 11:10:04 +00:00
|
|
|
@Published var chatItems: [ChatItem] = []
|
2022-02-12 15:59:43 +00:00
|
|
|
@Published var chatToTop: String?
|
2022-02-02 12:51:39 +00:00
|
|
|
// items in the terminal view
|
2022-01-29 23:37:02 +00:00
|
|
|
@Published var terminalItems: [TerminalItem] = []
|
2022-02-01 17:34:06 +00:00
|
|
|
@Published var userAddress: String?
|
2022-03-10 15:45:40 +04:00
|
|
|
@Published var userSMPServers: [String]?
|
2022-02-01 20:30:33 +00:00
|
|
|
@Published var appOpenUrl: URL?
|
2022-06-27 23:03:27 +01:00
|
|
|
@Published var deviceToken: DeviceToken?
|
2022-07-01 09:49:30 +01:00
|
|
|
@Published var savedToken: DeviceToken?
|
2022-07-09 09:29:56 +01:00
|
|
|
@Published var tokenRegistered = false
|
2022-07-01 09:49:30 +01:00
|
|
|
@Published var tokenStatus: NtfTknStatus?
|
|
|
|
@Published var notificationMode = NotificationsMode.off
|
2022-07-06 11:52:10 +01:00
|
|
|
@Published var notificationPreview: NotificationPreviewMode? = ntfPreviewModeGroupDefault.get()
|
2022-07-22 08:10:37 +01:00
|
|
|
// pending notification actions
|
|
|
|
@Published var ntfContactRequest: ChatId?
|
|
|
|
@Published var ntfCallInvitationAction: (ChatId, NtfCallAction)?
|
2022-05-07 06:40:46 +01:00
|
|
|
// current WebRTC call
|
2022-07-05 15:15:15 +04:00
|
|
|
@Published var callInvitations: Dictionary<ChatId, RcvCallInvitation> = [:]
|
2022-05-07 06:40:46 +01:00
|
|
|
@Published var activeCall: Call?
|
|
|
|
@Published var callCommand: WCallCommand?
|
2022-05-21 12:13:37 +01:00
|
|
|
@Published var showCallView = false
|
2022-05-24 19:34:27 +01:00
|
|
|
var callWebView: WKWebView?
|
2022-02-28 10:44:48 +00:00
|
|
|
|
|
|
|
var messageDelivery: Dictionary<Int64, () -> Void> = [:]
|
|
|
|
|
2022-02-09 22:53:06 +00:00
|
|
|
static let shared = ChatModel()
|
2022-02-02 12:51:39 +00:00
|
|
|
|
|
|
|
func hasChat(_ id: String) -> Bool {
|
|
|
|
chats.first(where: { $0.id == id }) != nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getChat(_ id: String) -> Chat? {
|
|
|
|
chats.first(where: { $0.id == id })
|
|
|
|
}
|
|
|
|
|
2022-02-05 20:10:47 +00:00
|
|
|
private func getChatIndex(_ id: String) -> Int? {
|
|
|
|
chats.firstIndex(where: { $0.id == id })
|
|
|
|
}
|
|
|
|
|
2022-07-10 14:28:00 +01:00
|
|
|
func addChat(_ chat: Chat, at position: Int = 0) {
|
2022-02-02 16:46:05 +00:00
|
|
|
withAnimation {
|
2022-07-10 14:28:00 +01:00
|
|
|
chats.insert(chat, at: position)
|
2022-02-02 16:46:05 +00:00
|
|
|
}
|
2022-02-02 12:51:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func updateChatInfo(_ cInfo: ChatInfo) {
|
2022-02-12 15:59:43 +00:00
|
|
|
if let i = getChatIndex(cInfo.id) {
|
|
|
|
chats[i].chatInfo = cInfo
|
2022-02-02 12:51:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-25 10:39:28 +01:00
|
|
|
func updateContactConnection(_ contactConnection: PendingContactConnection) {
|
|
|
|
updateChat(.contactConnection(contactConnection: contactConnection))
|
|
|
|
}
|
|
|
|
|
2022-02-05 20:10:47 +00:00
|
|
|
func updateContact(_ contact: Contact) {
|
2022-07-18 21:58:32 +04:00
|
|
|
updateChat(.direct(contact: contact), addMissing: !contact.isIndirectContact())
|
2022-04-25 10:39:28 +01:00
|
|
|
}
|
|
|
|
|
2022-07-18 21:58:32 +04:00
|
|
|
func updateGroup(_ groupInfo: GroupInfo) {
|
|
|
|
updateChat(.group(groupInfo: groupInfo))
|
|
|
|
}
|
|
|
|
|
|
|
|
private func updateChat(_ cInfo: ChatInfo, addMissing: Bool = true) {
|
2022-04-25 10:39:28 +01:00
|
|
|
if hasChat(cInfo.id) {
|
2022-02-05 20:10:47 +00:00
|
|
|
updateChatInfo(cInfo)
|
2022-07-18 21:58:32 +04:00
|
|
|
} else if addMissing {
|
2022-02-05 20:10:47 +00:00
|
|
|
addChat(Chat(chatInfo: cInfo, chatItems: []))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-26 07:51:06 +01:00
|
|
|
func updateNetworkStatus(_ id: ChatId, _ status: Chat.NetworkStatus) {
|
|
|
|
if let i = getChatIndex(id) {
|
|
|
|
chats[i].serverInfo.networkStatus = status
|
2022-02-05 20:10:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-02 12:51:39 +00:00
|
|
|
func replaceChat(_ id: String, _ chat: Chat) {
|
2022-02-12 15:59:43 +00:00
|
|
|
if let i = getChatIndex(id) {
|
2022-07-01 22:45:58 +01:00
|
|
|
let serverInfo = chats[i].serverInfo
|
2022-02-12 15:59:43 +00:00
|
|
|
chats[i] = chat
|
2022-07-01 22:45:58 +01:00
|
|
|
chats[i].serverInfo = serverInfo
|
2022-02-02 12:51:39 +00:00
|
|
|
} else {
|
|
|
|
// invalid state, correcting
|
|
|
|
chats.insert(chat, at: 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-02 17:18:45 +01:00
|
|
|
func updateChats(with newChats: [ChatData]) {
|
2022-07-10 14:28:00 +01:00
|
|
|
for i in 0..<newChats.count {
|
|
|
|
let c = newChats[i]
|
|
|
|
if let j = getChatIndex(c.id) {
|
|
|
|
let chat = chats[j]
|
2022-07-02 17:18:45 +01:00
|
|
|
chat.chatInfo = c.chatInfo
|
|
|
|
chat.chatItems = c.chatItems
|
|
|
|
chat.chatStats = c.chatStats
|
2022-07-10 14:28:00 +01:00
|
|
|
if i != j {
|
|
|
|
if chatId != c.chatInfo.id {
|
|
|
|
popChat_(j, to: i)
|
|
|
|
} else if i == 0 {
|
|
|
|
chatToTop = c.chatInfo.id
|
|
|
|
}
|
|
|
|
}
|
2022-07-02 17:18:45 +01:00
|
|
|
} else {
|
2022-07-10 14:28:00 +01:00
|
|
|
addChat(Chat(c), at: i)
|
2022-07-02 17:18:45 +01:00
|
|
|
}
|
2022-07-01 22:45:58 +01:00
|
|
|
}
|
2022-07-20 08:58:53 +01:00
|
|
|
NtfManager.shared.setNtfBadgeCount(totalUnreadCount())
|
2022-07-01 22:45:58 +01:00
|
|
|
}
|
|
|
|
|
2022-07-14 16:40:32 +04:00
|
|
|
// func addGroup(_ group: SimpleXChat.Group) {
|
|
|
|
// groups[group.groupInfo.id] = group
|
|
|
|
// }
|
|
|
|
|
2022-02-02 12:51:39 +00:00
|
|
|
func addChatItem(_ cInfo: ChatInfo, _ cItem: ChatItem) {
|
2022-02-12 15:59:43 +00:00
|
|
|
// update previews
|
|
|
|
if let i = getChatIndex(cInfo.id) {
|
|
|
|
chats[i].chatItems = [cItem]
|
|
|
|
if case .rcvNew = cItem.meta.itemStatus {
|
|
|
|
chats[i].chatStats.unreadCount = chats[i].chatStats.unreadCount + 1
|
2022-07-20 08:58:53 +01:00
|
|
|
NtfManager.shared.incNtfBadgeCount()
|
2022-02-12 15:59:43 +00:00
|
|
|
}
|
|
|
|
if i > 0 {
|
2022-02-05 14:24:23 +00:00
|
|
|
if chatId == nil {
|
2022-02-12 15:59:43 +00:00
|
|
|
withAnimation { popChat_(i) }
|
|
|
|
} else if chatId == cInfo.id {
|
|
|
|
chatToTop = cInfo.id
|
2022-02-05 14:24:23 +00:00
|
|
|
} else {
|
2022-02-12 15:59:43 +00:00
|
|
|
popChat_(i)
|
2022-02-05 14:24:23 +00:00
|
|
|
}
|
2022-02-02 12:51:39 +00:00
|
|
|
}
|
2022-02-12 15:59:43 +00:00
|
|
|
} else {
|
|
|
|
addChat(Chat(chatInfo: cInfo, chatItems: [cItem]))
|
2022-02-02 12:51:39 +00:00
|
|
|
}
|
2022-02-12 15:59:43 +00:00
|
|
|
// add to current chat
|
2022-02-02 12:51:39 +00:00
|
|
|
if chatId == cInfo.id {
|
2022-02-03 07:16:29 +00:00
|
|
|
withAnimation { chatItems.append(cItem) }
|
2022-02-12 15:59:43 +00:00
|
|
|
if case .rcvNew = cItem.meta.itemStatus {
|
|
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
|
|
|
|
if self.chatId == cInfo.id {
|
2022-08-04 22:25:52 +01:00
|
|
|
Task {
|
|
|
|
await apiMarkChatItemRead(cInfo, cItem)
|
|
|
|
NtfManager.shared.decNtfBadgeCount()
|
|
|
|
}
|
2022-02-12 15:59:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func upsertChatItem(_ cInfo: ChatInfo, _ cItem: ChatItem) -> Bool {
|
|
|
|
// update previews
|
|
|
|
var res: Bool
|
|
|
|
if let chat = getChat(cInfo.id) {
|
|
|
|
if let pItem = chat.chatItems.last, pItem.id == cItem.id {
|
|
|
|
chat.chatItems = [cItem]
|
|
|
|
}
|
|
|
|
res = false
|
|
|
|
} else {
|
|
|
|
addChat(Chat(chatInfo: cInfo, chatItems: [cItem]))
|
|
|
|
res = true
|
|
|
|
}
|
|
|
|
// update current chat
|
|
|
|
if chatId == cInfo.id {
|
|
|
|
if let i = chatItems.firstIndex(where: { $0.id == cItem.id }) {
|
|
|
|
withAnimation(.default) {
|
|
|
|
self.chatItems[i] = cItem
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
} else {
|
|
|
|
withAnimation { chatItems.append(cItem) }
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
}
|
2022-03-30 20:37:47 +04:00
|
|
|
|
|
|
|
func removeChatItem(_ cInfo: ChatInfo, _ cItem: ChatItem) {
|
|
|
|
// update previews
|
|
|
|
if let chat = getChat(cInfo.id) {
|
|
|
|
if let pItem = chat.chatItems.last, pItem.id == cItem.id {
|
|
|
|
chat.chatItems = [cItem]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// remove from current chat
|
|
|
|
if chatId == cInfo.id {
|
|
|
|
if let i = chatItems.firstIndex(where: { $0.id == cItem.id }) {
|
2022-07-20 08:58:53 +01:00
|
|
|
if chatItems[i].isRcvNew() == true {
|
|
|
|
NtfManager.shared.decNtfBadgeCount()
|
|
|
|
}
|
2022-03-30 20:37:47 +04:00
|
|
|
_ = withAnimation {
|
|
|
|
self.chatItems.remove(at: i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-02-12 15:59:43 +00:00
|
|
|
|
|
|
|
func markChatItemsRead(_ cInfo: ChatInfo) {
|
|
|
|
// update preview
|
|
|
|
if let chat = getChat(cInfo.id) {
|
2022-07-20 08:58:53 +01:00
|
|
|
NtfManager.shared.decNtfBadgeCount(by: chat.chatStats.unreadCount)
|
2022-02-12 15:59:43 +00:00
|
|
|
chat.chatStats = ChatStats()
|
|
|
|
}
|
|
|
|
// update current chat
|
|
|
|
if chatId == cInfo.id {
|
|
|
|
var i = 0
|
|
|
|
while i < chatItems.count {
|
|
|
|
if case .rcvNew = chatItems[i].meta.itemStatus {
|
|
|
|
chatItems[i].meta.itemStatus = .rcvRead
|
|
|
|
}
|
|
|
|
i = i + 1
|
|
|
|
}
|
2022-02-02 12:51:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-17 22:48:54 +04:00
|
|
|
func clearChat(_ cInfo: ChatInfo) {
|
|
|
|
// clear preview
|
|
|
|
if let chat = getChat(cInfo.id) {
|
2022-07-20 08:58:53 +01:00
|
|
|
NtfManager.shared.decNtfBadgeCount(by: chat.chatStats.unreadCount)
|
2022-05-17 22:48:54 +04:00
|
|
|
chat.chatItems = []
|
|
|
|
chat.chatStats = ChatStats()
|
2022-05-20 12:00:58 +04:00
|
|
|
chat.chatInfo = cInfo
|
2022-05-17 22:48:54 +04:00
|
|
|
}
|
|
|
|
// clear current chat
|
|
|
|
if chatId == cInfo.id {
|
|
|
|
chatItems = []
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-12 15:59:43 +00:00
|
|
|
func markChatItemRead(_ cInfo: ChatInfo, _ cItem: ChatItem) {
|
|
|
|
// update preview
|
|
|
|
if let i = getChatIndex(cInfo.id) {
|
|
|
|
chats[i].chatStats.unreadCount = chats[i].chatStats.unreadCount - 1
|
|
|
|
}
|
|
|
|
// update current chat
|
|
|
|
if chatId == cInfo.id, let j = chatItems.firstIndex(where: { $0.id == cItem.id }) {
|
|
|
|
chatItems[j].meta.itemStatus = .rcvRead
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-20 08:58:53 +01:00
|
|
|
func totalUnreadCount() -> Int {
|
|
|
|
chats.reduce(0, { count, chat in count + chat.chatStats.unreadCount })
|
|
|
|
}
|
|
|
|
|
2022-03-30 08:57:42 +01:00
|
|
|
func getPrevChatItem(_ ci: ChatItem) -> ChatItem? {
|
|
|
|
if let i = chatItems.firstIndex(where: { $0.id == ci.id }), i > 0 {
|
|
|
|
return chatItems[i - 1]
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-12 15:59:43 +00:00
|
|
|
func popChat(_ id: String) {
|
|
|
|
if let i = getChatIndex(id) {
|
|
|
|
popChat_(i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-10 14:28:00 +01:00
|
|
|
private func popChat_(_ i: Int, to position: Int = 0) {
|
2022-02-12 15:59:43 +00:00
|
|
|
let chat = chats.remove(at: i)
|
2022-07-10 14:28:00 +01:00
|
|
|
chats.insert(chat, at: position)
|
2022-02-05 14:24:23 +00:00
|
|
|
}
|
|
|
|
|
2022-02-02 12:51:39 +00:00
|
|
|
func removeChat(_ id: String) {
|
2022-02-02 16:46:05 +00:00
|
|
|
withAnimation {
|
|
|
|
chats.removeAll(where: { $0.id == id })
|
|
|
|
}
|
2022-02-02 12:51:39 +00:00
|
|
|
}
|
2022-01-24 16:07:17 +00:00
|
|
|
}
|
|
|
|
|
2022-02-02 12:51:39 +00:00
|
|
|
final class Chat: ObservableObject, Identifiable {
|
|
|
|
@Published var chatInfo: ChatInfo
|
|
|
|
@Published var chatItems: [ChatItem]
|
2022-02-12 15:59:43 +00:00
|
|
|
@Published var chatStats: ChatStats
|
2022-02-05 20:10:47 +00:00
|
|
|
@Published var serverInfo = ServerInfo(networkStatus: .unknown)
|
2022-07-01 22:45:58 +01:00
|
|
|
var created = Date.now
|
2022-02-05 20:10:47 +00:00
|
|
|
|
|
|
|
struct ServerInfo: Decodable {
|
|
|
|
var networkStatus: NetworkStatus
|
|
|
|
}
|
|
|
|
|
|
|
|
enum NetworkStatus: Decodable, Equatable {
|
|
|
|
case unknown
|
|
|
|
case connected
|
|
|
|
case disconnected
|
|
|
|
case error(String)
|
|
|
|
|
2022-04-16 09:37:01 +01:00
|
|
|
var statusString: LocalizedStringKey {
|
2022-02-05 20:10:47 +00:00
|
|
|
get {
|
|
|
|
switch self {
|
2022-07-27 13:40:26 +04:00
|
|
|
case .connected: return "connected"
|
|
|
|
case .error: return "error"
|
|
|
|
default: return "connecting"
|
2022-02-07 10:36:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-16 09:37:01 +01:00
|
|
|
var statusExplanation: LocalizedStringKey {
|
2022-02-07 10:36:11 +00:00
|
|
|
get {
|
|
|
|
switch self {
|
2022-03-14 20:58:19 +00:00
|
|
|
case .connected: return "You are connected to the server used to receive messages from this contact."
|
|
|
|
case let .error(err): return "Trying to connect to the server used to receive messages from this contact (error: \(err))."
|
|
|
|
default: return "Trying to connect to the server used to receive messages from this contact."
|
2022-02-05 20:10:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var imageName: String {
|
|
|
|
get {
|
|
|
|
switch self {
|
|
|
|
case .unknown: return "circle.dotted"
|
|
|
|
case .connected: return "circle.fill"
|
|
|
|
case .disconnected: return "ellipsis.circle.fill"
|
|
|
|
case .error: return "exclamationmark.circle.fill"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-02-02 12:51:39 +00:00
|
|
|
|
|
|
|
init(_ cData: ChatData) {
|
|
|
|
self.chatInfo = cData.chatInfo
|
|
|
|
self.chatItems = cData.chatItems
|
2022-02-12 15:59:43 +00:00
|
|
|
self.chatStats = cData.chatStats
|
2022-02-02 12:51:39 +00:00
|
|
|
}
|
2022-01-29 23:37:02 +00:00
|
|
|
|
2022-05-10 08:04:18 +01:00
|
|
|
init(chatInfo: ChatInfo, chatItems: [ChatItem] = [], chatStats: ChatStats = ChatStats(), serverInfo: ServerInfo = ServerInfo(networkStatus: .unknown)) {
|
2022-01-29 23:37:02 +00:00
|
|
|
self.chatInfo = chatInfo
|
|
|
|
self.chatItems = chatItems
|
2022-02-12 15:59:43 +00:00
|
|
|
self.chatStats = chatStats
|
2022-05-10 08:04:18 +01:00
|
|
|
self.serverInfo = serverInfo
|
2022-01-29 23:37:02 +00:00
|
|
|
}
|
2022-01-31 21:28:07 +00:00
|
|
|
|
2022-02-09 22:53:06 +00:00
|
|
|
var id: ChatId { get { chatInfo.id } }
|
2022-07-01 22:45:58 +01:00
|
|
|
|
|
|
|
var viewId: String { get { "\(chatInfo.id) \(created.timeIntervalSince1970)" } }
|
2022-01-29 11:10:04 +00:00
|
|
|
}
|