226 lines
9.2 KiB
Swift
226 lines
9.2 KiB
Swift
//
|
|
// Copyright 2025 Element Creations Ltd.
|
|
// Copyright 2022-2025 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 SwiftUI
|
|
|
|
typealias RoomChangeRolesScreenViewModelType = StateStoreViewModelV2<RoomChangeRolesScreenViewState, RoomChangeRolesScreenViewAction>
|
|
|
|
class RoomChangeRolesScreenViewModel: RoomChangeRolesScreenViewModelType, RoomChangeRolesScreenViewModelProtocol {
|
|
private let roomProxy: JoinedRoomProxyProtocol
|
|
private let userIndicatorController: UserIndicatorControllerProtocol
|
|
private let analytics: AnalyticsService
|
|
|
|
private let actionsSubject: PassthroughSubject<RoomChangeRolesScreenViewModelAction, Never> = .init()
|
|
var actionsPublisher: AnyPublisher<RoomChangeRolesScreenViewModelAction, Never> {
|
|
actionsSubject.eraseToAnyPublisher()
|
|
}
|
|
|
|
init(mode: RoomRole,
|
|
roomProxy: JoinedRoomProxyProtocol,
|
|
mediaProvider: MediaProviderProtocol,
|
|
userIndicatorController: UserIndicatorControllerProtocol,
|
|
analytics: AnalyticsService) {
|
|
guard mode != .user || mode != .creator else { fatalError("Invalid screen configuration: \(mode)") }
|
|
|
|
self.roomProxy = roomProxy
|
|
self.userIndicatorController = userIndicatorController
|
|
self.analytics = analytics
|
|
|
|
super.init(initialViewState: RoomChangeRolesScreenViewState(mode: mode,
|
|
ownRole: roomProxy.membersPublisher.value.first { $0.userID == roomProxy.id }?.role ?? .administrator),
|
|
mediaProvider: mediaProvider)
|
|
|
|
roomProxy.membersPublisher
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { [weak self] members in
|
|
self?.updateMembers(members)
|
|
}
|
|
.store(in: &cancellables)
|
|
|
|
roomProxy.timeline.timelineItemProvider.membershipChangePublisher.sink { [roomProxy] in
|
|
Task { await roomProxy.updateMembers() }
|
|
}
|
|
.store(in: &cancellables)
|
|
|
|
updateMembers(roomProxy.membersPublisher.value)
|
|
}
|
|
|
|
// MARK: - Public
|
|
|
|
override func process(viewAction: RoomChangeRolesScreenViewAction) {
|
|
MXLog.info("View model: received view action: \(viewAction)")
|
|
|
|
switch viewAction {
|
|
case .toggleMember(let member):
|
|
toggleMember(member)
|
|
case .demoteMember(let member):
|
|
demoteMember(member)
|
|
case .save:
|
|
if !state.membersToPromote.isEmpty {
|
|
if state.mode == .administrator, state.ownRole == .administrator {
|
|
showPromotionWarning()
|
|
return
|
|
} else if state.mode == .owner {
|
|
showTransferOwnershipWarning()
|
|
return
|
|
}
|
|
}
|
|
Task { await save() }
|
|
case .cancel:
|
|
confirmDiscardChanges()
|
|
}
|
|
}
|
|
|
|
// MARK: - Private
|
|
|
|
private func updateMembers(_ members: [RoomMemberProxyProtocol]) {
|
|
var owners = [RoomMemberDetails]()
|
|
var administrators = [RoomMemberDetails]()
|
|
var moderators = [RoomMemberDetails]()
|
|
var users = [RoomMemberDetails]()
|
|
|
|
for member in members.sorted() {
|
|
if member.userID == roomProxy.ownUserID {
|
|
state.ownRole = member.role
|
|
}
|
|
|
|
guard member.isActive else { continue }
|
|
let memberDetails = RoomMemberDetails(withProxy: member)
|
|
|
|
switch memberDetails.role {
|
|
case .creator, .owner:
|
|
owners.append(memberDetails)
|
|
case .administrator:
|
|
administrators.append(memberDetails)
|
|
case .moderator:
|
|
moderators.append(memberDetails)
|
|
case .user:
|
|
users.append(memberDetails)
|
|
}
|
|
}
|
|
|
|
state.owners = owners
|
|
state.administrators = administrators
|
|
state.moderators = moderators
|
|
state.users = users
|
|
}
|
|
|
|
private func toggleMember(_ member: RoomMemberDetails) {
|
|
if state.membersToPromote.contains(member) {
|
|
state.membersToPromote.remove(member)
|
|
} else if state.membersToDemote.contains(member) {
|
|
state.membersToDemote.remove(member)
|
|
state.lastPromotedMember = member
|
|
} else if member.role >= state.mode, member.role <= state.maxDemotableRole {
|
|
state.membersToDemote.insert(member)
|
|
} else if member.role < state.mode {
|
|
state.membersToPromote.insert(member)
|
|
state.lastPromotedMember = member
|
|
}
|
|
}
|
|
|
|
private func demoteMember(_ member: RoomMemberDetails) {
|
|
if state.membersToPromote.contains(member) {
|
|
state.membersToPromote.remove(member)
|
|
} else {
|
|
state.membersToDemote.insert(member)
|
|
}
|
|
}
|
|
|
|
private func showPromotionWarning() {
|
|
context.alertInfo = AlertInfo(id: .promotionWarning,
|
|
title: L10n.screenRoomChangeRoleConfirmAddAdminTitle,
|
|
message: L10n.screenRoomChangeRoleConfirmAddAdminDescription,
|
|
primaryButton: .init(title: L10n.actionContinue) {
|
|
Task { await self.save() }
|
|
},
|
|
secondaryButton: .init(title: L10n.actionCancel, role: .cancel, action: nil))
|
|
}
|
|
|
|
private func showTransferOwnershipWarning() {
|
|
context.alertInfo = AlertInfo(id: .transferOwnershipWarning,
|
|
title: L10n.screenRoomChangeRoleConfirmChangeOwnersTitle,
|
|
message: L10n.screenRoomChangeRoleConfirmChangeOwnersDescription,
|
|
primaryButton: .init(title: L10n.actionContinue, role: .destructive) {
|
|
Task { await self.save() }
|
|
},
|
|
secondaryButton: .init(title: L10n.actionCancel, role: .cancel, action: nil))
|
|
}
|
|
|
|
private func save() async {
|
|
showSavingIndicator()
|
|
|
|
defer {
|
|
hideSavingIndicator()
|
|
}
|
|
|
|
let promotingUpdates = state.membersToPromote.map { ($0.id, state.mode.powerLevelValue) }
|
|
let demotingUpdates = state.membersToDemote.map { ($0.id, Int64(0)) }
|
|
|
|
// A task we can await until the room's info gets modified with the new power levels.
|
|
// Note: Ignore the first value as the publisher is backed by a current value subject.
|
|
let infoTask = Task { await roomProxy.infoPublisher.dropFirst().values.first { _ in true } }
|
|
|
|
switch await roomProxy.updatePowerLevelsForUsers(promotingUpdates + demotingUpdates) {
|
|
case .success:
|
|
MXLog.info("Success")
|
|
|
|
// Call updateMembers so the count is correct on the root screen.
|
|
_ = await infoTask.value
|
|
await roomProxy.updateMembers()
|
|
|
|
trackChanges(promotionCount: promotingUpdates.count,
|
|
demotionCount: demotingUpdates.count)
|
|
|
|
actionsSubject.send(.complete)
|
|
case .failure:
|
|
context.alertInfo = AlertInfo(id: .error)
|
|
}
|
|
}
|
|
|
|
private func confirmDiscardChanges() {
|
|
if state.hasChanges {
|
|
state.bindings.alertInfo = AlertInfo(id: .discardChanges,
|
|
title: L10n.screenRoomChangeRoleUnsavedChangesTitle,
|
|
message: L10n.screenRoomChangeRoleUnsavedChangesDescription,
|
|
primaryButton: .init(title: L10n.actionSave) { Task { await self.save() } },
|
|
secondaryButton: .init(title: L10n.actionDiscard, role: .cancel) { self.actionsSubject.send(.complete) })
|
|
} else {
|
|
actionsSubject.send(.complete)
|
|
}
|
|
}
|
|
|
|
// MARK: Loading indicator
|
|
|
|
private static let indicatorID = "SavingRoomRoles"
|
|
|
|
private func showSavingIndicator() {
|
|
userIndicatorController.submitIndicator(UserIndicator(id: Self.indicatorID,
|
|
type: .modal(progress: .indeterminate, interactiveDismissDisabled: true, allowsInteraction: false),
|
|
title: L10n.commonSaving,
|
|
persistent: true))
|
|
}
|
|
|
|
private func hideSavingIndicator() {
|
|
userIndicatorController.retractIndicatorWithId(Self.indicatorID)
|
|
}
|
|
|
|
// MARK: Analytics
|
|
|
|
private func trackChanges(promotionCount: Int, demotionCount: Int) {
|
|
for _ in 0..<promotionCount {
|
|
analytics.trackRoomModeration(action: .ChangeMemberRole, role: state.mode)
|
|
}
|
|
|
|
for _ in 0..<demotionCount {
|
|
analytics.trackRoomModeration(action: .ChangeMemberRole, role: .user)
|
|
}
|
|
}
|
|
}
|