tokenary/Tokenary iOS/Screens/PasswordViewController.swift

151 lines
4.8 KiB
Swift
Raw Normal View History

2021-12-03 19:08:05 +03:00
// Copyright © 2021 Tokenary. All rights reserved.
2021-12-06 19:16:03 +03:00
import UIKit
2021-12-10 18:42:58 +03:00
import LocalAuthentication
2021-12-06 19:16:03 +03:00
class PasswordViewController: UIViewController {
2021-12-09 15:09:11 +03:00
enum Mode {
case create, repeatAfterCreate, enter
}
private let keychain = Keychain.shared
private var mode = Mode.create
var passwordToRepeat: String?
2021-12-07 20:35:14 +03:00
@IBOutlet weak var passwordTextField: UITextField! {
didSet {
passwordTextField.delegate = self
passwordTextField.addTarget(self, action: #selector(textFieldChanged), for: .editingChanged)
}
}
2021-12-10 17:56:57 +03:00
@IBOutlet weak var initialOverlayView: UIView!
2021-12-07 20:35:14 +03:00
@IBOutlet weak var okButton: UIButton!
override func viewDidLoad() {
super.viewDidLoad()
navigationController?.navigationBar.prefersLargeTitles = true
navigationItem.largeTitleDisplayMode = .always
2021-12-09 15:09:11 +03:00
navigationItem.backButtonDisplayMode = .minimal
if passwordToRepeat != nil {
switchToMode(.repeatAfterCreate)
} else if keychain.password != nil {
switchToMode(.enter)
} else {
switchToMode(.create)
}
2021-12-10 18:42:58 +03:00
if mode == .enter {
navigationController?.setNavigationBarHidden(true, animated: false)
askForLocalAuthentication()
}
}
override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
if mode != .enter {
passwordTextField.becomeFirstResponder()
}
2021-12-09 15:09:11 +03:00
}
2021-12-10 18:42:58 +03:00
private func switchToMode(_ mode: Mode) {
2021-12-09 15:09:11 +03:00
self.mode = mode
switch mode {
case .create:
navigationItem.title = Strings.createPassword
case .repeatAfterCreate:
navigationItem.title = Strings.repeatPassword
case .enter:
navigationItem.title = Strings.enterPassword
}
}
2021-12-10 18:42:58 +03:00
private func askForLocalAuthentication() {
let context = LAContext()
var error: NSError?
let policy = LAPolicy.deviceOwnerAuthenticationWithBiometrics
let canDoLocalAuthentication = context.canEvaluatePolicy(policy, error: &error)
if canDoLocalAuthentication {
context.localizedCancelTitle = Strings.cancel
context.evaluatePolicy(.deviceOwnerAuthenticationWithBiometrics, localizedReason: Strings.enterTokenary) { [weak self] success, _ in
DispatchQueue.main.async {
if success {
self?.showAccountsList()
} else {
self?.didFailLocalAuthentication()
}
}
}
} else {
didFailLocalAuthentication()
}
}
private func didFailLocalAuthentication() {
navigationController?.setNavigationBarHidden(false, animated: false)
initialOverlayView.isHidden = true
passwordTextField.becomeFirstResponder()
}
2021-12-07 20:35:14 +03:00
@IBAction func okButtonTapped(_ sender: Any) {
proceedIfPossible()
}
@objc private func textFieldChanged() {
let isEnabled = passwordTextField.text?.isOkAsPassword == true
if okButton.isEnabled != isEnabled {
okButton.isEnabled = isEnabled
}
}
private func proceedIfPossible() {
2021-12-09 15:09:11 +03:00
switch mode {
case .create:
let passwordViewController = instantiate(PasswordViewController.self, from: .main)
passwordViewController.passwordToRepeat = passwordTextField.text
navigationController?.pushViewController(passwordViewController, animated: true)
case .repeatAfterCreate:
if let password = passwordTextField.text, !password.isEmpty, password == passwordToRepeat {
keychain.save(password: password)
showAccountsList()
} else {
showMessageAlert(text: Strings.passwordDoesNotMatch)
}
case .enter:
if passwordTextField.text == keychain.password {
showAccountsList()
} else {
showMessageAlert(text: Strings.passwordDoesNotMatch)
}
}
2021-12-07 20:35:14 +03:00
}
private func showAccountsList() {
let accountsList = instantiate(AccountsListViewController.self, from: .main)
UIApplication.shared.replaceRootViewController(with: accountsList.inNavigationController)
}
}
extension PasswordViewController: UITextFieldDelegate {
func textFieldDidBeginEditing(_ textField: UITextField) {
textFieldChanged()
}
func textFieldDidEndEditing(_ textField: UITextField, reason: UITextField.DidEndEditingReason) {
textFieldChanged()
}
func textFieldShouldReturn(_ textField: UITextField) -> Bool {
2021-12-09 15:09:11 +03:00
if passwordTextField.text?.isOkAsPassword == true {
proceedIfPossible()
}
2021-12-07 20:35:14 +03:00
return true
}
2021-12-06 19:16:03 +03:00
}