2024-08-17 11:22:47 +00:00
|
|
|
import AVFoundation
|
2024-08-14 13:48:30 +00:00
|
|
|
import Combine
|
2024-08-13 08:40:27 +00:00
|
|
|
import Foundation
|
2024-08-14 13:48:30 +00:00
|
|
|
import GRDB
|
2024-08-17 11:22:47 +00:00
|
|
|
import Photos
|
2024-08-13 08:40:27 +00:00
|
|
|
|
|
|
|
@MainActor
|
|
|
|
final class ConversationStore: ObservableObject {
|
2024-08-15 15:15:49 +00:00
|
|
|
@Published private(set) var messages: [Message] = []
|
|
|
|
@Published var replyText = ""
|
2024-08-17 11:22:47 +00:00
|
|
|
@Published var cameraAccessGranted = false
|
|
|
|
@Published var galleryAccessGranted = false
|
2024-08-13 08:40:27 +00:00
|
|
|
|
2024-08-17 11:22:47 +00:00
|
|
|
private(set) var roster: Roster
|
2024-08-13 08:40:27 +00:00
|
|
|
private let client: Client
|
2024-08-14 13:48:30 +00:00
|
|
|
private let blockSize = Const.messagesPageSize
|
|
|
|
private let messagesMax = Const.messagesMaxSize
|
|
|
|
|
2024-08-15 11:37:21 +00:00
|
|
|
private var messagesCancellable: AnyCancellable?
|
2024-08-13 08:40:27 +00:00
|
|
|
|
|
|
|
init(roster: Roster, client: Client) {
|
|
|
|
self.client = client
|
|
|
|
self.roster = roster
|
2024-08-15 15:15:49 +00:00
|
|
|
subscribe()
|
2024-08-14 13:48:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension ConversationStore {
|
2024-08-15 15:15:49 +00:00
|
|
|
func sendMessage(_ message: String) async {
|
|
|
|
// prepare message
|
|
|
|
let message = Message(
|
|
|
|
id: UUID().uuidString,
|
|
|
|
type: .chat,
|
|
|
|
date: Date(),
|
|
|
|
contentType: .text,
|
|
|
|
status: .pending,
|
|
|
|
from: roster.bareJid,
|
|
|
|
to: roster.contactBareJid,
|
|
|
|
body: message,
|
|
|
|
subject: nil,
|
|
|
|
thread: nil,
|
|
|
|
oobUrl: nil
|
|
|
|
)
|
2024-08-15 11:37:21 +00:00
|
|
|
|
2024-08-15 15:15:49 +00:00
|
|
|
// store as pending on db, and send
|
|
|
|
do {
|
|
|
|
try await message.save()
|
2024-08-17 08:06:28 +00:00
|
|
|
try await client.sendMessage(message)
|
|
|
|
try await message.setStatus(.sent)
|
|
|
|
} catch {
|
|
|
|
try? await message.setStatus(.error)
|
|
|
|
}
|
2024-08-14 13:48:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-17 11:22:47 +00:00
|
|
|
extension ConversationStore {
|
|
|
|
func checkCameraAuthorization() async {
|
|
|
|
let status = AVCaptureDevice.authorizationStatus(for: .video)
|
|
|
|
var isAuthorized = status == .authorized
|
|
|
|
if status == .notDetermined {
|
|
|
|
isAuthorized = await AVCaptureDevice.requestAccess(for: .video)
|
|
|
|
}
|
|
|
|
cameraAccessGranted = isAuthorized
|
|
|
|
}
|
|
|
|
|
|
|
|
func checkGalleryAuthorization() async {
|
|
|
|
let status = PHPhotoLibrary.authorizationStatus()
|
|
|
|
var isAuthorized = status == .authorized
|
|
|
|
if status == .notDetermined {
|
|
|
|
let req = await PHPhotoLibrary.requestAuthorization(for: .readWrite)
|
|
|
|
isAuthorized = (req == .authorized) || (req == .limited)
|
|
|
|
}
|
|
|
|
galleryAccessGranted = isAuthorized
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-15 11:37:21 +00:00
|
|
|
private extension ConversationStore {
|
2024-08-15 15:15:49 +00:00
|
|
|
func subscribe() {
|
|
|
|
messagesCancellable = ValueObservation.tracking(Message
|
|
|
|
.filter(
|
|
|
|
(Column("to") == roster.bareJid && Column("from") == roster.contactBareJid) ||
|
|
|
|
(Column("from") == roster.bareJid && Column("to") == roster.contactBareJid)
|
2024-08-15 11:37:21 +00:00
|
|
|
)
|
2024-08-15 15:15:49 +00:00
|
|
|
.order(Column("date").desc)
|
|
|
|
.fetchAll
|
|
|
|
)
|
|
|
|
.publisher(in: Database.shared.dbQueue, scheduling: .immediate)
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { _ in
|
|
|
|
} receiveValue: { [weak self] messages in
|
|
|
|
self?.messages = messages
|
2024-08-15 11:37:21 +00:00
|
|
|
}
|
2024-08-13 08:40:27 +00:00
|
|
|
}
|
|
|
|
}
|