// // Copyright 2022 New Vector Ltd // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // import Combine import SwiftUI typealias MessageForwardingScreenViewModelType = StateStoreViewModel class MessageForwardingScreenViewModel: MessageForwardingScreenViewModelType, MessageForwardingScreenViewModelProtocol { private let roomSummaryProvider: RoomSummaryProviderProtocol? private let sourceRoomID: String private var actionsSubject: PassthroughSubject = .init() var actions: AnyPublisher { actionsSubject.eraseToAnyPublisher() } init(roomSummaryProvider: RoomSummaryProviderProtocol, mediaProvider: MediaProviderProtocol, sourceRoomID: String) { self.roomSummaryProvider = roomSummaryProvider self.sourceRoomID = sourceRoomID super.init(initialViewState: MessageForwardingScreenViewState(), imageProvider: mediaProvider) roomSummaryProvider.roomListPublisher .receive(on: DispatchQueue.main) .sink { [weak self] _ in self?.updateRooms() } .store(in: &cancellables) context.$viewState .map(\.bindings.searchQuery) .removeDuplicates() .sink { [weak self] searchQuery in guard let self else { return } self.roomSummaryProvider?.setFilter(.search(query: searchQuery)) } .store(in: &cancellables) updateRooms() } override func process(viewAction: MessageForwardingScreenViewAction) { switch viewAction { case .cancel: actionsSubject.send(.dismiss) roomSummaryProvider?.setFilter(.all(filters: [])) case .send: guard let roomID = state.selectedRoomID else { fatalError() } actionsSubject.send(.send(roomID: roomID)) case .selectRoom(let roomID): state.selectedRoomID = roomID case .reachedTop: updateVisibleRange(edge: .top) case .reachedBottom: updateVisibleRange(edge: .bottom) } } // MARK: - Private private func updateRooms() { guard let roomSummaryProvider else { MXLog.error("Room summary provider unavailable") return } MXLog.verbose("Updating rooms") var rooms = [MessageForwardingRoom]() for summary in roomSummaryProvider.roomListPublisher.value { switch summary { case .empty, .invalidated: continue case .filled(let details): if details.id == sourceRoomID { continue } let room = MessageForwardingRoom(id: details.id, name: details.name, alias: details.canonicalAlias, avatarURL: details.avatarURL) rooms.append(room) } } state.rooms = rooms MXLog.verbose("Finished updating rooms") } /// The actual range values don't matter as long as they contain the lower /// or upper bounds. updateVisibleRange is a hybrid API that powers both /// sliding sync visible range update and list paginations /// For lists other than the home screen one we don't care about visible ranges, /// we just need the respective bounds to be there to trigger a next page load or /// a reset to just one page private func updateVisibleRange(edge: UIRectEdge) { guard let roomSummaryProvider else { return } switch edge { case .top: roomSummaryProvider.updateVisibleRange(0..<0) case .bottom: let roomCount = roomSummaryProvider.roomListPublisher.value.count roomSummaryProvider.updateVisibleRange(roomCount..