* Enable key-share-on-invite irrespective of feature flag * Remove feature-flag dep: warning on starting chat with new people * Remove feature-flag dep: invite from room member details * Remove feature-flag dep: warning on new users in invite screen * Remove feature-flag dep: from room details screen * Remove feature-flag dep: starting chat from user profile screen * Remove feature-flag dep: timeline info on forwarded keys * Remove feature-flag dep: RoomScreenModel * Remove `enableKeyShareOnInvite` from AppSettings * Remove `enableKeyShareOnInvite` feature flag * Remove outdated comments * Update preview test room snapshots as their header now includes the history sharing icon
279 lines
12 KiB
Swift
279 lines
12 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 RoomMemberDetailsScreenViewModelType = StateStoreViewModel<RoomMemberDetailsScreenViewState, RoomMemberDetailsScreenViewAction>
|
|
|
|
class RoomMemberDetailsScreenViewModel: RoomMemberDetailsScreenViewModelType, RoomMemberDetailsScreenViewModelProtocol {
|
|
private let roomProxy: JoinedRoomProxyProtocol
|
|
private let userSession: UserSessionProtocol
|
|
private let userIndicatorController: UserIndicatorControllerProtocol
|
|
private let analytics: AnalyticsService
|
|
private let appSettings: AppSettings
|
|
|
|
private var actionsSubject: PassthroughSubject<RoomMemberDetailsScreenViewModelAction, Never> = .init()
|
|
|
|
private var roomMemberProxy: RoomMemberProxyProtocol?
|
|
|
|
var actions: AnyPublisher<RoomMemberDetailsScreenViewModelAction, Never> {
|
|
actionsSubject.eraseToAnyPublisher()
|
|
}
|
|
|
|
init(userID: String,
|
|
roomProxy: JoinedRoomProxyProtocol,
|
|
userSession: UserSessionProtocol,
|
|
userIndicatorController: UserIndicatorControllerProtocol,
|
|
analytics: AnalyticsService,
|
|
appSettings: AppSettings) {
|
|
self.roomProxy = roomProxy
|
|
self.userSession = userSession
|
|
self.userIndicatorController = userIndicatorController
|
|
self.analytics = analytics
|
|
self.appSettings = appSettings
|
|
|
|
let initialViewState = RoomMemberDetailsScreenViewState(userID: userID, bindings: .init())
|
|
|
|
super.init(initialViewState: initialViewState, mediaProvider: userSession.mediaProvider)
|
|
|
|
showMemberLoadingIndicator()
|
|
|
|
Task {
|
|
await loadMember()
|
|
hideMemberLoadingIndicator()
|
|
}
|
|
|
|
roomProxy.identityStatusChangesPublisher
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { [weak self] changes in
|
|
if changes.map(\.userId).contains(userID) {
|
|
Task { await self?.loadMember() }
|
|
}
|
|
}
|
|
.store(in: &cancellables)
|
|
}
|
|
|
|
// MARK: - Public
|
|
|
|
func stop() {
|
|
// Work around QLPreviewController dismissal issues, see the InteractiveQuickLookModifier.
|
|
state.bindings.mediaPreviewItem = nil
|
|
|
|
hideMemberLoadingIndicator()
|
|
}
|
|
|
|
override func process(viewAction: RoomMemberDetailsScreenViewAction) {
|
|
switch viewAction {
|
|
case .showUnignoreAlert:
|
|
state.bindings.ignoreUserAlert = .init(action: .unignore)
|
|
case .showIgnoreAlert:
|
|
state.bindings.ignoreUserAlert = .init(action: .ignore)
|
|
case .ignoreConfirmed:
|
|
Task { await ignoreUser() }
|
|
case .unignoreConfirmed:
|
|
Task { await unignoreUser() }
|
|
case .displayAvatar(let url):
|
|
Task { await displayFullScreenAvatar(url) }
|
|
case .openDirectChat:
|
|
openDirectChat()
|
|
case .createDirectChat:
|
|
Task { await createDirectChat() }
|
|
case .startCall(let roomID, let isVoiceCall):
|
|
Task { await startCall(roomID: roomID, isVoiceCall: isVoiceCall) }
|
|
case .verifyUser:
|
|
actionsSubject.send(.verifyUser(userID: state.userID))
|
|
case .withdrawVerification:
|
|
Task { await userSession.clientProxy.withdrawUserIdentityVerification(state.userID) }
|
|
}
|
|
}
|
|
|
|
// MARK: - Private
|
|
|
|
private func loadMember() async {
|
|
switch await roomProxy.getMember(userID: state.userID) {
|
|
case .success(let member):
|
|
roomMemberProxy = member
|
|
state.memberDetails = RoomMemberDetails(withProxy: member)
|
|
state.isOwnMemberDetails = member.userID == roomProxy.ownUserID
|
|
switch userSession.clientProxy.directRoomForUserID(member.userID) {
|
|
case .success(let roomID):
|
|
state.dmRoomID = roomID
|
|
case .failure:
|
|
break
|
|
}
|
|
case .failure(let error):
|
|
MXLog.warning("Failed to find member: \(error)")
|
|
// As we didn't find a member with the specified user ID in this room we instead
|
|
// fall back to showing a generic user profile screen as the source is likely
|
|
// a message containing a permalink to someone who's not in this room.
|
|
actionsSubject.send(.openUserProfile)
|
|
}
|
|
|
|
if case let .success(.some(identity)) = await userSession.clientProxy.userIdentity(for: state.userID, fallBackToServer: true) {
|
|
state.verificationState = identity.verificationState
|
|
} else {
|
|
MXLog.error("Failed to find the member's identity.")
|
|
}
|
|
}
|
|
|
|
private func ignoreUser() async {
|
|
guard let roomMemberProxy else {
|
|
fatalError()
|
|
}
|
|
|
|
state.isProcessingIgnoreRequest = true
|
|
let result = await userSession.clientProxy.ignoreUser(roomMemberProxy.userID)
|
|
state.isProcessingIgnoreRequest = false
|
|
switch result {
|
|
case .success:
|
|
var details = state.memberDetails
|
|
details?.isIgnored = true
|
|
state.memberDetails = details
|
|
|
|
updateMembers()
|
|
case .failure:
|
|
state.bindings.alertInfo = .init(id: .unknown)
|
|
}
|
|
}
|
|
|
|
@MainActor
|
|
private func unignoreUser() async {
|
|
guard let roomMemberProxy else {
|
|
fatalError()
|
|
}
|
|
|
|
state.isProcessingIgnoreRequest = true
|
|
let result = await userSession.clientProxy.unignoreUser(roomMemberProxy.userID)
|
|
state.isProcessingIgnoreRequest = false
|
|
switch result {
|
|
case .success:
|
|
var details = state.memberDetails
|
|
details?.isIgnored = false
|
|
state.memberDetails = details
|
|
|
|
updateMembers()
|
|
case .failure:
|
|
state.bindings.alertInfo = .init(id: .unknown)
|
|
}
|
|
}
|
|
|
|
private func updateMembers() {
|
|
Task.detached {
|
|
await self.roomProxy.updateMembers()
|
|
}
|
|
}
|
|
|
|
private func displayFullScreenAvatar(_ url: URL) async {
|
|
guard let roomMemberProxy else {
|
|
fatalError()
|
|
}
|
|
|
|
let loadingIndicatorIdentifier = "roomMemberAvatarLoadingIndicator"
|
|
userIndicatorController.submitIndicator(UserIndicator(id: loadingIndicatorIdentifier, type: .modal, title: L10n.commonLoading, persistent: true))
|
|
defer { userIndicatorController.retractIndicatorWithId(loadingIndicatorIdentifier) }
|
|
|
|
// We don't actually know the mime type here, assume it's an image.
|
|
if let mediaSource = try? MediaSourceProxy(url: url, mimeType: "image/jpeg"),
|
|
case let .success(file) = await userSession.mediaProvider.loadFileFromSource(mediaSource) {
|
|
state.bindings.mediaPreviewItem = MediaPreviewItem(file: file, title: roomMemberProxy.displayName)
|
|
}
|
|
}
|
|
|
|
private func openDirectChat() {
|
|
guard let roomMemberProxy else { fatalError() }
|
|
|
|
let loadingIndicatorIdentifier = "openDirectChatLoadingIndicator"
|
|
userIndicatorController.submitIndicator(UserIndicator(id: loadingIndicatorIdentifier,
|
|
type: .modal(progress: .indeterminate, interactiveDismissDisabled: true, allowsInteraction: false),
|
|
title: L10n.commonLoading,
|
|
persistent: true),
|
|
delay: .milliseconds(200))
|
|
defer { userIndicatorController.retractIndicatorWithId(loadingIndicatorIdentifier) }
|
|
|
|
switch userSession.clientProxy.directRoomForUserID(roomMemberProxy.userID) {
|
|
case .success(let roomID):
|
|
if let roomID {
|
|
actionsSubject.send(.openDirectChat(roomID: roomID))
|
|
} else if roomProxy.details.historySharingState != RoomHistorySharingState.hidden {
|
|
Task {
|
|
let identity = await self.userSession.clientProxy.userIdentity(for: roomMemberProxy.userID, fallBackToServer: false)
|
|
let user: UserProfileProxy = .init(userID: roomMemberProxy.userID, displayName: roomMemberProxy.displayName, avatarURL: roomMemberProxy.avatarURL)
|
|
let isUnknown = if case .success(let identity) = identity {
|
|
identity == nil
|
|
} else {
|
|
true
|
|
}
|
|
self.state.bindings.inviteConfirmationUser = .init(user: user, isUnknown: isUnknown)
|
|
}
|
|
} else {
|
|
state.bindings.inviteConfirmationUser = .init(user: .init(userID: roomMemberProxy.userID, displayName: roomMemberProxy.displayName, avatarURL: roomMemberProxy.avatarURL), isUnknown: false)
|
|
}
|
|
case .failure:
|
|
state.bindings.alertInfo = .init(id: .failedOpeningDirectChat)
|
|
}
|
|
}
|
|
|
|
private func createDirectChat() async {
|
|
guard let roomMemberProxy else { fatalError() }
|
|
|
|
let loadingIndicatorIdentifier = "createDirectChatLoadingIndicator"
|
|
userIndicatorController.submitIndicator(UserIndicator(id: loadingIndicatorIdentifier,
|
|
type: .modal(progress: .indeterminate, interactiveDismissDisabled: true, allowsInteraction: false),
|
|
title: L10n.commonLoading,
|
|
persistent: true),
|
|
delay: .milliseconds(200))
|
|
defer { userIndicatorController.retractIndicatorWithId(loadingIndicatorIdentifier) }
|
|
|
|
switch await userSession.clientProxy.createDirectRoom(with: roomMemberProxy.userID, expectedRoomName: roomMemberProxy.displayName) {
|
|
case .success(let roomID):
|
|
analytics.trackCreatedRoom(isDM: true)
|
|
actionsSubject.send(.openDirectChat(roomID: roomID))
|
|
case .failure:
|
|
state.bindings.alertInfo = .init(id: .failedOpeningDirectChat)
|
|
}
|
|
}
|
|
|
|
private func startCall(roomID: String, isVoiceCall: Bool) async {
|
|
guard case let .joined(roomProxy) = await userSession.clientProxy.roomForIdentifier(roomID) else {
|
|
showErrorIndicator()
|
|
return
|
|
}
|
|
actionsSubject.send(.startCall(roomProxy: roomProxy, isVoiceCall: isVoiceCall))
|
|
}
|
|
|
|
// MARK: User Indicators
|
|
|
|
private var loadingIndicatorIdentifier: String {
|
|
"\(Self.self)-Loading"
|
|
}
|
|
|
|
private var statusIndicatorIdentifier: String {
|
|
"\(Self.self)-Status"
|
|
}
|
|
|
|
private func showMemberLoadingIndicator() {
|
|
userIndicatorController.submitIndicator(UserIndicator(id: loadingIndicatorIdentifier,
|
|
type: .modal(progress: .indeterminate, interactiveDismissDisabled: false, allowsInteraction: true),
|
|
title: L10n.commonLoading,
|
|
persistent: true),
|
|
delay: .milliseconds(100))
|
|
}
|
|
|
|
private func hideMemberLoadingIndicator() {
|
|
userIndicatorController.retractIndicatorWithId(loadingIndicatorIdentifier)
|
|
}
|
|
|
|
private func showErrorIndicator() {
|
|
userIndicatorController.submitIndicator(UserIndicator(id: statusIndicatorIdentifier,
|
|
type: .toast,
|
|
title: L10n.errorUnknown,
|
|
iconName: "xmark"))
|
|
}
|
|
}
|