417 lines
16 KiB
Swift
417 lines
16 KiB
Swift
// File created from ScreenTemplate
|
|
// $ createScreen.sh Verify KeyVerificationVerifyByScanning
|
|
/*
|
|
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
|
|
import MatrixSDK
|
|
|
|
final class KeyVerificationVerifyByScanningViewController: UIViewController {
|
|
|
|
// MARK: - Constants
|
|
|
|
// MARK: - Properties
|
|
|
|
// MARK: Outlets
|
|
|
|
@IBOutlet private weak var closeButton: UIButton!
|
|
|
|
@IBOutlet private weak var titleView: UIView!
|
|
|
|
@IBOutlet private weak var titleLabel: UILabel!
|
|
@IBOutlet private weak var informationLabel: UILabel!
|
|
|
|
@IBOutlet private weak var closeButtonContainer: UIView!
|
|
|
|
@IBOutlet private weak var codeImageView: UIImageView!
|
|
|
|
@IBOutlet private weak var scanCodeButton: UIButton!
|
|
@IBOutlet private weak var cannotScanButton: UIButton!
|
|
|
|
@IBOutlet private weak var qrCodeContainerView: UIView!
|
|
@IBOutlet private weak var qrCodeScannerContainerView: UIView!
|
|
@IBOutlet private weak var qrCodeReaderContainerView: UIView!
|
|
|
|
@IBOutlet private weak var scanButtonContainerView: UIView!
|
|
|
|
// MARK: Private
|
|
|
|
private var viewModel: KeyVerificationVerifyByScanningViewModelType!
|
|
private var theme: Theme!
|
|
private var errorPresenter: MXKErrorPresentation!
|
|
private var activityPresenter: ActivityIndicatorPresenter!
|
|
private var cameraAccessAlertPresenter: CameraAccessAlertPresenter!
|
|
private var cameraAccessManager: CameraAccessManager!
|
|
|
|
private weak var qrCodeReaderViewController: QRCodeReaderViewController?
|
|
private var qrCodeReaderView: QRCodeReaderView?
|
|
|
|
private var alertPresentingViewController: UIViewController {
|
|
return self.qrCodeReaderViewController ?? self
|
|
}
|
|
|
|
// MARK: - Setup
|
|
|
|
class func instantiate(with viewModel: KeyVerificationVerifyByScanningViewModelType) -> KeyVerificationVerifyByScanningViewController {
|
|
let viewController = StoryboardScene.KeyVerificationVerifyByScanningViewController.initialScene.instantiate()
|
|
viewController.viewModel = viewModel
|
|
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.cameraAccessAlertPresenter = CameraAccessAlertPresenter()
|
|
self.cameraAccessManager = CameraAccessManager()
|
|
|
|
self.registerThemeServiceDidChangeThemeNotification()
|
|
self.update(theme: self.theme)
|
|
|
|
self.viewModel.viewDelegate = self
|
|
|
|
self.viewModel.process(viewAction: .loadData)
|
|
}
|
|
|
|
override func viewWillAppear(_ animated: Bool) {
|
|
super.viewWillAppear(animated)
|
|
|
|
// Hide back button
|
|
self.navigationItem.setHidesBackButton(true, animated: animated)
|
|
}
|
|
|
|
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.textPrimaryColor
|
|
|
|
if let themableCloseButton = self.closeButton as? Themable {
|
|
themableCloseButton.update(theme: theme)
|
|
}
|
|
|
|
theme.applyStyle(onButton: self.scanCodeButton)
|
|
theme.applyStyle(onButton: self.cannotScanButton)
|
|
}
|
|
|
|
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() {
|
|
let cancelBarButtonItem = MXKBarButtonItem(title: VectorL10n.cancel, style: .plain) { [weak self] in
|
|
self?.cancelButtonAction()
|
|
}
|
|
|
|
self.navigationItem.rightBarButtonItem = cancelBarButtonItem
|
|
self.closeButtonContainer.isHidden = self.navigationController != nil
|
|
|
|
self.titleLabel.text = VectorL10n.keyVerificationVerifyQrCodeTitle
|
|
self.informationLabel.text = VectorL10n.keyVerificationVerifyQrCodeInformation
|
|
|
|
// Hide until we have the type of the verification request
|
|
self.scanCodeButton.isHidden = true
|
|
self.cannotScanButton.setTitle(VectorL10n.keyVerificationVerifyQrCodeCannotScanAction, for: .normal)
|
|
|
|
removeQRCodeReaderView()
|
|
}
|
|
|
|
private func render(viewState: KeyVerificationVerifyByScanningViewState) {
|
|
switch viewState {
|
|
case .loading:
|
|
self.renderLoading()
|
|
case .loaded(viewData: let viewData) where viewData.qrCodeData == nil && viewData.showScanAction:
|
|
self.renderLoadedWithoutQRCodeData(viewData: viewData)
|
|
case .loaded(viewData: let viewData):
|
|
self.renderLoaded(viewData: viewData)
|
|
case .error(let error):
|
|
self.render(error: error)
|
|
case .scannedCodeValidated(let isValid):
|
|
self.renderScannedCode(valid: isValid)
|
|
case .cancelled(let reason, let verificationKind):
|
|
self.renderCancelled(reason: reason, verificationKind: verificationKind)
|
|
case .cancelledByMe(let reason):
|
|
self.renderCancelledByMe(reason: reason)
|
|
}
|
|
}
|
|
|
|
private func renderLoading() {
|
|
self.activityPresenter.presentActivityIndicator(on: self.view, animated: true)
|
|
}
|
|
|
|
private func addQRCodeReaderView() {
|
|
if self.qrCodeReaderView == nil {
|
|
// configure QRCodeReaderView
|
|
let qrCodeReaderView = QRCodeReaderView()
|
|
qrCodeReaderView.didFoundData = { [weak self] data in
|
|
self?.viewModel.process(viewAction: .scannedCode(payloadData: data))
|
|
}
|
|
self.qrCodeReaderView = qrCodeReaderView
|
|
self.qrCodeReaderContainerView.vc_addSubViewMatchingParent(qrCodeReaderView)
|
|
}
|
|
self.qrCodeScannerContainerView.isHidden = false
|
|
}
|
|
|
|
private func removeQRCodeReaderView() {
|
|
if let qrCodeReaderView {
|
|
qrCodeReaderView.removeFromSuperview()
|
|
self.qrCodeReaderView = nil
|
|
}
|
|
self.qrCodeScannerContainerView.isHidden = true
|
|
}
|
|
|
|
private func renderLoadedWithoutQRCodeData(viewData: KeyVerificationVerifyByScanningViewData) {
|
|
self.activityPresenter.removeCurrentActivityIndicator(animated: true)
|
|
|
|
// We don't have a QR code to display
|
|
self.qrCodeContainerView.isHidden = true
|
|
// We will display a QR code scanner view, so no need to display the scan button
|
|
self.scanButtonContainerView.isHidden = true
|
|
|
|
self.titleLabel.text = VectorL10n.keyVerificationScanQrCodeTitle
|
|
|
|
let informationText: String
|
|
switch viewData.verificationKind {
|
|
case .user:
|
|
informationText = VectorL10n.keyVerificationScanQrCodeInformationOtherUser
|
|
case .newSession:
|
|
informationText = VectorL10n.keyVerificationScanQrCodeInformationNewSession
|
|
case .otherSession:
|
|
informationText = VectorL10n.keyVerificationScanQrCodeInformationOtherSession
|
|
default:
|
|
informationText = VectorL10n.keyVerificationScanQrCodeInformationOtherDevice
|
|
}
|
|
self.informationLabel.text = informationText
|
|
|
|
addQRCodeReaderView()
|
|
}
|
|
|
|
private func renderLoaded(viewData: KeyVerificationVerifyByScanningViewData) {
|
|
self.activityPresenter.removeCurrentActivityIndicator(animated: true)
|
|
|
|
let hideQRCodeImage: Bool
|
|
|
|
if let qrCodePayloadData = viewData.qrCodeData {
|
|
hideQRCodeImage = false
|
|
self.codeImageView.image = self.qrCodeImage(from: qrCodePayloadData)
|
|
} else {
|
|
hideQRCodeImage = true
|
|
}
|
|
|
|
self.titleLabel.text = viewData.verificationKind.verificationTitle
|
|
self.qrCodeContainerView.isHidden = hideQRCodeImage
|
|
self.scanButtonContainerView.isHidden = !viewData.showScanAction
|
|
|
|
if viewData.qrCodeData == nil && viewData.showScanAction == false {
|
|
// Update the copy if QR code scanning is not possible at all
|
|
self.informationLabel.text = VectorL10n.keyVerificationVerifyQrCodeEmojiInformation
|
|
self.cannotScanButton.setTitle(VectorL10n.keyVerificationVerifyQrCodeStartEmojiAction, for: .normal)
|
|
} else {
|
|
let informationText: String
|
|
|
|
switch viewData.verificationKind {
|
|
case .user:
|
|
informationText = VectorL10n.keyVerificationVerifyQrCodeInformation
|
|
self.scanCodeButton.setTitle(VectorL10n.keyVerificationVerifyQrCodeScanCodeAction, for: .normal)
|
|
default:
|
|
informationText = VectorL10n.keyVerificationVerifyQrCodeInformationOtherDevice
|
|
self.scanCodeButton.setTitle(VectorL10n.keyVerificationVerifyQrCodeScanCodeOtherDeviceAction, for: .normal)
|
|
}
|
|
|
|
self.scanCodeButton.isHidden = false
|
|
self.informationLabel.text = informationText
|
|
}
|
|
}
|
|
|
|
private func render(error: Error) {
|
|
self.activityPresenter.removeCurrentActivityIndicator(animated: true)
|
|
self.errorPresenter.presentError(from: self, forError: error, animated: true, handler: nil)
|
|
}
|
|
|
|
private func qrCodeImage(from data: Data) -> UIImage? {
|
|
let codeGenerator = QRCodeGenerator()
|
|
do {
|
|
return try codeGenerator.generateCode(from: data, with: codeImageView.frame.size)
|
|
} catch {
|
|
MXLog.error("[KeyVerificationVerifyByScanningViewController] qrCodeImage: cannot generate QR code", context: error)
|
|
return nil
|
|
}
|
|
}
|
|
|
|
private func presentQRCodeReader(animated: Bool) {
|
|
let qrCodeViewController = QRCodeReaderViewController.instantiate()
|
|
qrCodeViewController.delegate = self
|
|
self.present(qrCodeViewController, animated: animated, completion: nil)
|
|
self.qrCodeReaderViewController = qrCodeViewController
|
|
}
|
|
|
|
private func renderScannedCode(valid: Bool) {
|
|
if valid {
|
|
self.stopQRCodeScanningIfPresented()
|
|
self.presentCodeValidated(animated: true) {
|
|
self.dismissQRCodeScanningIfPresented(animated: true, completion: {
|
|
self.viewModel.process(viewAction: .acknowledgeMyUserScannedOtherCode)
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
private func renderCancelled(reason: MXTransactionCancelCode,
|
|
verificationKind: KeyVerificationKind) {
|
|
self.activityPresenter.removeCurrentActivityIndicator(animated: true)
|
|
|
|
self.stopQRCodeScanningIfPresented()
|
|
|
|
// if we're verifying with someone else, let the user know they cancelled.
|
|
// if we're verifying our own device, assume the user probably knows since it was them who
|
|
// cancelled on their other device
|
|
if verificationKind == .user {
|
|
self.errorPresenter.presentError(from: self.alertPresentingViewController, title: "", message: VectorL10n.deviceVerificationCancelled, animated: true) {
|
|
self.dismissQRCodeScanningIfPresented(animated: false)
|
|
self.viewModel.process(viewAction: .cancel)
|
|
}
|
|
} else {
|
|
self.dismissQRCodeScanningIfPresented(animated: false)
|
|
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: alertPresentingViewController, title: "", message: VectorL10n.deviceVerificationCancelledByMe(reason.humanReadable), animated: true) {
|
|
self.dismissQRCodeScanningIfPresented(animated: false)
|
|
self.viewModel.process(viewAction: .cancel)
|
|
}
|
|
} else {
|
|
self.activityPresenter.removeCurrentActivityIndicator(animated: true)
|
|
}
|
|
}
|
|
|
|
private func presentCodeValidated(animated: Bool, completion: @escaping (() -> Void)) {
|
|
|
|
let alert = UIAlertController(title: VectorL10n.keyVerificationVerifyQrCodeScanOtherCodeSuccessTitle,
|
|
message: VectorL10n.keyVerificationVerifyQrCodeScanOtherCodeSuccessMessage,
|
|
preferredStyle: .alert)
|
|
|
|
let okAction = UIAlertAction(title: VectorL10n.ok, style: .default, handler: { _ in
|
|
completion()
|
|
})
|
|
alert.addAction(okAction)
|
|
|
|
if let qrCodeReaderViewController = self.qrCodeReaderViewController {
|
|
qrCodeReaderViewController.present(alert, animated: animated, completion: nil)
|
|
} else {
|
|
self.present(alert, animated: true)
|
|
}
|
|
}
|
|
|
|
private func checkCameraAccessAndPresentQRCodeReader(animated: Bool) {
|
|
guard self.cameraAccessManager.isCameraAvailable else {
|
|
self.cameraAccessAlertPresenter.presentCameraUnavailableAlert(from: self, animated: animated)
|
|
return
|
|
}
|
|
|
|
self.cameraAccessManager.askAndRequestCameraAccessIfNeeded { (granted) in
|
|
if granted {
|
|
self.presentQRCodeReader(animated: animated)
|
|
} else {
|
|
self.cameraAccessAlertPresenter.presentPermissionDeniedAlert(from: self, animated: animated)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func stopQRCodeScanningIfPresented() {
|
|
guard let qrCodeReaderViewController = self.qrCodeReaderViewController else {
|
|
return
|
|
}
|
|
qrCodeReaderViewController.view.isUserInteractionEnabled = false
|
|
qrCodeReaderViewController.stopScanning()
|
|
}
|
|
|
|
private func dismissQRCodeScanningIfPresented(animated: Bool, completion: (() -> Void)? = nil) {
|
|
guard self.qrCodeReaderViewController?.presentingViewController != nil else {
|
|
completion?()
|
|
return
|
|
}
|
|
self.dismiss(animated: animated, completion: completion)
|
|
}
|
|
|
|
// MARK: - Actions
|
|
|
|
@IBAction private func scanButtonAction(_ sender: Any) {
|
|
self.checkCameraAccessAndPresentQRCodeReader(animated: true)
|
|
}
|
|
|
|
@IBAction private func cannotScanAction(_ sender: Any) {
|
|
qrCodeReaderView?.stopScanning()
|
|
self.viewModel.process(viewAction: .cannotScan)
|
|
}
|
|
|
|
@IBAction private func closeButtonAction(_ sender: Any) {
|
|
self.viewModel.process(viewAction: .cancel)
|
|
}
|
|
|
|
private func cancelButtonAction() {
|
|
self.viewModel.process(viewAction: .cancel)
|
|
}
|
|
}
|
|
|
|
|
|
// MARK: - KeyVerificationVerifyByScanningViewModelViewDelegate
|
|
extension KeyVerificationVerifyByScanningViewController: KeyVerificationVerifyByScanningViewModelViewDelegate {
|
|
|
|
func keyVerificationVerifyByScanningViewModel(_ viewModel: KeyVerificationVerifyByScanningViewModelType, didUpdateViewState viewSate: KeyVerificationVerifyByScanningViewState) {
|
|
self.render(viewState: viewSate)
|
|
}
|
|
}
|
|
|
|
// MARK: - QRCodeReaderViewControllerDelegate
|
|
extension KeyVerificationVerifyByScanningViewController: QRCodeReaderViewControllerDelegate {
|
|
|
|
func qrCodeReaderViewController(_ viewController: QRCodeReaderViewController, didFound payloadData: Data) {
|
|
self.viewModel.process(viewAction: .scannedCode(payloadData: payloadData))
|
|
}
|
|
|
|
func qrCodeReaderViewControllerDidCancel(_ viewController: QRCodeReaderViewController) {
|
|
self.dismissQRCodeScanningIfPresented(animated: true)
|
|
}
|
|
}
|