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-01-24 16:07:17 +00:00
|
|
|
|
|
|
|
final class ChatModel: ObservableObject {
|
|
|
|
@Published var currentUser: User?
|
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-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-02-01 20:30:33 +00:00
|
|
|
@Published var appOpenUrl: URL?
|
|
|
|
@Published var connectViaUrl = false
|
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 })
|
|
|
|
}
|
|
|
|
|
|
|
|
func addChat(_ chat: Chat) {
|
2022-02-02 16:46:05 +00:00
|
|
|
withAnimation {
|
|
|
|
chats.insert(chat, at: 0)
|
|
|
|
}
|
2022-02-02 12:51:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func updateChatInfo(_ cInfo: ChatInfo) {
|
|
|
|
if let ix = chats.firstIndex(where: { $0.id == cInfo.id }) {
|
|
|
|
chats[ix].chatInfo = cInfo
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func replaceChat(_ id: String, _ chat: Chat) {
|
|
|
|
if let ix = chats.firstIndex(where: { $0.id == id }) {
|
|
|
|
chats[ix] = chat
|
|
|
|
} else {
|
|
|
|
// invalid state, correcting
|
|
|
|
chats.insert(chat, at: 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func addChatItem(_ cInfo: ChatInfo, _ cItem: ChatItem) {
|
|
|
|
if let ix = chats.firstIndex(where: { $0.id == cInfo.id }) {
|
|
|
|
chats[ix].chatItems = [cItem]
|
|
|
|
if chatId != cInfo.id {
|
|
|
|
let chat = chats.remove(at: ix)
|
|
|
|
chats.insert(chat, at: 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if chatId == cInfo.id {
|
|
|
|
chatItems.append(cItem)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
struct User: Decodable {
|
2022-01-24 16:07:17 +00:00
|
|
|
var userId: Int64
|
|
|
|
var userContactId: Int64
|
|
|
|
var localDisplayName: ContactName
|
|
|
|
var profile: Profile
|
|
|
|
var activeUser: Bool
|
2022-01-31 21:28:07 +00:00
|
|
|
|
2022-02-02 12:51:39 +00:00
|
|
|
// internal init(userId: Int64, userContactId: Int64, localDisplayName: ContactName, profile: Profile, activeUser: Bool) {
|
|
|
|
// self.userId = userId
|
|
|
|
// self.userContactId = userContactId
|
|
|
|
// self.localDisplayName = localDisplayName
|
|
|
|
// self.profile = profile
|
|
|
|
// self.activeUser = activeUser
|
|
|
|
// }
|
2022-01-24 16:07:17 +00:00
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
let sampleUser = User(
|
|
|
|
userId: 1,
|
|
|
|
userContactId: 1,
|
|
|
|
localDisplayName: "alice",
|
|
|
|
profile: sampleProfile,
|
|
|
|
activeUser: true
|
|
|
|
)
|
|
|
|
|
2022-01-24 16:07:17 +00:00
|
|
|
typealias ContactName = String
|
|
|
|
|
|
|
|
typealias GroupName = String
|
|
|
|
|
|
|
|
struct Profile: Codable {
|
|
|
|
var displayName: String
|
|
|
|
var fullName: String
|
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
let sampleProfile = Profile(
|
|
|
|
displayName: "alice",
|
|
|
|
fullName: "Alice"
|
|
|
|
)
|
|
|
|
|
|
|
|
enum ChatType: String {
|
2022-01-30 18:27:20 +00:00
|
|
|
case direct = "@"
|
|
|
|
case group = "#"
|
2022-02-01 17:34:06 +00:00
|
|
|
case contactRequest = "<@"
|
2022-01-29 23:37:02 +00:00
|
|
|
}
|
|
|
|
|
2022-02-01 17:34:06 +00:00
|
|
|
enum ChatInfo: Identifiable, Decodable {
|
2022-01-29 11:10:04 +00:00
|
|
|
case direct(contact: Contact)
|
2022-01-30 00:35:20 +04:00
|
|
|
case group(groupInfo: GroupInfo)
|
2022-02-01 17:34:06 +00:00
|
|
|
case contactRequest(contactRequest: UserContactRequest)
|
2022-01-29 11:10:04 +00:00
|
|
|
|
2022-01-30 18:27:20 +00:00
|
|
|
var localDisplayName: String {
|
2022-01-29 11:10:04 +00:00
|
|
|
get {
|
|
|
|
switch self {
|
|
|
|
case let .direct(contact): return "@\(contact.localDisplayName)"
|
2022-01-30 00:35:20 +04:00
|
|
|
case let .group(groupInfo): return "#\(groupInfo.localDisplayName)"
|
2022-02-01 17:34:06 +00:00
|
|
|
case let .contactRequest(contactRequest): return "< @\(contactRequest.localDisplayName)"
|
2022-01-29 11:10:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var id: String {
|
|
|
|
get {
|
|
|
|
switch self {
|
2022-02-01 17:34:06 +00:00
|
|
|
case let .direct(contact): return contact.id
|
|
|
|
case let .group(groupInfo): return groupInfo.id
|
|
|
|
case let .contactRequest(contactRequest): return contactRequest.id
|
2022-01-29 11:10:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
var chatType: ChatType {
|
2022-01-29 11:10:04 +00:00
|
|
|
get {
|
|
|
|
switch self {
|
2022-01-29 23:37:02 +00:00
|
|
|
case .direct: return .direct
|
|
|
|
case .group: return .group
|
2022-02-01 17:34:06 +00:00
|
|
|
case .contactRequest: return .contactRequest
|
2022-01-29 11:10:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var apiId: Int64 {
|
|
|
|
get {
|
|
|
|
switch self {
|
2022-02-02 12:51:39 +00:00
|
|
|
case let .direct(contact): return contact.apiId
|
|
|
|
case let .group(groupInfo): return groupInfo.apiId
|
|
|
|
case let .contactRequest(contactRequest): return contactRequest.apiId
|
2022-01-29 11:10:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
let sampleDirectChatInfo = ChatInfo.direct(contact: sampleContact)
|
|
|
|
|
|
|
|
let sampleGroupChatInfo = ChatInfo.group(groupInfo: sampleGroupInfo)
|
|
|
|
|
2022-02-01 17:34:06 +00:00
|
|
|
let sampleContactRequestChatInfo = ChatInfo.contactRequest(contactRequest: sampleContactRequest)
|
|
|
|
|
2022-02-02 12:51:39 +00:00
|
|
|
final class Chat: ObservableObject, Identifiable {
|
|
|
|
@Published var chatInfo: ChatInfo
|
|
|
|
@Published var chatItems: [ChatItem]
|
|
|
|
|
|
|
|
init(_ cData: ChatData) {
|
|
|
|
self.chatInfo = cData.chatInfo
|
|
|
|
self.chatItems = cData.chatItems
|
|
|
|
}
|
2022-01-29 23:37:02 +00:00
|
|
|
|
2022-02-02 12:51:39 +00:00
|
|
|
init(chatInfo: ChatInfo, chatItems: [ChatItem] = []) {
|
2022-01-29 23:37:02 +00:00
|
|
|
self.chatInfo = chatInfo
|
|
|
|
self.chatItems = chatItems
|
|
|
|
}
|
2022-01-31 21:28:07 +00:00
|
|
|
|
|
|
|
var id: String { get { chatInfo.id } }
|
2022-01-29 11:10:04 +00:00
|
|
|
}
|
|
|
|
|
2022-02-02 12:51:39 +00:00
|
|
|
struct ChatData: Decodable, Identifiable {
|
|
|
|
var chatInfo: ChatInfo
|
|
|
|
var chatItems: [ChatItem]
|
|
|
|
|
|
|
|
var id: String { get { chatInfo.id } }
|
|
|
|
}
|
|
|
|
|
2022-02-01 17:34:06 +00:00
|
|
|
struct Contact: Identifiable, Decodable {
|
2022-01-24 16:07:17 +00:00
|
|
|
var contactId: Int64
|
|
|
|
var localDisplayName: ContactName
|
|
|
|
var profile: Profile
|
2022-02-01 17:34:06 +00:00
|
|
|
var activeConn: Connection
|
2022-01-24 16:07:17 +00:00
|
|
|
var viaGroup: Int64?
|
2022-01-29 11:10:04 +00:00
|
|
|
|
|
|
|
var id: String { get { "@\(contactId)" } }
|
2022-02-02 12:51:39 +00:00
|
|
|
var apiId: Int64 { get { contactId } }
|
2022-02-01 17:34:06 +00:00
|
|
|
var connected: Bool { get { activeConn.connStatus == "ready" || activeConn.connStatus == "snd-ready" } }
|
2022-01-24 16:07:17 +00:00
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
let sampleContact = Contact(
|
|
|
|
contactId: 1,
|
|
|
|
localDisplayName: "alice",
|
2022-02-01 17:34:06 +00:00
|
|
|
profile: sampleProfile,
|
|
|
|
activeConn: sampleConnection
|
|
|
|
)
|
|
|
|
|
|
|
|
struct Connection: Decodable {
|
|
|
|
var connStatus: String
|
|
|
|
}
|
|
|
|
|
|
|
|
let sampleConnection = Connection(connStatus: "ready")
|
|
|
|
|
|
|
|
struct UserContactRequest: Decodable {
|
|
|
|
var contactRequestId: Int64
|
|
|
|
var localDisplayName: ContactName
|
|
|
|
var profile: Profile
|
|
|
|
|
|
|
|
var id: String { get { "<@\(contactRequestId)" } }
|
2022-02-02 12:51:39 +00:00
|
|
|
|
|
|
|
var apiId: Int64 { get { contactRequestId } }
|
2022-02-01 17:34:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let sampleContactRequest = UserContactRequest(
|
|
|
|
contactRequestId: 1,
|
|
|
|
localDisplayName: "alice",
|
2022-01-29 23:37:02 +00:00
|
|
|
profile: sampleProfile
|
|
|
|
)
|
|
|
|
|
2022-02-01 17:34:06 +00:00
|
|
|
struct GroupInfo: Identifiable, Decodable {
|
2022-01-24 16:07:17 +00:00
|
|
|
var groupId: Int64
|
|
|
|
var localDisplayName: GroupName
|
|
|
|
var groupProfile: GroupProfile
|
2022-01-29 11:10:04 +00:00
|
|
|
|
|
|
|
var id: String { get { "#\(groupId)" } }
|
2022-02-02 12:51:39 +00:00
|
|
|
|
|
|
|
var apiId: Int64 { get { groupId } }
|
2022-01-24 16:07:17 +00:00
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
let sampleGroupInfo = GroupInfo(
|
|
|
|
groupId: 1,
|
|
|
|
localDisplayName: "team",
|
|
|
|
groupProfile: sampleGroupProfile
|
|
|
|
)
|
|
|
|
|
2022-01-24 16:07:17 +00:00
|
|
|
struct GroupProfile: Codable {
|
|
|
|
var displayName: String
|
|
|
|
var fullName: String
|
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
let sampleGroupProfile = GroupProfile(
|
|
|
|
displayName: "team",
|
|
|
|
fullName: "My Team"
|
|
|
|
)
|
|
|
|
|
2022-02-01 17:34:06 +00:00
|
|
|
struct GroupMember: Decodable {
|
2022-01-29 11:10:04 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
struct AChatItem: Decodable {
|
|
|
|
var chatInfo: ChatInfo
|
|
|
|
var chatItem: ChatItem
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ChatItem: Identifiable, Decodable {
|
2022-01-29 11:10:04 +00:00
|
|
|
var chatDir: CIDirection
|
|
|
|
var meta: CIMeta
|
|
|
|
var content: CIContent
|
|
|
|
|
|
|
|
var id: Int64 { get { meta.itemId } }
|
2022-01-24 16:07:17 +00:00
|
|
|
}
|
|
|
|
|
2022-01-31 21:28:07 +00:00
|
|
|
func chatItemSample(_ id: Int64, _ dir: CIDirection, _ ts: Date, _ text: String) -> ChatItem {
|
|
|
|
ChatItem(
|
|
|
|
chatDir: dir,
|
|
|
|
meta: ciMetaSample(id, ts, text),
|
|
|
|
content: .sndMsgContent(msgContent: .text(text))
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
enum CIDirection: Decodable {
|
2022-01-29 11:10:04 +00:00
|
|
|
case directSnd
|
|
|
|
case directRcv
|
|
|
|
case groupSnd
|
|
|
|
case groupRcv(GroupMember)
|
2022-01-31 21:28:07 +00:00
|
|
|
|
|
|
|
var sent: Bool {
|
|
|
|
get {
|
|
|
|
switch self {
|
|
|
|
case .directSnd: return true
|
|
|
|
case .directRcv: return false
|
|
|
|
case .groupSnd: return true
|
|
|
|
case .groupRcv: return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-01-29 11:10:04 +00:00
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
struct CIMeta: Decodable {
|
2022-01-29 11:10:04 +00:00
|
|
|
var itemId: Int64
|
|
|
|
var itemTs: Date
|
|
|
|
var itemText: String
|
|
|
|
var createdAt: Date
|
|
|
|
}
|
|
|
|
|
2022-01-31 21:28:07 +00:00
|
|
|
func ciMetaSample(_ id: Int64, _ ts: Date, _ text: String) -> CIMeta {
|
|
|
|
CIMeta(
|
|
|
|
itemId: id,
|
|
|
|
itemTs: ts,
|
|
|
|
itemText: text,
|
|
|
|
createdAt: ts
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
enum CIContent: Decodable {
|
2022-01-29 11:10:04 +00:00
|
|
|
case sndMsgContent(msgContent: MsgContent)
|
|
|
|
case rcvMsgContent(msgContent: MsgContent)
|
|
|
|
// files etc.
|
|
|
|
|
|
|
|
var text: String {
|
|
|
|
get {
|
|
|
|
switch self {
|
|
|
|
case let .sndMsgContent(mc): return mc.string
|
|
|
|
case let .rcvMsgContent(mc): return mc.string
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
enum MsgContent {
|
2022-01-24 16:07:17 +00:00
|
|
|
case text(String)
|
2022-01-29 23:37:02 +00:00
|
|
|
case unknown(type: String, text: String)
|
|
|
|
case invalid(error: String)
|
2022-01-29 11:10:04 +00:00
|
|
|
|
|
|
|
var string: String {
|
|
|
|
get {
|
|
|
|
switch self {
|
2022-01-29 23:37:02 +00:00
|
|
|
case let .text(text): return text
|
|
|
|
case let .unknown(_, text): return text
|
2022-01-29 11:10:04 +00:00
|
|
|
case .invalid: return "invalid"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-01-29 23:37:02 +00:00
|
|
|
|
2022-01-30 18:27:20 +00:00
|
|
|
var cmdString: String {
|
|
|
|
get {
|
|
|
|
switch self {
|
|
|
|
case let .text(text): return "text \(text)"
|
|
|
|
default: return ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-29 23:37:02 +00:00
|
|
|
enum CodingKeys: String, CodingKey {
|
|
|
|
case type
|
|
|
|
case text
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension MsgContent: Decodable {
|
|
|
|
init(from decoder: Decoder) throws {
|
|
|
|
let container = try decoder.container(keyedBy: CodingKeys.self)
|
|
|
|
do {
|
|
|
|
let type = try container.decode(String.self, forKey: CodingKeys.type)
|
|
|
|
switch type {
|
|
|
|
case "text":
|
|
|
|
let text = try container.decode(String.self, forKey: CodingKeys.text)
|
|
|
|
self = .text(text)
|
|
|
|
default:
|
|
|
|
let text = try? container.decode(String.self, forKey: CodingKeys.text)
|
|
|
|
self = .unknown(type: type, text: text ?? "unknown message format")
|
|
|
|
}
|
|
|
|
} catch {
|
|
|
|
self = .invalid(error: String(describing: error))
|
|
|
|
}
|
|
|
|
}
|
2022-01-24 16:07:17 +00:00
|
|
|
}
|