Various fixes (#437)

* Fix information leaking on RoomSummaryDetails logging

* Prevent crashes when force quitting the application

* Cleanup crash detected alert presentation and exposed home screen view model actions

* Fixes #340 - Wait for logout confirmation before changing the app state

* Add changelogs

* Fix unit tests

* Add missing softLogout logout handling
This commit is contained in:
Stefan Ceriu
2023-01-11 15:10:26 +02:00
committed by GitHub
parent 02787d85f7
commit 9d80e79eca
11 changed files with 82 additions and 52 deletions

View File

@@ -85,6 +85,11 @@ class AppCoordinator: AppCoordinatorProtocol {
}
func start() {
guard stateMachine.state == .initial else {
MXLog.error("Received a start request when already started")
return
}
stateMachine.processEvent(userSessionStore.hasSessions ? .startWithExistingSession : .startWithAuthentication)
}
@@ -255,21 +260,27 @@ class AppCoordinator: AppCoordinatorProtocol {
deobserveUserSessionChanges()
if !isSoftLogout {
Task {
// first log out from the server
_ = await userSession.clientProxy.logout()
// regardless of the result, clear user data
userSessionStore.logout(userSession: userSession)
userSession = nil
notificationManager?.delegate = nil
notificationManager = nil
}
guard !isSoftLogout else {
stateMachine.processEvent(.completedSigningOut)
return
}
// complete logging out
stateMachine.processEvent(.completedSigningOut)
Task {
showLoadingIndicator()
// first log out from the server
_ = await userSession.clientProxy.logout()
// regardless of the result, clear user data
userSessionStore.logout(userSession: userSession)
userSession = nil
notificationManager?.delegate = nil
notificationManager = nil
stateMachine.processEvent(.completedSigningOut)
hideLoadingIndicator()
}
}
private func presentSplashScreen(isSoftLogout: Bool = false) {

View File

@@ -61,6 +61,10 @@ class AppCoordinatorStateMachine {
private let stateMachine: StateMachine<State, Event>
var state: AppCoordinatorStateMachine.State {
stateMachine.state
}
init() {
stateMachine = StateMachine(state: .initial)
configure()

View File

@@ -25,7 +25,7 @@ struct HomeScreenCoordinatorParameters {
}
enum HomeScreenCoordinatorAction {
case presentRoomScreen(roomIdentifier: String)
case presentRoom(roomIdentifier: String)
case presentSettingsScreen
case presentFeedbackScreen
case presentSessionVerificationScreen
@@ -50,12 +50,18 @@ final class HomeScreenCoordinator: CoordinatorProtocol {
guard let self else { return }
switch action {
case .selectRoom(let roomIdentifier):
self.callback?(.presentRoomScreen(roomIdentifier: roomIdentifier))
case .userMenu(let action):
self.processUserMenuAction(action)
case .verifySession:
case .presentRoom(let roomIdentifier):
self.callback?(.presentRoom(roomIdentifier: roomIdentifier))
case .presentFeedbackScreen:
self.callback?(.presentFeedbackScreen)
case .presentSettingsScreen:
self.callback?(.presentSettingsScreen)
case .presentInviteFriendsScreen:
self.presentInviteFriends()
case .presentSessionVerificationScreen:
self.callback?(.presentSessionVerificationScreen)
case .signOut:
self.callback?(.signOut)
}
}
}
@@ -63,15 +69,11 @@ final class HomeScreenCoordinator: CoordinatorProtocol {
// MARK: - Public
func start() {
#if !DEBUG
if parameters.bugReportService.crashedLastRun {
viewModel.presentAlert(
AlertInfo(id: UUID(),
title: ElementL10n.sendBugReportAppCrashed,
primaryButton: .init(title: ElementL10n.no, action: nil),
secondaryButton: .init(title: ElementL10n.yes) { [weak self] in
self?.callback?(.presentFeedbackScreen)
}))
viewModel.presentCrashedLastRunAlert()
}
#endif
}
func toPresentable() -> AnyView {
@@ -79,19 +81,6 @@ final class HomeScreenCoordinator: CoordinatorProtocol {
}
// MARK: - Private
private func processUserMenuAction(_ action: HomeScreenViewUserMenuAction) {
switch action {
case .settings:
callback?(.presentSettingsScreen)
case .inviteFriends:
presentInviteFriends()
case .feedback:
callback?(.presentFeedbackScreen)
case .signOut:
callback?(.signOut)
}
}
private func presentInviteFriends() {
parameters.navigationStackCoordinator.setSheetCoordinator(InviteFriendsCoordinator(userId: parameters.userSession.userID))

View File

@@ -18,9 +18,12 @@ import Foundation
import UIKit
enum HomeScreenViewModelAction {
case selectRoom(roomIdentifier: String)
case userMenu(action: HomeScreenViewUserMenuAction)
case verifySession
case presentRoom(roomIdentifier: String)
case presentSessionVerificationScreen
case presentSettingsScreen
case presentInviteFriendsScreen
case presentFeedbackScreen
case signOut
}
enum HomeScreenViewUserMenuAction {

View File

@@ -112,6 +112,7 @@ class HomeScreenViewModel: HomeScreenViewModelType, HomeScreenViewModelProtocol
// MARK: - Public
// swiftlint:disable:next cyclomatic_complexity
override func process(viewAction: HomeScreenViewAction) async {
switch viewAction {
case .loadRoomData(let roomIdentifier):
@@ -119,11 +120,20 @@ class HomeScreenViewModel: HomeScreenViewModelType, HomeScreenViewModelProtocol
loadDataForRoomIdentifier(roomIdentifier)
}
case .selectRoom(let roomIdentifier):
callback?(.selectRoom(roomIdentifier: roomIdentifier))
callback?(.presentRoom(roomIdentifier: roomIdentifier))
case .userMenu(let action):
callback?(.userMenu(action: action))
switch action {
case .feedback:
callback?(.presentFeedbackScreen)
case .settings:
callback?(.presentSettingsScreen)
case .inviteFriends:
callback?(.presentInviteFriendsScreen)
case .signOut:
callback?(.signOut)
}
case .verifySession:
callback?(.verifySession)
callback?(.presentSessionVerificationScreen)
case .skipSessionVerification:
state.showSessionVerificationBanner = false
case .updatedVisibleItemIdentifiers(let identifiers):
@@ -131,8 +141,13 @@ class HomeScreenViewModel: HomeScreenViewModelType, HomeScreenViewModelProtocol
}
}
func presentAlert(_ alertInfo: AlertInfo<UUID>) {
state.bindings.alertInfo = alertInfo
func presentCrashedLastRunAlert() {
state.bindings.alertInfo = AlertInfo(id: UUID(),
title: ElementL10n.sendBugReportAppCrashed,
primaryButton: .init(title: ElementL10n.no, action: nil),
secondaryButton: .init(title: ElementL10n.yes) { [weak self] in
self?.callback?(.presentFeedbackScreen)
})
}
// MARK: - Private

View File

@@ -23,5 +23,5 @@ protocol HomeScreenViewModelProtocol {
var context: HomeScreenViewModelType.Context { get }
func presentAlert(_ alert: AlertInfo<UUID>)
func presentCrashedLastRunAlert()
}

View File

@@ -25,3 +25,9 @@ struct RoomSummaryDetails {
let lastMessageTimestamp: Date?
let unreadNotificationCount: UInt
}
extension RoomSummaryDetails: CustomStringConvertible {
var description: String {
"id: \"\(id)\", isDirect: \"\(isDirect)\", unreadNotificationCount: \"\(unreadNotificationCount)\""
}
}

View File

@@ -125,7 +125,7 @@ class UserSessionFlowCoordinator: CoordinatorProtocol {
guard let self else { return }
switch action {
case .presentRoomScreen(let roomIdentifier):
case .presentRoom(let roomIdentifier):
self.stateMachine.processEvent(.selectRoom(roomId: roomIdentifier))
case .presentSettingsScreen:
self.stateMachine.processEvent(.showSettingsScreen)

View File

@@ -35,7 +35,7 @@ class HomeScreenViewModelTests: XCTestCase {
var selectedRoomId = ""
viewModel.callback = { result in
switch result {
case .selectRoom(let roomId):
case .presentRoom(let roomId):
correctResult = true
selectedRoomId = roomId
default:
@@ -53,8 +53,8 @@ class HomeScreenViewModelTests: XCTestCase {
var correctResult = false
viewModel.callback = { result in
switch result {
case .userMenu(let action):
correctResult = action == .settings
case .presentSettingsScreen:
correctResult = true
default:
break
}

1
changelog.d/340.bugfix Normal file
View File

@@ -0,0 +1 @@
Wait for logout confirmation before changing the app state

View File

@@ -0,0 +1 @@
Prevent crash popups when force quitting the application