Files
letro-ios/ElementX/Sources/Services/RoomDirectorySearch/RoomDirectorySearchProxy.swift
2025-05-23 16:08:18 +03:00

155 lines
6.4 KiB
Swift

//
// Copyright 2024 New Vector Ltd.
//
// SPDX-License-Identifier: AGPL-3.0-only OR LicenseRef-Element-Commercial
// Please see LICENSE files in the repository root for full details.
//
import Combine
import Foundation
import MatrixRustSDK
final class RoomDirectorySearchProxy: RoomDirectorySearchProxyProtocol {
private let roomDirectorySearch: RoomDirectorySearchProtocol
private let appSettings: AppSettings
private let serialDispatchQueue = DispatchQueue(label: "io.element.elementx.room_directory_search_proxy", qos: .default)
private let resultsSubject = CurrentValueSubject<[RoomDirectorySearchResult], Never>([])
var resultsPublisher: CurrentValuePublisher<[RoomDirectorySearchResult], Never> {
resultsSubject.asCurrentValuePublisher()
}
private var results: [RoomDirectorySearchResult] {
get { resultsSubject.value }
set { resultsSubject.send(newValue) }
}
private let diffsPublisher = PassthroughSubject<[RoomDirectorySearchEntryUpdate], Never>()
private var searchEntriesSubscription: TaskHandle?
private var cancellables = Set<AnyCancellable>()
init(roomDirectorySearch: RoomDirectorySearchProtocol,
appSettings: AppSettings) {
self.roomDirectorySearch = roomDirectorySearch
self.appSettings = appSettings
diffsPublisher
.receive(on: serialDispatchQueue)
.sink { [weak self] in self?.updateResultsWithDiffs($0) }
.store(in: &cancellables)
Task {
searchEntriesSubscription = await roomDirectorySearch.results(listener: SDKListener { [weak self] updates in
self?.diffsPublisher.send(updates)
})
}
}
func search(query: String?) async -> Result<Void, RoomDirectorySearchError> {
do {
try await roomDirectorySearch.search(filter: query, batchSize: 50, viaServerName: nil)
return .success(())
} catch {
return .failure(.searchFailed)
}
}
func nextPage() async -> Result<Void, RoomDirectorySearchError> {
do {
try await roomDirectorySearch.nextPage()
return .success(())
} catch {
return .failure(.nextPageQueryFailed)
}
}
private func updateResultsWithDiffs(_ updates: [RoomDirectorySearchEntryUpdate]) {
results = updates.reduce(results) { currentItems, diff in
processDiff(diff, on: currentItems)
}
}
private func processDiff(_ diff: RoomDirectorySearchEntryUpdate, on currentItems: [RoomDirectorySearchResult]) -> [RoomDirectorySearchResult] {
guard let collectionDiff = buildDiff(from: diff, on: currentItems) else {
return currentItems
}
guard let updatedItems = currentItems.applying(collectionDiff) else {
return currentItems
}
return updatedItems
}
private func buildDiff(from diff: RoomDirectorySearchEntryUpdate, on currentItems: [RoomDirectorySearchResult]) -> CollectionDifference<RoomDirectorySearchResult>? {
var changes = [CollectionDifference<RoomDirectorySearchResult>.Change]()
switch diff {
case .append(let values):
for (index, value) in values.enumerated() {
let result = buildResultForRoomDescription(value)
changes.append(.insert(offset: results.count + index, element: result, associatedWith: nil))
}
case .clear:
for (index, value) in results.enumerated() {
changes.append(.remove(offset: index, element: value, associatedWith: nil))
}
case .insert(let index, let value):
let result = buildResultForRoomDescription(value)
changes.append(.insert(offset: Int(index), element: result, associatedWith: nil))
case .popBack:
guard let value = results.last else {
fatalError()
}
changes.append(.remove(offset: results.count - 1, element: value, associatedWith: nil))
case .popFront:
let result = results[0]
changes.append(.remove(offset: 0, element: result, associatedWith: nil))
case .pushBack(let value):
let result = buildResultForRoomDescription(value)
changes.append(.insert(offset: results.count, element: result, associatedWith: nil))
case .pushFront(let value):
let result = buildResultForRoomDescription(value)
changes.append(.insert(offset: 0, element: result, associatedWith: nil))
case .remove(let index):
let result = results[Int(index)]
changes.append(.remove(offset: Int(index), element: result, associatedWith: nil))
case .reset(let values):
for (index, result) in results.enumerated() {
changes.append(.remove(offset: index, element: result, associatedWith: nil))
}
for (index, value) in values.enumerated() {
changes.append(.insert(offset: index, element: buildResultForRoomDescription(value), associatedWith: nil))
}
case .set(let index, let value):
let result = buildResultForRoomDescription(value)
changes.append(.remove(offset: Int(index), element: result, associatedWith: nil))
changes.append(.insert(offset: Int(index), element: result, associatedWith: nil))
case .truncate(let length):
for (index, value) in results.enumerated() {
if index < length {
continue
}
changes.append(.remove(offset: index, element: value, associatedWith: nil))
}
}
return CollectionDifference(changes)
}
private func buildResultForRoomDescription(_ value: RoomDescription) -> RoomDirectorySearchResult {
RoomDirectorySearchResult(id: value.roomId,
alias: value.alias,
name: value.name,
topic: value.topic,
avatar: .room(id: value.roomId, name: value.name, avatarURL: value.avatarUrl.flatMap(URL.init(string:))),
canBeJoined: value.joinRule == .public || (appSettings.knockingEnabled && value.joinRule == .knock))
}
}