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
|
//
// SettingsViewController.swift
// MullvadVPN
//
// Created by pronebird on 20/03/2019.
// Copyright © 2019 Mullvad VPN AB. All rights reserved.
//
import Foundation
import MullvadSettings
import Routing
import UIKit
protocol SettingsViewControllerDelegate: AnyObject, Sendable {
func settingsViewControllerDidFinish(_ controller: SettingsViewController)
func settingsViewController(
_ controller: SettingsViewController,
didRequestRoutePresentation route: SettingsNavigationRoute
)
}
class SettingsViewController: UITableViewController {
weak var delegate: SettingsViewControllerDelegate?
private var dataSource: SettingsDataSource?
private let interactor: SettingsInteractor
private let alertPresenter: AlertPresenter
override var preferredStatusBarStyle: UIStatusBarStyle {
.lightContent
}
init(interactor: SettingsInteractor, alertPresenter: AlertPresenter) {
self.interactor = interactor
self.alertPresenter = alertPresenter
super.init(style: .grouped)
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override func viewDidLoad() {
super.viewDidLoad()
navigationItem.title = NSLocalizedString(
"NAVIGATION_TITLE",
tableName: "Settings",
value: "Settings",
comment: ""
)
let doneButton = UIBarButtonItem(
systemItem: .done,
primaryAction: UIAction(handler: { [weak self] _ in
guard let self else { return }
delegate?.settingsViewControllerDidFinish(self)
})
)
doneButton.setAccessibilityIdentifier(.settingsDoneButton)
navigationItem.rightBarButtonItem = doneButton
tableView.setAccessibilityIdentifier(.settingsTableView)
tableView.backgroundColor = .secondaryColor
tableView.separatorColor = .secondaryColor
tableView.rowHeight = UITableView.automaticDimension
tableView.estimatedRowHeight = 60
dataSource = SettingsDataSource(tableView: tableView, interactor: interactor)
dataSource?.delegate = self
interactor.didUpdateTunnelSettings = { [weak self] newSettings in
self?.dataSource?.reload(from: newSettings)
}
}
}
extension SettingsViewController: @preconcurrency SettingsDataSourceDelegate {
func didSelectItem(item: SettingsDataSource.Item) {
guard let route = item.navigationRoute else { return }
delegate?.settingsViewController(self, didRequestRoutePresentation: route)
}
func showInfo(for item: SettingsInfoButtonItem) {
let presentation = AlertPresentation(
id: "settings-info-alert",
icon: .info,
message: item.description,
buttons: [
AlertAction(
title: NSLocalizedString(
"SETTINGS_INFO_ALERT_OK_ACTION",
tableName: "Settings",
value: "Got it!",
comment: ""
),
style: .default
),
]
)
alertPresenter.showAlert(presentation: presentation, animated: true)
}
}
extension SettingsDataSource.Item {
var navigationRoute: SettingsNavigationRoute? {
switch self {
case .vpnSettings:
return .vpnSettings
case .changelog:
return .changelog
case .problemReport:
return .problemReport
case .faq:
return .faq
case .apiAccess:
return .apiAccess
case .daita:
return .daita
case .multihop:
return .multihop
}
}
}
|