// File created from ScreenTemplate // $ createScreen.sh KeyVerification KeyVerificationSelfVerifyWait /* Copyright 2020 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 UIKit final class KeyVerificationSelfVerifyWaitViewController: UIViewController { // MARK: - Constants private enum Constants { static let clientNamesLineSpacing: CGFloat = 3.0 } // MARK: - Properties // MARK: Outlets @IBOutlet private weak var titleLabel: UILabel! @IBOutlet private weak var informationLabel: UILabel! @IBOutlet private weak var desktopClientImageView: UIImageView! @IBOutlet private weak var mobileClientImageView: UIImageView! @IBOutlet private weak var recoverSecretsAvailabilityLoadingContainerView: UIView! @IBOutlet private weak var recoverSecretsAvailabilityLoadingLabel: UILabel! @IBOutlet private weak var recoverSecretsAvailabilityActivityIndicatorView: UIActivityIndicatorView! @IBOutlet private weak var recoverSecretsContainerView: UIView! @IBOutlet private weak var recoverSecretsButton: RoundedButton! // bwi #5271 don't do crosssigning request at view start but as action on button click @IBOutlet private weak var crosssigningButton: RoundedButton! @IBOutlet private weak var recoverSecretsAdditionalInformationLabel: UILabel! // MARK: Private private var viewModel: KeyVerificationSelfVerifyWaitViewModelType! private var cancellable: Bool! private var theme: Theme! private var errorPresenter: MXKErrorPresentation! private var activityPresenter: ActivityIndicatorPresenter! private weak var cancelBarButtonItem: UIBarButtonItem? // MARK: - Setup class func instantiate(with viewModel: KeyVerificationSelfVerifyWaitViewModelType, cancellable: Bool) -> KeyVerificationSelfVerifyWaitViewController { let viewController = StoryboardScene.KeyVerificationSelfVerifyWaitViewController.initialScene.instantiate() viewController.viewModel = viewModel viewController.cancellable = cancellable viewController.theme = ThemeService.shared().theme return viewController } // MARK: - Life cycle override func viewDidLoad() { super.viewDidLoad() // Do any additional setup after loading the view. self.setupViews() self.activityPresenter = ActivityIndicatorPresenter() self.errorPresenter = MXKErrorAlertPresentation() self.registerThemeServiceDidChangeThemeNotification() self.update(theme: self.theme) } override func viewDidAppear(_ animated: Bool) { super.viewDidAppear(animated) self.viewModel.viewDelegate = self } override var preferredStatusBarStyle: UIStatusBarStyle { return self.theme.statusBarStyle } // MARK: - Private private func update(theme: Theme) { self.theme = theme self.view.backgroundColor = theme.headerBackgroundColor if let navigationBar = self.navigationController?.navigationBar { theme.applyStyle(onNavigationBar: navigationBar) } self.titleLabel.textColor = theme.textPrimaryColor self.informationLabel.textColor = theme.textSecondaryColor self.desktopClientImageView.tintColor = theme.tintColor self.mobileClientImageView.tintColor = theme.tintColor self.recoverSecretsAvailabilityLoadingLabel.textColor = theme.textSecondaryColor self.recoverSecretsAvailabilityActivityIndicatorView.color = theme.tintColor // BWI: #4966 self.recoverSecretsButton.update(theme: theme) self.crosssigningButton.update(theme: theme) } private func registerThemeServiceDidChangeThemeNotification() { NotificationCenter.default.addObserver(self, selector: #selector(themeDidChange), name: .themeServiceDidChangeTheme, object: nil) } @objc private func themeDidChange() { self.update(theme: ThemeService.shared().theme) } private func setupViews() { if self.cancellable { let cancelBarButtonItem = MXKBarButtonItem(title: VectorL10n.skip, style: .plain) { [weak self] in self?.cancelButtonAction() } self.vc_removeBackTitle() self.navigationItem.rightBarButtonItem = cancelBarButtonItem self.cancelBarButtonItem = cancelBarButtonItem } self.title = BWIL10n.deviceVerificationSelfVerifyWaitTitle self.titleLabel.text = BWIL10n.deviceVerificationSelfVerifyWaitTitle self.informationLabel.text = BWIL10n.deviceVerificationSelfVerifyWaitInformation self.desktopClientImageView.image = Asset.Images.monitor.image.withRenderingMode(.alwaysTemplate) self.mobileClientImageView.image = Asset.Images.smartphone.image.withRenderingMode(.alwaysTemplate) self.recoverSecretsAdditionalInformationLabel.text = BWIL10n.deviceVerificationSelfVerifyWaitRecoverSecretsAdditionalInformation self.recoverSecretsButton.setTitle(BWIL10n.deviceVerificationSelfVerifyWaitRecoverSecretsWithPassphrase, for: .normal) self.crosssigningButton.setTitle(BWIL10n.deviceVerificationCrosssigningWaitRecoverSecrets, for: .normal) self.recoverSecretsAvailabilityLoadingContainerView.isHidden = true } private func render(viewState: KeyVerificationSelfVerifyWaitViewState) { switch viewState { case .loading: self.renderLoading() case .secretsRecoveryCheckingAvailability(let text): self.renderSecretsRecoveryCheckingAvailability(withText: text) case .loaded(let viewData): self.renderLoaded(viewData: viewData) case .cancelled(let reason): self.renderCancelled(reason: reason) case .cancelledByMe(let reason): self.renderCancelledByMe(reason: reason) case .error(let error): self.render(error: error) } } private func renderLoading() { self.activityPresenter.presentActivityIndicator(on: self.view, animated: true) } private func renderSecretsRecoveryCheckingAvailability(withText text: String?) { self.recoverSecretsAvailabilityLoadingLabel.text = text self.recoverSecretsAvailabilityActivityIndicatorView.startAnimating() self.recoverSecretsAvailabilityLoadingContainerView.isHidden = false self.recoverSecretsContainerView.isHidden = true } private func renderLoaded(viewData: KeyVerificationSelfVerifyWaitViewData) { self.activityPresenter.removeCurrentActivityIndicator(animated: true) self.title = viewData.isNewSignIn ? VectorL10n.deviceVerificationSelfVerifyWaitNewSignInTitle : BWIL10n.deviceVerificationSelfVerifyWaitTitle self.cancelBarButtonItem?.title = viewData.isNewSignIn ? VectorL10n.skip : VectorL10n.cancel let hideRecoverSecrets: Bool let recoverSecretsButtonTitle: String? switch viewData.secretsRecoveryAvailability { case .notAvailable: hideRecoverSecrets = true recoverSecretsButtonTitle = nil case .available(let secretsRecoveryMode): hideRecoverSecrets = false switch secretsRecoveryMode { case .passphraseOrKey: recoverSecretsButtonTitle = BWIL10n.deviceVerificationSelfVerifyWaitRecoverSecretsWithPassphrase case .onlyKey: recoverSecretsButtonTitle = BWIL10n.deviceVerificationSelfVerifyWaitRecoverSecretsWithoutPassphrase } } self.recoverSecretsAvailabilityLoadingContainerView.isHidden = true self.recoverSecretsAvailabilityActivityIndicatorView.stopAnimating() self.recoverSecretsContainerView.isHidden = hideRecoverSecrets self.recoverSecretsButton.setTitle(recoverSecretsButtonTitle, for: .normal) } private func renderCancelled(reason: MXTransactionCancelCode) { self.activityPresenter.removeCurrentActivityIndicator(animated: true) self.errorPresenter.presentError(from: self, title: "", message: BWIL10n.deviceVerificationCancelled, animated: true) { self.viewModel.process(viewAction: .cancel) } } private func renderCancelledByMe(reason: MXTransactionCancelCode) { if reason.value != MXTransactionCancelCode.user().value { self.activityPresenter.removeCurrentActivityIndicator(animated: true) self.errorPresenter.presentError(from: self, title: "", message: VectorL10n.deviceVerificationCancelledByMe(reason.humanReadable), animated: true) { self.viewModel.process(viewAction: .cancel) } } else { self.activityPresenter.removeCurrentActivityIndicator(animated: true) } } private func render(error: Error) { self.activityPresenter.removeCurrentActivityIndicator(animated: true) // bwi: don't show unneccessary error alert when user only has one device -> he is able to use passphrase if BWIBuildSettings.shared.showNoOtherDeviceError { self.errorPresenter.presentError(from: self, forError: error, animated: true, handler: nil) } } // MARK: - Actions private func cancelButtonAction() { self.viewModel.process(viewAction: .cancel) } @IBAction private func recoverSecretsButtonAction(_ sender: Any) { self.viewModel.process(viewAction: .recoverSecrets) } @IBAction private func crossSigningButtonAction(_ sender: Any) { self.viewModel.process(viewAction: .loadData) } } // MARK: - KeyVerificationSelfVerifyWaitViewModelViewDelegate extension KeyVerificationSelfVerifyWaitViewController: KeyVerificationSelfVerifyWaitViewModelViewDelegate { func keyVerificationSelfVerifyWaitViewModel(_ viewModel: KeyVerificationSelfVerifyWaitViewModelType, didUpdateViewState viewSate: KeyVerificationSelfVerifyWaitViewState) { self.render(viewState: viewSate) } }