// // Copyright 2022-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 Foundation import CommonKit /// AllChatsCoordinator input parameters class AllChatsCoordinatorParameters { let userSessionsService: UserSessionsService let appNavigator: AppNavigatorProtocol init(userSessionsService: UserSessionsService, appNavigator: AppNavigatorProtocol) { self.userSessionsService = userSessionsService self.appNavigator = appNavigator } } class AllChatsCoordinator: NSObject, SplitViewMasterCoordinatorProtocol { // MARK: Properties // MARK: Private private let parameters: AllChatsCoordinatorParameters private let activityIndicatorPresenter: ActivityIndicatorPresenterType private let indicatorPresenter: UserIndicatorTypePresenterProtocol private let userIndicatorStore: UserIndicatorStore private var appStateIndicatorCancel: UserIndicatorCancel? private var appSateIndicator: UserIndicator? private var maintenanceTimer: Timer? // Indicate if the Coordinator has started once private var hasStartedOnce: Bool { return self.allChatsViewController != nil } // TODO: Move MasterTabBarController navigation code here private var allChatsViewController: AllChatsViewController! // TODO: Embed UINavigationController in each tab like recommended by Apple and remove these properties. UITabBarViewController shoud not be embed in a UINavigationController (https://github.com/vector-im/riot-ios/issues/3086). private let navigationRouter: NavigationRouterType private var currentSpaceId: String? private weak var versionCheckCoordinator: VersionCheckCoordinator? private var currentMatrixSession: MXSession? { return parameters.userSessionsService.mainUserSession?.matrixSession } private var isAllChatsControllerTopMostController: Bool { return self.navigationRouter.modules.last is AllChatsViewController } private var detailUserIndicatorPresenter: UserIndicatorTypePresenterProtocol { guard let presenter = splitViewMasterPresentableDelegate?.detailUserIndicatorPresenter else { MXLog.debug("[AllChatsCoordinator]: Missing defautl user indicator presenter") return UserIndicatorTypePresenter(presentingViewController: toPresentable()) } return presenter } private var indicators = [UserIndicator]() private var signOutFlowPresenter: SignOutFlowPresenter? // MARK: Public // Must be used only internally var childCoordinators: [Coordinator] = [] weak var delegate: SplitViewMasterCoordinatorDelegate? weak var splitViewMasterPresentableDelegate: SplitViewMasterPresentableDelegate? // MARK: - Setup init(parameters: AllChatsCoordinatorParameters) { self.parameters = parameters let masterNavigationController = RiotNavigationController() self.navigationRouter = NavigationRouter(navigationController: masterNavigationController) self.activityIndicatorPresenter = ActivityIndicatorPresenter() self.indicatorPresenter = UserIndicatorTypePresenter(presentingViewController: masterNavigationController) self.userIndicatorStore = UserIndicatorStore(presenter: indicatorPresenter) } // MARK: - Public methods func start() { self.start(with: nil) } func start(with spaceId: String?) { // If start has been done once do not setup view controllers again if self.hasStartedOnce == false { let allChatsViewController = AllChatsViewController.instantiate() allChatsViewController.allChatsDelegate = self allChatsViewController.userIndicatorStore = UserIndicatorStore(presenter: indicatorPresenter) createLeftButtonItem(for: allChatsViewController) self.allChatsViewController = allChatsViewController self.navigationRouter.setRootModule(allChatsViewController) // Add existing Matrix sessions if any for userSession in self.parameters.userSessionsService.userSessions { self.addMatrixSessionToAllChatsController(userSession.matrixSession) } self.registerUserSessionsServiceNotifications() self.registerSessionChange() let versionCheckCoordinator = createVersionCheckCoordinator(withRootViewController: allChatsViewController, bannerPresentrer: allChatsViewController) versionCheckCoordinator.start() self.add(childCoordinator: versionCheckCoordinator) } self.allChatsViewController?.switchSpace(withId: spaceId) self.currentSpaceId = spaceId } func toPresentable() -> UIViewController { return self.navigationRouter.toPresentable() } func releaseSelectedItems() { self.allChatsViewController.releaseSelectedItem() } func pinUnlocked() { onBWIAppStart() } func bwiOnUnlockedByPin() { } func popToHome(animated: Bool, completion: (() -> Void)?) { // Force back to the main screen if this is not the one that is displayed if allChatsViewController != self.navigationRouter.modules.last?.toPresentable() { // Listen to the masterNavigationController changes // We need to be sure that allChatsViewController is back to the screen // If the AllChatsViewController is not visible because there is a modal above it // but still the top view controller of navigation controller if self.isAllChatsControllerTopMostController { completion?() } else { // Otherwise AllChatsViewController is not the top controller of the navigation controller // Waiting for `self.navigationRouter` popping to AllChatsViewController var token: NSObjectProtocol? token = NotificationCenter.default.addObserver(forName: NavigationRouter.didPopModule, object: self.navigationRouter, queue: OperationQueue.main) { [weak self] (notification) in guard let self = self else { return } // If AllChatsViewController is now the top most controller in navigation controller stack call the completion if self.isAllChatsControllerTopMostController { completion?() if let token = token { NotificationCenter.default.removeObserver(token) } } } // Pop to root view controller self.navigationRouter.popToRootModule(animated: animated) } } else { // the AllChatsViewController is already visible completion?() } } func showErroIndicator(with error: Error) { let error = error as NSError // Ignore fake error, or connection cancellation error guard error.domain != NSURLErrorDomain || error.code != NSURLErrorCancelled else { return } // Ignore GDPR Consent not given error. Already caught by kMXHTTPClientUserConsentNotGivenErrorNotification observation let mxError = MXError.isMXError(error) ? MXError(nsError: error) : nil guard mxError?.errcode != kMXErrCodeStringConsentNotGiven else { return } let msg = error.userInfo[NSLocalizedFailureReasonErrorKey] as? String let localizedDescription = error.userInfo[NSLocalizedDescriptionKey] as? String let title = (error.userInfo[NSLocalizedFailureReasonErrorKey] as? String) ?? (msg ?? (localizedDescription ?? VectorL10n.error)) indicators.append(self.indicatorPresenter.present(.failure(label: title))) } func showAppStateIndicator(with text: String, icon: UIImage?) { hideAppStateIndicator() appSateIndicator = self.indicatorPresenter.present(.custom(label: text, icon: icon)) } func hideAppStateIndicator() { appSateIndicator?.cancel() appSateIndicator = nil } func onLogout() { onBWILogout() } // MARK: - SplitViewMasterPresentable var selectedNavigationRouter: NavigationRouterType? { return self.navigationRouter } // MARK: Split view /// If the split view is collapsed (one column visible) it will push the Presentable on the primary navigation controller, otherwise it will show the Presentable as the secondary view of the split view. private func replaceSplitViewDetails(with presentable: Presentable, popCompletion: (() -> Void)? = nil) { self.splitViewMasterPresentableDelegate?.splitViewMasterPresentable(self, wantsToReplaceDetailWith: presentable, popCompletion: popCompletion) } /// If the split view is collapsed (one column visible) it will push the Presentable on the primary navigation controller, otherwise it will show the Presentable as the secondary view of the split view on top of existing views. private func stackSplitViewDetails(with presentable: Presentable, popCompletion: (() -> Void)? = nil) { self.splitViewMasterPresentableDelegate?.splitViewMasterPresentable(self, wantsToStack: presentable, popCompletion: popCompletion) } private func showSplitViewDetails(with presentable: Presentable, stackedOnSplitViewDetail: Bool, popCompletion: (() -> Void)? = nil) { if stackedOnSplitViewDetail { self.stackSplitViewDetails(with: presentable, popCompletion: popCompletion) } else { self.replaceSplitViewDetails(with: presentable, popCompletion: popCompletion) } } private func showSplitViewDetails(with modules: [NavigationModule], stack: Bool) { if stack { self.splitViewMasterPresentableDelegate?.splitViewMasterPresentable(self, wantsToStack: modules) } else { self.splitViewMasterPresentableDelegate?.splitViewMasterPresentable(self, wantsToReplaceDetailsWith: modules) } } private func resetSplitViewDetails() { self.splitViewMasterPresentableDelegate?.splitViewMasterPresentableWantsToResetDetail(self) } // MARK: UserSessions management private func registerUserSessionsServiceNotifications() { // Listen only notifications from the current UserSessionsService instance let userSessionService = self.parameters.userSessionsService NotificationCenter.default.addObserver(self, selector: #selector(userSessionsServiceDidAddUserSession(_:)), name: UserSessionsService.didAddUserSession, object: userSessionService) NotificationCenter.default.addObserver(self, selector: #selector(userSessionsServiceWillRemoveUserSession(_:)), name: UserSessionsService.willRemoveUserSession, object: userSessionService) } @objc private func userSessionsServiceDidAddUserSession(_ notification: Notification) { guard let userSession = notification.userInfo?[UserSessionsService.NotificationUserInfoKey.userSession] as? UserSession else { return } self.addMatrixSessionToAllChatsController(userSession.matrixSession) } @objc private func userSessionsServiceWillRemoveUserSession(_ notification: Notification) { guard let userSession = notification.userInfo?[UserSessionsService.NotificationUserInfoKey.userSession] as? UserSession else { return } self.removeMatrixSessionFromAllChatsController(userSession.matrixSession) } // MARK: - Matrix Session management // TODO: Remove Matrix session handling from the view controller private func addMatrixSessionToAllChatsController(_ matrixSession: MXSession) { MXLog.debug("[TabBarCoordinator] masterTabBarController.addMatrixSession") self.allChatsViewController.addMatrixSession(matrixSession) } // TODO: Remove Matrix session handling from the view controller private func removeMatrixSessionFromAllChatsController(_ matrixSession: MXSession) { MXLog.debug("[TabBarCoordinator] masterTabBarController.removeMatrixSession") self.allChatsViewController.removeMatrixSession(matrixSession) } private func registerSessionChange() { NotificationCenter.default.addObserver(self, selector: #selector(sessionDidSync(_:)), name: NSNotification.Name.mxSessionDidSync, object: nil) } @objc private func sessionDidSync(_ notification: Notification) { updateAvatarButtonItem() } // MARK: Navigation private func showSettings() { let viewController = self.createSettingsViewController() self.navigationRouter.push(viewController, animated: true, popCompletion: nil) } private func showContactDetails(with contact: MXKContact, presentationParameters: ScreenPresentationParameters) { let coordinatorParameters = ContactDetailsCoordinatorParameters(contact: contact) let coordinator = ContactDetailsCoordinator(parameters: coordinatorParameters) coordinator.start() self.add(childCoordinator: coordinator) self.showSplitViewDetails(with: coordinator, stackedOnSplitViewDetail: presentationParameters.stackAboveVisibleViews) { [weak self] in self?.remove(childCoordinator: coordinator) } } // MARK: Navigation bar items management private weak var avatarMenuView: AvatarView? private weak var avatarMenuButton: UIButton? private func createLeftButtonItem(for viewController: UIViewController) { DispatchQueue.main.async { self.createAvatarButtonItem(for: viewController) } } private var avatarMenu: UIMenu { var actions: [UIMenuElement] = [] actions.append(UIAction(title: VectorL10n.allChatsUserMenuSettings, image: UIImage(systemName: "gearshape")) { [weak self] action in self?.showSettings() }) var subMenuActions: [UIAction] = [] if BWIBuildSettings.shared.sideMenuShowInviteFriends { subMenuActions.append(UIAction(title: VectorL10n.sideMenuActionInviteFriends, image: UIImage(systemName: "square.and.arrow.up.fill")) { [weak self] action in guard let self = self else { return } self.showInviteFriends(from: self.avatarMenuButton) }) } subMenuActions.append(UIAction(title: VectorL10n.sideMenuActionFeedback, image: UIImage(systemName: "questionmark.circle")) { [weak self] action in self?.showBugReport() }) actions.append(UIMenu(title: "", options: .displayInline, children: subMenuActions)) actions.append(UIMenu(title: "", options: .displayInline, children: [ UIAction(title: VectorL10n.settingsSignOut, image: UIImage(systemName: "rectangle.portrait.and.arrow.right.fill"), attributes: .destructive) { [weak self] action in self?.signOut() } ])) return UIMenu(options: .displayInline, children: actions) } private func createAvatarButtonItem(for viewController: UIViewController) { let view = UIView(frame: CGRect(x: 0, y: 0, width: 36, height: 36)) view.backgroundColor = .clear let avatarInsets: UIEdgeInsets = .init(top: 7, left: 7, bottom: 7, right: 7) var button: UIButton = .init(frame: view.bounds) button.imageEdgeInsets = avatarInsets // bwi: 4704 - ui improvements let gearshapeImage = Asset.Images.bwiSettingsFilled.image button.setImage(gearshapeImage, for: .normal) button.tintColor = ThemeService.shared().theme.tintColor if BWIBuildSettings.shared.enableSideMenu { button.menu = avatarMenu button.showsMenuAsPrimaryAction = true } else { button.addTarget(self, action: #selector(bwiAvatarAction), for: .touchUpInside) } button.autoresizingMask = [.flexibleHeight, .flexibleWidth] button.accessibilityLabel = VectorL10n.allChatsUserMenuAccessibilityLabel if ServerDowntimeDefaultService.shared.isDowntimePresentable() { button = ServerDowntimeBadge().applyBadgeToButton(button: button, color: ServerDowntimeDefaultService.shared.downtimeColor()) } else if BWIBuildSettings.shared.bwiCheckAppVersion && ValidAppVersionsDefaultService.shared.isCurrentAppVersionOutdated() { button = ServerDowntimeBadge().applyBadgeToButton(button: button, color: .yellow) } view.addSubview(button) self.avatarMenuButton = button let avatarView = UserAvatarView(frame: view.bounds.inset(by: avatarInsets)) avatarView.isUserInteractionEnabled = false avatarView.update(theme: ThemeService.shared().theme) avatarView.autoresizingMask = [.flexibleTopMargin, .flexibleBottomMargin] view.addSubview(avatarView) view.bringSubviewToFront(button) NSLayoutConstraint.activate([ view.widthAnchor.constraint(equalToConstant: 36), view.heightAnchor.constraint(equalToConstant: 36) ]) self.avatarMenuView = avatarView updateAvatarButtonItem() viewController.navigationItem.leftBarButtonItem = UIBarButtonItem(customView: view) } private func updateAvatarButtonItem() { // bwi: 4704 - ui improvements (disable view updates) // MXLog.info("[AllChatsCoordinator] updating avatar button item.") // if let avatar = userAvatarViewData(from: currentMatrixSession) { // if avatarMenuView == nil { // MXLog.warning("[AllChatsCoordinator] updateAvatarButtonItem: avatarMenuView is nil.") // } // avatarMenuView?.fill(with: avatar) // avatarMenuButton?.setImage(nil, for: .normal) // } else { // avatarMenuButton?.setImage(Asset.Images.tabPeople.image, for: .normal) // } } private func showRoom(withId roomId: String, eventId: String? = nil) { guard let matrixSession = self.parameters.userSessionsService.mainUserSession?.matrixSession else { return } self.showRoom(with: roomId, eventId: eventId, matrixSession: matrixSession) } private func showRoom(withNavigationParameters roomNavigationParameters: RoomNavigationParameters, completion: (() -> Void)?) { if let threadParameters = roomNavigationParameters.threadParameters, threadParameters.stackRoomScreen { showRoomAndThread(with: roomNavigationParameters, completion: completion) } else { let threadId = roomNavigationParameters.threadParameters?.threadId let displayConfig: RoomDisplayConfiguration if threadId != nil { displayConfig = .forThreads } else { displayConfig = .default } let roomCoordinatorParameters = RoomCoordinatorParameters(navigationRouterStore: NavigationRouterStore.shared, userIndicatorPresenter: detailUserIndicatorPresenter, session: roomNavigationParameters.mxSession, parentSpaceId: self.currentSpaceId, roomId: roomNavigationParameters.roomId, eventId: roomNavigationParameters.eventId, threadId: threadId, userId: roomNavigationParameters.userId, showSettingsInitially: roomNavigationParameters.showSettingsInitially, displayConfiguration: displayConfig, autoJoinInvitedRoom: roomNavigationParameters.autoJoinInvitedRoom) self.showRoom(with: roomCoordinatorParameters, stackOnSplitViewDetail: roomNavigationParameters.presentationParameters.stackAboveVisibleViews, completion: completion) } } private func showRoom(with roomId: String, eventId: String?, matrixSession: MXSession, completion: (() -> Void)? = nil) { // RoomCoordinator will be presented by the split view. // As we don't know which navigation controller instance will be used, // give the NavigationRouterStore instance and let it find the associated navigation controller let roomCoordinatorParameters = RoomCoordinatorParameters(navigationRouterStore: NavigationRouterStore.shared, userIndicatorPresenter: detailUserIndicatorPresenter, session: matrixSession, parentSpaceId: self.currentSpaceId, roomId: roomId, eventId: eventId, showSettingsInitially: false) self.showRoom(with: roomCoordinatorParameters, completion: completion) } private func showRoomPreview(with previewData: RoomPreviewData) { // RoomCoordinator will be presented by the split view // We don't which navigation controller instance will be used // Give the NavigationRouterStore instance and let it find the associated navigation controller if needed let roomCoordinatorParameters = RoomCoordinatorParameters(navigationRouterStore: NavigationRouterStore.shared, userIndicatorPresenter: detailUserIndicatorPresenter, parentSpaceId: self.currentSpaceId, previewData: previewData) self.showRoom(with: roomCoordinatorParameters) } private func showRoomPreview(withNavigationParameters roomPreviewNavigationParameters: RoomPreviewNavigationParameters, completion: (() -> Void)?) { let roomCoordinatorParameters = RoomCoordinatorParameters(navigationRouterStore: NavigationRouterStore.shared, userIndicatorPresenter: detailUserIndicatorPresenter, parentSpaceId: self.currentSpaceId, previewData: roomPreviewNavigationParameters.previewData) self.showRoom(with: roomCoordinatorParameters, stackOnSplitViewDetail: roomPreviewNavigationParameters.presentationParameters.stackAboveVisibleViews, completion: completion) } private func showRoom(with parameters: RoomCoordinatorParameters, stackOnSplitViewDetail: Bool = false, completion: (() -> Void)? = nil) { // try to find the desired room screen in the stack if let roomCoordinator = self.splitViewMasterPresentableDelegate?.detailModules.last(where: { presentable in guard let roomCoordinator = presentable as? RoomCoordinatorProtocol else { return false } return roomCoordinator.roomId == parameters.roomId && roomCoordinator.threadId == parameters.threadId && roomCoordinator.mxSession == parameters.session }) as? RoomCoordinatorProtocol { self.splitViewMasterPresentableDelegate?.splitViewMasterPresentable(self, wantsToPopTo: roomCoordinator) // go to a specific event if provided if let eventId = parameters.eventId { roomCoordinator.start(withEventId: eventId, completion: completion) } else { completion?() } return } let coordinator = RoomCoordinator(parameters: parameters) coordinator.delegate = self coordinator.start(withCompletion: completion) self.add(childCoordinator: coordinator) self.showSplitViewDetails(with: coordinator, stackedOnSplitViewDetail: stackOnSplitViewDetail) { [weak self] in // NOTE: The RoomDataSource releasing is handled in SplitViewCoordinator self?.remove(childCoordinator: coordinator) } } private func showRoomAndThread(with roomNavigationParameters: RoomNavigationParameters, completion: (() -> Void)? = nil) { self.activityIndicatorPresenter.presentActivityIndicator(on: toPresentable().view, animated: false) let dispatchGroup = DispatchGroup() // create room coordinator let roomCoordinatorParameters = RoomCoordinatorParameters(navigationRouterStore: NavigationRouterStore.shared, userIndicatorPresenter: detailUserIndicatorPresenter, session: roomNavigationParameters.mxSession, parentSpaceId: self.currentSpaceId, roomId: roomNavigationParameters.roomId, eventId: nil, threadId: nil, showSettingsInitially: false) dispatchGroup.enter() let roomCoordinator = RoomCoordinator(parameters: roomCoordinatorParameters) roomCoordinator.delegate = self roomCoordinator.start { dispatchGroup.leave() } self.add(childCoordinator: roomCoordinator) // create thread coordinator let threadCoordinatorParameters = RoomCoordinatorParameters(navigationRouterStore: NavigationRouterStore.shared, userIndicatorPresenter: detailUserIndicatorPresenter, session: roomNavigationParameters.mxSession, parentSpaceId: self.currentSpaceId, roomId: roomNavigationParameters.roomId, eventId: roomNavigationParameters.eventId, threadId: roomNavigationParameters.threadParameters?.threadId, showSettingsInitially: false, displayConfiguration: .forThreads) dispatchGroup.enter() let threadCoordinator = RoomCoordinator(parameters: threadCoordinatorParameters) threadCoordinator.delegate = self threadCoordinator.start { dispatchGroup.leave() } self.add(childCoordinator: threadCoordinator) dispatchGroup.notify(queue: .main) { [weak self] in guard let self = self else { return } let modules: [NavigationModule] = [ NavigationModule(presentable: roomCoordinator, popCompletion: { [weak self] in // NOTE: The RoomDataSource releasing is handled in SplitViewCoordinator self?.remove(childCoordinator: roomCoordinator) }), NavigationModule(presentable: threadCoordinator, popCompletion: { [weak self] in // NOTE: The RoomDataSource releasing is handled in SplitViewCoordinator self?.remove(childCoordinator: threadCoordinator) }) ] self.showSplitViewDetails(with: modules, stack: roomNavigationParameters.presentationParameters.stackAboveVisibleViews) self.activityIndicatorPresenter.removeCurrentActivityIndicator(animated: true) } } // MARK: Sign out process private func signOut() { guard let session = currentMatrixSession else { MXLog.warning("[AllChatsCoordinator] Unable to sign out due to missing current session.") return } let flowPresenter = SignOutFlowPresenter(session: session, presentingViewController: toPresentable()) flowPresenter.delegate = self flowPresenter.start(sourceView: avatarMenuButton) self.signOutFlowPresenter = flowPresenter } // MARK: - Private methods private func createVersionCheckCoordinator(withRootViewController rootViewController: UIViewController, bannerPresentrer: BannerPresentationProtocol) -> VersionCheckCoordinator { let versionCheckCoordinator = VersionCheckCoordinator(rootViewController: rootViewController, bannerPresenter: bannerPresentrer, themeService: ThemeService.shared()) return versionCheckCoordinator } private func showInviteFriends(from sourceView: UIView?) { let myUserId = self.parameters.userSessionsService.mainUserSession?.userId ?? "" let inviteFriendsPresenter = InviteFriendsPresenter() inviteFriendsPresenter.present(for: myUserId, from: self.navigationRouter.toPresentable(), sourceView: sourceView, animated: true) } private func showBugReport() { let bugReportViewController = BugReportViewController() // Show in fullscreen to animate presentation along side menu dismiss bugReportViewController.modalPresentationStyle = .fullScreen bugReportViewController.modalTransitionStyle = .crossDissolve self.navigationRouter.present(bugReportViewController, animated: true) } private func userAvatarViewData(from mxSession: MXSession?) -> UserAvatarViewData? { guard let mxSession = mxSession, let userId = mxSession.myUserId, let mediaManager = mxSession.mediaManager, let myUser = mxSession.myUser else { return nil } let userDisplayName = myUser.displayname let avatarUrl = myUser.avatarUrl return UserAvatarViewData(userId: userId, displayName: userDisplayName, avatarUrl: avatarUrl, mediaManager: mediaManager) } private func createUnifiedSearchController() -> UnifiedSearchViewController { let viewController: UnifiedSearchViewController = UnifiedSearchViewController.instantiate() viewController.loadViewIfNeeded() for userSession in self.parameters.userSessionsService.userSessions { viewController.addMatrixSession(userSession.matrixSession) } return viewController } private func createSettingsViewController() -> SettingsViewController { let viewController: SettingsViewController = SettingsViewController.instantiate() viewController.loadViewIfNeeded() return viewController } // bwi: directly call settings if sidemenu disables @objc private func bwiAvatarAction(sender: UIButton!) { self.showSettings() } // bwi: check for personalnotes room existence private func bwiCheckForPersonalNotesRoom() { if let session = self.currentMatrixSession { let service = PersonalNotesDefaultService(mxSession: session) if BWIBuildSettings.shared.bwiResetPersonalNotesAccountData { service.resetPersonalNotesRoom() } service.createPersonalNotesRoomIfNeeded() service.setAsFavoriteIfNeeded() } } // bwi: check if matomo promt was shown for this session otherwise check if config has changed private func bwiCheckForMatomoPromt() { if BWIBuildSettings.shared.bwiMatomoEnabled && BWIAnalytics.sharedTracker.needsToShowPromt() { self.allChatsViewController.bwiPresentMatomoConsentAlert() } else { // bwi: 5706 show federation announcement promt self.allChatsViewController.bwiCheckForFederationAnnouncementPromt() // bwi: 5660 introduce federation self.allChatsViewController.presentFederationIntroductionSheet() // bwi: 6570 Show birthday self.allChatsViewController.bwiCheckForBirthdayScreen() } } // bwi: check if there are changes in maintenance status @objc private func checkMaintenanceStatus() { // bwi: bwi specific if BWIBuildSettings.shared.useTestDataForDowntime { ServerDowntimeDefaultService.shared.fetchDowntimes(completion: { self.createLeftButtonItem(for: self.allChatsViewController) if ServerDowntimeDefaultService.shared.isBlocking() && ServerDowntimeDefaultService.shared.isDowntimeNow() && !ServerDowntimeDefaultService.shared.isManuallyIgnored() { UserDefaults.standard.set(true, forKey: "ServerDownTimeBlockingKey") } else { UserDefaults.standard.set(false, forKey: "ServerDownTimeBlockingKey") } DispatchQueue.main.async { self.allChatsViewController.checkAppVersionDeprecated() self.allChatsViewController.checkAppVersionOutdated() } }) } else { ServerDowntimeDefaultService.shared.fetchDowntimesWithDirectRequest(completion: { success, _, _, _ in self.createLeftButtonItem(for: self.allChatsViewController) if ServerDowntimeDefaultService.shared.isBlocking() && ServerDowntimeDefaultService.shared.isDowntimeNow() && !ServerDowntimeDefaultService.shared.isManuallyIgnored() { UserDefaults.standard.set(true, forKey: "ServerDownTimeBlockingKey") } else { UserDefaults.standard.set(false, forKey: "ServerDownTimeBlockingKey") } DispatchQueue.main.async { self.allChatsViewController.checkAppVersionDeprecated() self.allChatsViewController.checkAppVersionOutdated() } }) } } private func enableMaintenanceTimer(_ enable: Bool ) { maintenanceTimer?.invalidate() if enable { maintenanceTimer = Timer.scheduledTimer(timeInterval: 10.0, target: self, selector: #selector(checkMaintenanceStatus), userInfo: nil, repeats: true) } } private func onBWIAppStart() { checkMaintenanceStatus() enableMaintenanceTimer(true) BWIAnalytics.sharedTracker.readUserConfig() if BWIBuildSettings.shared.bwiPersonalNotesRoom { self.bwiCheckForPersonalNotesRoom() } // bwi #4478: refresh wellknown a bit more often self.currentMatrixSession?.refreshHomeserverWellknown(false, success: { wellknown in // bwi: #5706 fix crash: only show matomo alert when wellknown is available wellknown?.updateFederationStatus() // BWI #7564 add migration level if let migrationLevel = wellknown?.migrationInfoLevel() { BWIBuildSettings.shared.BuMXMigrationInfoLevel = migrationLevel // BWI #7555 migration part 3 if migrationLevel == 3 { // Inform Apple that this app on this device can be ignored UIApplication.shared.unregisterForRemoteNotifications() } // BWI #7564 END } // BWI #7564 END self.bwiCheckForMatomoPromt() }, failure: nil) } private func onBWILogout() { // bwi #5896 disable periodic maintenance fetching when user not logged in == No selected server to fetch from enableMaintenanceTimer(false) } } extension AllChatsCoordinator: SignOutFlowPresenterDelegate { func signOutFlowPresenterDidStartLoading(_ presenter: SignOutFlowPresenter) { allChatsViewController.view.isUserInteractionEnabled = false allChatsViewController.startActivityIndicator() } func signOutFlowPresenterDidStopLoading(_ presenter: SignOutFlowPresenter) { allChatsViewController.view.isUserInteractionEnabled = true allChatsViewController.stopActivityIndicator() } func signOutFlowPresenter(_ presenter: SignOutFlowPresenter, didFailWith error: Error) { AppDelegate.theDelegate().showError(asAlert: error) } } // MARK: - AllChatsViewControllerDelegate extension AllChatsCoordinator: AllChatsViewControllerDelegate { func allChatsViewControllerDidCompleteAuthentication(_ allChatsViewController: AllChatsViewController) { self.delegate?.splitViewMasterCoordinatorDidCompleteAuthentication(self) // BWI: #5706 fix alert triggering for new users. Only trigger onBWIAppStart when session has recovery set guard let matrixSession = self.parameters.userSessionsService.mainUserSession?.matrixSession else { return } guard let crypto = matrixSession.crypto else { return } if crypto.recoveryService.hasRecovery() { onBWIAppStart() } } func allChatsViewController(_ allChatsViewController: AllChatsViewController, didSelectRoomWithParameters roomNavigationParameters: RoomNavigationParameters, completion: @escaping () -> Void) { self.showRoom(withNavigationParameters: roomNavigationParameters, completion: completion) } func allChatsViewController(_ allChatsViewController: AllChatsViewController, didSelectRoomPreviewWithParameters roomPreviewNavigationParameters: RoomPreviewNavigationParameters, completion: (() -> Void)?) { self.showRoomPreview(withNavigationParameters: roomPreviewNavigationParameters, completion: completion) } func allChatsViewController(_ allChatsViewController: AllChatsViewController, didSelectContact contact: MXKContact, with presentationParameters: ScreenPresentationParameters) { self.showContactDetails(with: contact, presentationParameters: presentationParameters) } } // MARK: - RoomCoordinatorDelegate extension AllChatsCoordinator: RoomCoordinatorDelegate { func roomCoordinatorDidDismissInteractively(_ coordinator: RoomCoordinatorProtocol) { self.remove(childCoordinator: coordinator) } func roomCoordinatorDidLeaveRoom(_ coordinator: RoomCoordinatorProtocol) { // For the moment when a room is left, reset the split detail with placeholder self.resetSplitViewDetails() indicatorPresenter .present(.success(label: VectorL10n.roomParticipantsLeaveSuccess)) .store(in: &indicators) } func roomCoordinatorDidCancelRoomPreview(_ coordinator: RoomCoordinatorProtocol) { self.navigationRouter.popModule(animated: true) } func roomCoordinator(_ coordinator: RoomCoordinatorProtocol, didSelectRoomWithId roomId: String, eventId: String?) { self.showRoom(withId: roomId, eventId: eventId) } func roomCoordinator(_ coordinator: RoomCoordinatorProtocol, didReplaceRoomWithReplacementId roomId: String) { guard let matrixSession = self.parameters.userSessionsService.mainUserSession?.matrixSession else { return } let roomCoordinatorParameters = RoomCoordinatorParameters(navigationRouterStore: NavigationRouterStore.shared, userIndicatorPresenter: detailUserIndicatorPresenter, session: matrixSession, parentSpaceId: self.currentSpaceId, roomId: roomId, eventId: nil, showSettingsInitially: true) self.showRoom(with: roomCoordinatorParameters, stackOnSplitViewDetail: false) } func roomCoordinatorDidCancelNewDirectChat(_ coordinator: RoomCoordinatorProtocol) { self.navigationRouter.popModule(animated: true) } }