element-ios/RiotSwiftUI/Modules/Template/TemplateAdvancedRoomsExample/TemplateRoomChat/Service/MatrixSDK/TemplateRoomChatService.swift

109 lines
4.1 KiB
Swift

//
// Copyright 2021-2024 New Vector Ltd.
//
// SPDX-License-Identifier: AGPL-3.0-only
// Please see LICENSE in the repository root for full details.
//
import Combine
import Foundation
class TemplateRoomChatService: TemplateRoomChatServiceProtocol {
private let room: MXRoom
private let eventFormatter: EventFormatter
private var timeline: MXEventTimeline?
private var eventBatch: [MXEvent]
private var roomListenerReference: Any?
private(set) var chatMessagesSubject: CurrentValueSubject<[TemplateRoomChatMessage], Never>
private(set) var roomInitializationStatus: CurrentValueSubject<TemplateRoomChatRoomInitializationStatus, Never>
var roomName: String? {
room.summary.displayName
}
init(room: MXRoom) {
self.room = room
eventFormatter = EventFormatter(matrixSession: room.mxSession)
chatMessagesSubject = CurrentValueSubject([])
roomInitializationStatus = CurrentValueSubject(.notInitialized)
eventBatch = [MXEvent]()
initializeRoom()
}
deinit {
guard let reference = roomListenerReference else { return }
room.removeListener(reference)
}
// MARK: Public
func send(textMessage: String) {
var localEcho: MXEvent?
room.sendTextMessage(textMessage, threadId: nil, localEcho: &localEcho, completion: { _ in })
}
// MARK: Private
private func initializeRoom() {
room.liveTimeline { [weak self] timeline in
guard let self = self,
let timeline = timeline
else {
return
}
self.timeline = timeline
timeline.resetPagination()
self.roomListenerReference = timeline.listenToEvents([.roomMessage]) { [weak self] event, direction, _ in
guard let self = self else { return }
if direction == .backwards {
self.eventBatch.append(event)
} else {
self.chatMessagesSubject.value += self.mapChatMessages(from: [event])
}
}
timeline.paginate(200, direction: .backwards, onlyFromStore: false) { result in
guard result.isSuccess else {
self.roomInitializationStatus.value = .failedToInitialize
return
}
let sortedBatch = self.eventBatch.sorted(by: { $0.originServerTs < $1.originServerTs })
self.chatMessagesSubject.value = self.mapChatMessages(from: sortedBatch)
self.roomInitializationStatus.value = .initialized
}
}
}
private func mapChatMessages(from events: [MXEvent]) -> [TemplateRoomChatMessage] {
events
.filter { event in
event.type == kMXEventTypeStringRoomMessage
&& event.content[kMXMessageTypeKey] as? String == kMXMessageTypeText
// TODO: New to our SwiftUI Template? Why not implement another message type like image?
}
.compactMap { event -> TemplateRoomChatMessage? in
guard let eventId = event.eventId,
let body = event.content[kMXMessageBodyKey] as? String,
let sender = senderForMessage(event: event)
else { return nil }
return TemplateRoomChatMessage(
id: eventId,
content: .text(TemplateRoomChatMessageTextContent(body: body)),
sender: sender,
timestamp: Date(timeIntervalSince1970: TimeInterval(event.originServerTs / 1000))
)
}
}
private func senderForMessage(event: MXEvent) -> TemplateRoomChatMember? {
guard let sender = event.sender, let roomState = timeline?.state else {
return nil
}
let displayName = eventFormatter.senderDisplayName(for: event, with: roomState)
let avatarUrl = eventFormatter.senderAvatarUrl(for: event, with: roomState)
return TemplateRoomChatMember(id: sender, avatarUrl: avatarUrl, displayName: displayName)
}
}