-
Notifications
You must be signed in to change notification settings - Fork 384
/
Copy pathRedeemVoucherViewController.swift
148 lines (118 loc) · 4.06 KB
/
RedeemVoucherViewController.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
//
// RedeemVoucherViewController.swift
// MullvadVPN
//
// Created by Andreas Lif on 2022-08-05.
// Copyright © 2025 Mullvad VPN AB. All rights reserved.
//
import MullvadREST
import MullvadTypes
import UIKit
protocol RedeemVoucherViewControllerDelegate: AnyObject, Sendable {
func redeemVoucherDidSucceed(
_ controller: RedeemVoucherViewController,
with response: REST.SubmitVoucherResponse
)
func redeemVoucherDidCancel(_ controller: RedeemVoucherViewController)
}
@MainActor
class RedeemVoucherViewController: UIViewController, UINavigationControllerDelegate, RootContainment {
private let contentView: RedeemVoucherContentView
nonisolated(unsafe) private var interactor: RedeemVoucherInteractor
weak var delegate: RedeemVoucherViewControllerDelegate?
init(
configuration: RedeemVoucherViewConfiguration,
interactor: RedeemVoucherInteractor
) {
self.contentView = RedeemVoucherContentView(configuration: configuration)
self.interactor = interactor
self.contentView.isUserInteractionEnabled = false
super.init(nibName: nil, bundle: nil)
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override var preferredStatusBarStyle: UIStatusBarStyle {
.lightContent
}
var preferredHeaderBarPresentation: HeaderBarPresentation {
HeaderBarPresentation(style: .default, showsDivider: true)
}
var prefersHeaderBarHidden: Bool {
false
}
var prefersDeviceInfoBarHidden: Bool {
true
}
// MARK: - Life Cycle
override func viewDidLoad() {
super.viewDidLoad()
configureUI()
addActions()
}
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
contentView.isUserInteractionEnabled = true
contentView.isEditing = true
}
override func viewWillDisappear(_ animated: Bool) {
super.viewWillDisappear(animated)
contentView.isEditing = false
}
override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) {
contentView.isEditing = false
super.viewWillTransition(to: size, with: coordinator)
}
// MARK: - private functions
private func addActions() {
contentView.redeemAction = { [weak self] code in
self?.submit(code: code)
}
contentView.cancelAction = { [weak self] in
self?.cancel()
}
contentView.logoutAction = { [weak self] in
self?.logout()
}
interactor.showLogoutDialog = { [weak self] in
self?.contentView.isLogoutDialogHidden = false
}
}
private func configureUI() {
view.addConstrainedSubviews([contentView]) {
contentView.pinEdgesToSuperview(.all())
}
}
private func submit(code: String) {
contentView.state = .verifying
contentView.isEditing = false
interactor.redeemVoucher(code: code, completion: { [weak self] result in
guard let self else { return }
/// Safe to assume `@MainActor` isolation because
/// `TunnelManager.redeemVoucher` sets the `RedeemVoucherOperation`'s `completionQueue` to `.main`
MainActor.assumeIsolated {
switch result {
case let .success(value):
contentView.state = .success
delegate?.redeemVoucherDidSucceed(self, with: value)
case let .failure(error):
contentView.state = .failure(error)
}
}
})
}
private func cancel() {
contentView.isEditing = false
interactor.cancelAll()
delegate?.redeemVoucherDidCancel(self)
}
private func logout() {
contentView.isEditing = false
contentView.state = .logout
Task { [weak self] in
guard let self else { return }
await interactor.logout()
contentView.state = .initial
}
}
}