summaryrefslogtreecommitdiffhomepage
path: root/ios/MullvadVPN/View controllers/RedeemVoucher/RedeemVoucherViewController.swift
blob: c292ec0a2a7c173b433304cd88d053f6530b90e0 (plain)
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
149
150
//
//  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
        }
    }
}