- proxy errors have been getting repetitive and not particularly useful - differentiate between sdk and client errors and keep what provides value - add error logs everywhere a failure occurs
344 lines
14 KiB
Swift
344 lines
14 KiB
Swift
//
|
||
// 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 RoomDetailsScreenViewModelType = StateStoreViewModel<RoomDetailsScreenViewState, RoomDetailsScreenViewAction>
|
||
|
||
class RoomDetailsScreenViewModel: RoomDetailsScreenViewModelType, RoomDetailsScreenViewModelProtocol {
|
||
private let roomProxy: RoomProxyProtocol
|
||
private let clientProxy: ClientProxyProtocol
|
||
private let analyticsService: AnalyticsService
|
||
private let mediaProvider: MediaProviderProtocol
|
||
private let userIndicatorController: UserIndicatorControllerProtocol
|
||
private let notificationSettingsProxy: NotificationSettingsProxyProtocol
|
||
private let attributedStringBuilder: AttributedStringBuilderProtocol
|
||
private let appSettings: AppSettings
|
||
|
||
private var dmRecipient: RoomMemberProxyProtocol?
|
||
|
||
private var actionsSubject: PassthroughSubject<RoomDetailsScreenViewModelAction, Never> = .init()
|
||
|
||
var actions: AnyPublisher<RoomDetailsScreenViewModelAction, Never> {
|
||
actionsSubject.eraseToAnyPublisher()
|
||
}
|
||
|
||
init(roomProxy: RoomProxyProtocol,
|
||
clientProxy: ClientProxyProtocol,
|
||
mediaProvider: MediaProviderProtocol,
|
||
analyticsService: AnalyticsService,
|
||
userIndicatorController: UserIndicatorControllerProtocol,
|
||
notificationSettingsProxy: NotificationSettingsProxyProtocol,
|
||
attributedStringBuilder: AttributedStringBuilderProtocol,
|
||
appSettings: AppSettings) {
|
||
self.roomProxy = roomProxy
|
||
self.clientProxy = clientProxy
|
||
self.mediaProvider = mediaProvider
|
||
self.analyticsService = analyticsService
|
||
self.userIndicatorController = userIndicatorController
|
||
self.notificationSettingsProxy = notificationSettingsProxy
|
||
self.attributedStringBuilder = attributedStringBuilder
|
||
self.appSettings = appSettings
|
||
|
||
let topic = attributedStringBuilder.fromPlain(roomProxy.topic)
|
||
|
||
super.init(initialViewState: .init(details: roomProxy.details,
|
||
isEncrypted: roomProxy.isEncrypted,
|
||
isDirect: roomProxy.isDirect,
|
||
topic: topic,
|
||
topicSummary: topic?.unattributedStringByReplacingNewlinesWithSpaces(),
|
||
joinedMembersCount: roomProxy.joinedMembersCount,
|
||
notificationSettingsState: .loading,
|
||
bindings: .init()),
|
||
imageProvider: mediaProvider)
|
||
|
||
Task {
|
||
if case let .success(permalinkURL) = await roomProxy.matrixToPermalink() {
|
||
state.permalink = permalinkURL
|
||
}
|
||
}
|
||
|
||
updateRoomInfo()
|
||
Task { await updatePowerLevelPermissions() }
|
||
|
||
setupRoomSubscription()
|
||
Task { await fetchMembersIfNeeded() }
|
||
|
||
setupNotificationSettingsSubscription()
|
||
fetchNotificationSettings()
|
||
}
|
||
|
||
// MARK: - Public
|
||
|
||
func stop() {
|
||
// Work around QLPreviewController dismissal issues, see the InteractiveQuickLookModifier.
|
||
state.bindings.mediaPreviewItem = nil
|
||
}
|
||
|
||
override func process(viewAction: RoomDetailsScreenViewAction) {
|
||
switch viewAction {
|
||
case .processTapPeople:
|
||
actionsSubject.send(.requestMemberDetailsPresentation)
|
||
case .processTapInvite:
|
||
actionsSubject.send(.requestInvitePeoplePresentation)
|
||
case .processTapLeave:
|
||
guard state.joinedMembersCount > 1 else {
|
||
state.bindings.leaveRoomAlertItem = LeaveRoomAlertItem(roomID: roomProxy.id, isDM: roomProxy.isEncryptedOneToOneRoom, state: .empty)
|
||
return
|
||
}
|
||
state.bindings.leaveRoomAlertItem = LeaveRoomAlertItem(roomID: roomProxy.id, isDM: roomProxy.isEncryptedOneToOneRoom, state: roomProxy.isPublic ? .public : .private)
|
||
case .confirmLeave:
|
||
Task { await leaveRoom() }
|
||
case .processTapIgnore:
|
||
state.bindings.ignoreUserRoomAlertItem = .init(action: .ignore)
|
||
case .processTapUnignore:
|
||
state.bindings.ignoreUserRoomAlertItem = .init(action: .unignore)
|
||
case .processTapEdit, .processTapAddTopic:
|
||
actionsSubject.send(.requestEditDetailsPresentation)
|
||
case .ignoreConfirmed:
|
||
Task { await ignore() }
|
||
case .unignoreConfirmed:
|
||
Task { await unignore() }
|
||
case .processTapNotifications:
|
||
if state.notificationSettingsState.isError {
|
||
fetchNotificationSettings()
|
||
} else {
|
||
actionsSubject.send(.requestNotificationSettingsPresentation)
|
||
}
|
||
case .processToggleMuteNotifications:
|
||
Task { await toggleMuteNotifications() }
|
||
case .displayAvatar:
|
||
displayFullScreenAvatar()
|
||
case .processTapPolls:
|
||
actionsSubject.send(.requestPollsHistoryPresentation)
|
||
case .toggleFavourite(let isFavourite):
|
||
Task { await toggleFavourite(isFavourite) }
|
||
case .processTapRolesAndPermissions:
|
||
actionsSubject.send(.requestRolesAndPermissionsPresentation)
|
||
}
|
||
}
|
||
|
||
// MARK: - Private
|
||
|
||
private func setupRoomSubscription() {
|
||
roomProxy.actionsPublisher
|
||
.filter { $0 == .roomInfoUpdate }
|
||
.throttle(for: .milliseconds(200), scheduler: DispatchQueue.main, latest: true)
|
||
.sink { [weak self] _ in
|
||
self?.updateRoomInfo()
|
||
Task { await self?.updatePowerLevelPermissions() }
|
||
}
|
||
.store(in: &cancellables)
|
||
}
|
||
|
||
private func updateRoomInfo() {
|
||
state.details = roomProxy.details
|
||
|
||
let topic = attributedStringBuilder.fromPlain(roomProxy.topic)
|
||
state.topic = topic
|
||
state.topicSummary = topic?.unattributedStringByReplacingNewlinesWithSpaces()
|
||
state.joinedMembersCount = roomProxy.joinedMembersCount
|
||
|
||
Task {
|
||
state.bindings.isFavourite = await roomProxy.isFavourite
|
||
}
|
||
}
|
||
|
||
private func fetchMembersIfNeeded() async {
|
||
// We need to fetch members just in 1-to-1 chat to get the member object for the other person
|
||
guard roomProxy.isEncryptedOneToOneRoom else {
|
||
return
|
||
}
|
||
|
||
roomProxy.membersPublisher
|
||
.receive(on: DispatchQueue.main)
|
||
.sink { [weak self, ownUserID = roomProxy.ownUserID] members in
|
||
guard let self else { return }
|
||
let dmRecipient = members.first(where: { $0.userID != ownUserID })
|
||
self.dmRecipient = dmRecipient
|
||
self.state.dmRecipient = dmRecipient.map(RoomMemberDetails.init(withProxy:))
|
||
}
|
||
.store(in: &cancellables)
|
||
|
||
await roomProxy.updateMembers()
|
||
}
|
||
|
||
private func updatePowerLevelPermissions() async {
|
||
state.canEditRoomName = await (try? roomProxy.canUser(userID: roomProxy.ownUserID, sendStateEvent: .roomName).get()) == true
|
||
state.canEditRoomTopic = await (try? roomProxy.canUser(userID: roomProxy.ownUserID, sendStateEvent: .roomTopic).get()) == true
|
||
state.canEditRoomAvatar = await (try? roomProxy.canUser(userID: roomProxy.ownUserID, sendStateEvent: .roomAvatar).get()) == true
|
||
|
||
if appSettings.roomModerationEnabled {
|
||
state.canEditRolesOrPermissions = await (try? roomProxy.suggestedRole(for: roomProxy.ownUserID).get()) == .administrator
|
||
}
|
||
|
||
state.canInviteUsers = await (try? roomProxy.canUserInvite(userID: roomProxy.ownUserID).get()) == true
|
||
}
|
||
|
||
private func setupNotificationSettingsSubscription() {
|
||
notificationSettingsProxy.callbacks
|
||
.receive(on: DispatchQueue.main)
|
||
.sink { [weak self] callback in
|
||
guard let self else { return }
|
||
|
||
switch callback {
|
||
case .settingsDidChange:
|
||
self.fetchNotificationSettings()
|
||
}
|
||
}
|
||
.store(in: &cancellables)
|
||
}
|
||
|
||
private func fetchNotificationSettings() {
|
||
Task {
|
||
await fetchRoomNotificationSettings()
|
||
}
|
||
}
|
||
|
||
private func fetchRoomNotificationSettings() async {
|
||
do {
|
||
let notificationMode = try await notificationSettingsProxy.getNotificationSettings(roomId: roomProxy.id,
|
||
isEncrypted: roomProxy.isEncrypted,
|
||
isOneToOne: roomProxy.activeMembersCount == 2)
|
||
state.notificationSettingsState = .loaded(settings: notificationMode)
|
||
} catch {
|
||
state.notificationSettingsState = .error
|
||
state.bindings.alertInfo = AlertInfo(id: .alert,
|
||
title: L10n.commonError,
|
||
message: L10n.screenRoomDetailsErrorLoadingNotificationSettings)
|
||
}
|
||
}
|
||
|
||
private func toggleMuteNotifications() async {
|
||
guard case .loaded(let notificationMode) = state.notificationSettingsState else { return }
|
||
state.isProcessingMuteToggleAction = true
|
||
switch notificationMode.mode {
|
||
case .mute:
|
||
do {
|
||
try await notificationSettingsProxy.unmuteRoom(roomId: roomProxy.id,
|
||
isEncrypted: roomProxy.isEncrypted,
|
||
isOneToOne: roomProxy.activeMembersCount == 2)
|
||
} catch {
|
||
state.bindings.alertInfo = AlertInfo(id: .alert,
|
||
title: L10n.commonError,
|
||
message: L10n.screenRoomDetailsErrorUnmuting)
|
||
}
|
||
default:
|
||
do {
|
||
try await notificationSettingsProxy.setNotificationMode(roomId: roomProxy.id, mode: .mute)
|
||
} catch {
|
||
state.bindings.alertInfo = AlertInfo(id: .alert,
|
||
title: L10n.commonError,
|
||
message: L10n.screenRoomDetailsErrorMuting)
|
||
}
|
||
}
|
||
state.isProcessingMuteToggleAction = false
|
||
}
|
||
|
||
private func toggleFavourite(_ isFavourite: Bool) async {
|
||
if case let .failure(error) = await roomProxy.flagAsFavourite(isFavourite) {
|
||
MXLog.error("Failed flagging room as favourite with error: \(error)")
|
||
state.bindings.isFavourite = !isFavourite
|
||
} else {
|
||
analyticsService.trackInteraction(name: .MobileRoomFavouriteToggle)
|
||
}
|
||
}
|
||
|
||
private static let leaveRoomLoadingID = "LeaveRoomLoading"
|
||
|
||
private func leaveRoom() async {
|
||
userIndicatorController.submitIndicator(UserIndicator(id: Self.leaveRoomLoadingID, type: .modal, title: L10n.commonLeavingRoom, persistent: true))
|
||
let result = await roomProxy.leaveRoom()
|
||
userIndicatorController.retractIndicatorWithId(Self.leaveRoomLoadingID)
|
||
switch result {
|
||
case .failure:
|
||
state.bindings.alertInfo = AlertInfo(id: .unknown)
|
||
case .success:
|
||
actionsSubject.send(.leftRoom)
|
||
}
|
||
}
|
||
|
||
private func ignore() async {
|
||
guard let dmUserID = dmRecipient?.userID else {
|
||
MXLog.error("Attempting to ignore a nil DM Recipient")
|
||
state.bindings.alertInfo = .init(id: .unknown)
|
||
return
|
||
}
|
||
|
||
state.isProcessingIgnoreRequest = true
|
||
let result = await clientProxy.ignoreUser(dmUserID)
|
||
state.isProcessingIgnoreRequest = false
|
||
switch result {
|
||
case .success:
|
||
// Mutating the optional in place when built for Release crashes 🤷♂️
|
||
var dmRecipient = state.dmRecipient
|
||
dmRecipient?.isIgnored = true
|
||
state.dmRecipient = dmRecipient
|
||
case .failure:
|
||
state.bindings.alertInfo = .init(id: .unknown)
|
||
}
|
||
}
|
||
|
||
private func unignore() async {
|
||
guard let dmUserID = dmRecipient?.userID else {
|
||
MXLog.error("Attempting to unignore a nil DM Recipient")
|
||
state.bindings.alertInfo = .init(id: .unknown)
|
||
return
|
||
}
|
||
|
||
state.isProcessingIgnoreRequest = true
|
||
let result = await clientProxy.unignoreUser(dmUserID)
|
||
state.isProcessingIgnoreRequest = false
|
||
switch result {
|
||
case .success:
|
||
// Mutating the optional in place when built for Release crashes 🤷♂️
|
||
var dmRecipient = state.dmRecipient
|
||
dmRecipient?.isIgnored = false
|
||
state.dmRecipient = dmRecipient
|
||
case .failure:
|
||
state.bindings.alertInfo = .init(id: .unknown)
|
||
}
|
||
}
|
||
|
||
private func displayFullScreenAvatar() {
|
||
guard let avatarURL = roomProxy.avatarURL else {
|
||
return
|
||
}
|
||
|
||
let loadingIndicatorIdentifier = "roomAvatarLoadingIndicator"
|
||
userIndicatorController.submitIndicator(UserIndicator(id: loadingIndicatorIdentifier, type: .modal, title: L10n.commonLoading, persistent: true))
|
||
|
||
Task {
|
||
defer {
|
||
userIndicatorController.retractIndicatorWithId(loadingIndicatorIdentifier)
|
||
}
|
||
|
||
// We don't actually know the mime type here, assume it's an image.
|
||
if case let .success(file) = await mediaProvider.loadFileFromSource(.init(url: avatarURL, mimeType: "image/jpeg")) {
|
||
state.bindings.mediaPreviewItem = MediaPreviewItem(file: file, title: roomProxy.roomTitle)
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
private extension AttributedString {
|
||
/// Returns a new string without attributes and in which newlines are replaced with spaces
|
||
func unattributedStringByReplacingNewlinesWithSpaces() -> AttributedString {
|
||
AttributedString(characters.map { $0.isNewline ? " " : $0 })
|
||
}
|
||
}
|