// File created from FlowTemplate // $ createRootCoordinator.sh Onboarding/SplashScreen Onboarding OnboardingSplashScreen /* Copyright 2021 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 Foundation @objcMembers class OnboardingCoordinatorBridgePresenterParameters: NSObject { /// The external registration parameters for AuthenticationViewController. var externalRegistrationParameters: [AnyHashable: Any]? /// The credentials to use after a soft logout has taken place. var softLogoutCredentials: MXCredentials? } /// OnboardingCoordinatorBridgePresenter enables to start OnboardingCoordinator from a view controller. /// This bridge is used while waiting for global usage of coordinator pattern. /// **WARNING**: This class breaks the Coordinator abstraction and it has been introduced for **Objective-C compatibility only** (mainly for integration in legacy view controllers). Each bridge should be removed once the underlying Coordinator has been integrated by another Coordinator. @objcMembers final class OnboardingCoordinatorBridgePresenter: NSObject { // MARK: - Constants private enum NavigationType { case present case push } // MARK: - Properties // MARK: Private private let parameters: OnboardingCoordinatorBridgePresenterParameters private var navigationType: NavigationType = .present private var coordinator: OnboardingCoordinator? // MARK: Public var completion: (() -> Void)? // MARK: Setup init(with parameters: OnboardingCoordinatorBridgePresenterParameters) { self.parameters = parameters super.init() } // MARK: - Public func present(from viewController: UIViewController, animated: Bool) { let onboardingCoordinatorParameters = OnboardingCoordinatorParameters(softLogoutCredentials: parameters.softLogoutCredentials) let onboardingCoordinator = OnboardingCoordinator(parameters: onboardingCoordinatorParameters) onboardingCoordinator.completion = { [weak self] in self?.completion?() } if let externalRegistrationParameters = parameters.externalRegistrationParameters { onboardingCoordinator.update(externalRegistrationParameters: externalRegistrationParameters) } let presentable = onboardingCoordinator.toPresentable() presentable.modalPresentationStyle = .fullScreen presentable.modalTransitionStyle = .crossDissolve viewController.present(presentable, animated: animated, completion: nil) onboardingCoordinator.start() self.coordinator = onboardingCoordinator self.navigationType = .present } func push(from navigationController: UINavigationController, animated: Bool) { let navigationRouter = NavigationRouterStore.shared.navigationRouter(for: navigationController) let onboardingCoordinatorParameters = OnboardingCoordinatorParameters(router: navigationRouter, softLogoutCredentials: parameters.softLogoutCredentials) let onboardingCoordinator = OnboardingCoordinator(parameters: onboardingCoordinatorParameters) onboardingCoordinator.completion = { [weak self] in self?.completion?() } if let externalRegistrationParameters = parameters.externalRegistrationParameters { onboardingCoordinator.update(externalRegistrationParameters: externalRegistrationParameters) } onboardingCoordinator.start() // Will trigger the view controller push self.coordinator = onboardingCoordinator self.navigationType = .push } /// Force a registration process based on a predefined set of parameters from a server provisioning link. /// For more information see `AuthenticationViewController.externalRegistrationParameters`. func update(externalRegistrationParameters: [AnyHashable: Any]) { coordinator?.update(externalRegistrationParameters: externalRegistrationParameters) } /// Set up the authentication screen with the specified homeserver and/or identity server. func updateHomeserver(_ homeserver: String?, andIdentityServer identityServer: String?) { coordinator?.updateHomeserver(homeserver, andIdentityServer: identityServer) } /// When SSO login succeeded, when SFSafariViewController is used, continue login with success parameters. func continueSSOLogin(withToken loginToken: String, transactionID: String) -> Bool { guard let coordinator = coordinator else { return false } return coordinator.continueSSOLogin(withToken: loginToken, transactionID: transactionID) } func dismiss(animated: Bool, completion: (() -> Void)?) { guard let coordinator = self.coordinator else { return } switch navigationType { case .present: // Dismiss modal coordinator.toPresentable().dismiss(animated: animated) { self.coordinator = nil if let completion = completion { completion() } } case .push: // Pop view controller from UINavigationController guard let navigationController = coordinator.toPresentable() as? UINavigationController else { return } navigationController.popViewController(animated: animated) self.coordinator = nil if let completion = completion { completion() } } } }