Skip to content

Commit 9830155

Browse files
committed
Squashed commit of the following:
commit 1c1067e30104c2670c6339d1f16b7bd55406481b Merge: d3d2df9 70a2319 Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Wed Jun 28 14:07:47 2023 +0300 Merge branch 'local/KB-6393_questionnaire' of https://github.com/joinappex/lockdown-ios-2.0 into local/KB-6393_questionnaire commit d3d2df991e9b6358982ba6d38541f726772f2fd1 Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Wed Jun 28 13:19:42 2023 +0300 add sending message commit 70a23193dcb3deae1341ee318bf536e49ecf3c2a Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Wed Jun 28 13:19:42 2023 +0300 add sending message commit 9a290eb556f0ad4751b49cd53be17de903edd0ac Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Wed Jun 28 13:05:32 2023 +0300 add generating message commit 341293cab9466eab31a1886b6d43d59cc43a3e61 Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Wed Jun 28 12:35:53 2023 +0300 refactoring model of questions commit 0482bce1c1955a0562deaaabeb48aca04f16302c Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Wed Jun 28 12:12:02 2023 +0300 add select region commit 151b94c742fe9cc29e730c18d57a7cf868878ea3 Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Wed Jun 28 11:30:03 2023 +0300 add select country commit f75596a13b9e17dacfc05a57674acb4d65680229 Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Tue Jun 27 14:19:40 2023 +0300 add navigation link view commit a9a83f462ffa3c7ec066d8ee5254b2d8cb6416ed Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Mon Jun 26 18:25:23 2023 +0300 add questions title view commit 08d3eb1d7873bf4ccedd4b8fbf280213520bb9c3 Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Mon Jun 26 16:20:01 2023 +0300 add yes/no views commit b6a16aa2bbd0c1bbe607170d16f430ad522c4d5f Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Mon Jun 26 12:48:23 2023 +0300 fixed skip button commit 5bf7fb8799f4b79e1a37a58a2f3b5862ea9f1da5 Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Mon Jun 26 12:41:59 2023 +0300 refactoring step view model commit 8871dc2ebc0728d3491e6807b4426963741baa4a Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Mon Jun 26 12:04:26 2023 +0300 refactoring step view model commit e5b2e02191217546d47454a471e101c242d89464 Author: Pavel Vilbik <Pavel.Vilbik@joinappex.com> Date: Fri Jun 23 18:49:07 2023 +0300 add view model for content commit b84b9130771aaa521acee0abdf28f3bba2b82771 Author: Pavel Vilbik <p.vilbik@softteco.com> Date: Fri Jun 23 14:44:41 2023 +0300 add textview cell commit e0985ae6c1a5c519e4fd83d5359b68621a18093b Author: Pavel Vilbik <p.vilbik@softteco.com> Date: Fri Jun 23 12:16:51 2023 +0300 add title view commit 24908af4d5b914364df3542b77cecdaaa497d105 Author: Pavel Vilbik <p.vilbik@softteco.com> Date: Thu Jun 22 17:34:07 2023 +0300 add view for problem cell commit 9df1a479a404528efba0452bde0d4291648b49b7 Author: Pavel Vilbik <p.vilbik@softteco.com> Date: Thu Jun 22 17:14:49 2023 +0300 add radio switcher view commit 5ed6c1d3bd4acf00d40f5dbe81cfcbbbd8bf63c0 Author: Pavel Vilbik <p.vilbik@softteco.com> Date: Thu Jun 22 15:49:32 2023 +0300 add base view for questionnaire
1 parent b2d7bdc commit 9830155

30 files changed

+1805
-4
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
{
2+
"info" : {
3+
"author" : "xcode",
4+
"version" : 1
5+
}
6+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
"images" : [
3+
{
4+
"filename" : "Group 16680.pdf",
5+
"idiom" : "universal"
6+
}
7+
],
8+
"info" : {
9+
"author" : "xcode",
10+
"version" : 1
11+
},
12+
"properties" : {
13+
"preserves-vector-representation" : true
14+
}
15+
}
Binary file not shown.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
"images" : [
3+
{
4+
"filename" : "Group 16682.pdf",
5+
"idiom" : "universal"
6+
}
7+
],
8+
"info" : {
9+
"author" : "xcode",
10+
"version" : 1
11+
},
12+
"properties" : {
13+
"preserves-vector-representation" : true
14+
}
15+
}
Binary file not shown.

LockdowniOS.xcodeproj/project.pbxproj

+125
Large diffs are not rendered by default.

LockdowniOS/AccountVC.swift

+16
Original file line numberDiff line numberDiff line change
@@ -438,6 +438,9 @@ final class AccountViewController: BaseViewController, Loadable {
438438
.custom(title: NSLocalizedString("Email Us", comment: ""), completion: {
439439
self.emailTeam()
440440
}),
441+
.custom(title: NSLocalizedString("Fill Questions", comment: "")) { [weak self] in
442+
self?.openQuestionnaire()
443+
},
441444
.cancel()
442445
]
443446
)
@@ -510,6 +513,19 @@ final class AccountViewController: BaseViewController, Loadable {
510513
break
511514
}
512515
}
516+
517+
private func openQuestionnaire() {
518+
let stepsViewController = StepsViewController()
519+
var viewModel = StepsViewModel { [weak self] message in
520+
self?.sendMessage(
521+
message,
522+
subject: "Lockdown Error Reporting Form (iOS \(Bundle.main.versionString))"
523+
)
524+
}
525+
stepsViewController.viewModel = viewModel
526+
stepsViewController.modalPresentationStyle = .fullScreen
527+
present(stepsViewController, animated: true)
528+
}
513529
}
514530

515531
// MARK: - Helpers / Extensions

LockdowniOS/BaseViewController.swift

+5-1
Original file line numberDiff line numberDiff line change
@@ -293,7 +293,6 @@ open class BaseViewController: UIViewController, MFMailComposeViewControllerDele
293293
PacketTunnelProviderLogs.flush()
294294
DDLogInfo("")
295295

296-
let recipient = "team@lockdownprivacy.com"
297296
var appendString = ""
298297
if (getUserWantsVPNEnabled()) {
299298
appendString = appendString + " - S"
@@ -306,6 +305,11 @@ open class BaseViewController: UIViewController, MFMailComposeViewControllerDele
306305
}
307306
message += "\n\n\n"
308307

308+
sendMessage(message, subject: subject)
309+
}
310+
311+
func sendMessage(_ message: String, subject: String) {
312+
let recipient = "team@lockdownprivacy.com"
309313
if MFMailComposeViewController.canSendMail() {
310314
let composeVC = MFMailComposeViewController()
311315
composeVC.mailComposeDelegate = self

LockdowniOS/ConfiguredNavigationView.swift

+5-3
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ import UIKit
1111
final class ConfiguredNavigationView: UIView {
1212

1313
private(set) var buttonCallback: () -> () = { }
14+
var accentColor = UIColor.tunnelsBlue
1415

1516
lazy var titleLabel: UILabel = {
1617
let label = UILabel()
@@ -23,21 +24,22 @@ final class ConfiguredNavigationView: UIView {
2324
lazy var leftNavButton: UIButton = {
2425
let button = UIButton(type: .system)
2526
button.titleLabel?.font = fontBold13
26-
button.tintColor = .tunnelsBlue
27+
button.tintColor = accentColor
2728
button.addTarget(self, action: #selector(buttonDidPress), for: .touchUpInside)
2829
return button
2930
}()
3031

3132
lazy var rightNavButton: UIButton = {
3233
let button = UIButton(type: .system)
3334
button.titleLabel?.font = fontBold13
34-
button.tintColor = .tunnelsBlue
35+
button.tintColor = accentColor
3536
button.addTarget(self, action: #selector(buttonDidPress), for: .touchUpInside)
3637
return button
3738
}()
3839

39-
init() {
40+
init(accentColor: UIColor = .tunnelsBlue) {
4041
super.init(frame: .zero)
42+
self.accentColor = accentColor
4143
configureUI()
4244
}
4345

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,101 @@
1+
//
2+
// SelectCountryViewController.swift
3+
// Lockdown
4+
//
5+
// Created by Pavel Vilbik on 27.06.23.
6+
// Copyright © 2023 Confirmed Inc. All rights reserved.
7+
//
8+
9+
import UIKit
10+
11+
protocol SelectCountryViewModelProtocol {
12+
var countries: [Country] { get }
13+
var selectedCountry: Country? { get set }
14+
func bind(_ view: SelectCountryViewController)
15+
func donePressed()
16+
}
17+
18+
class SelectCountryViewController: UIViewController {
19+
20+
// MARK: - models
21+
22+
private let staticTableView = StaticTableView()
23+
var viewModel: SelectCountryViewModelProtocol?
24+
25+
// MARK: - views
26+
27+
private lazy var navigationView: ConfiguredNavigationView = {
28+
let view = ConfiguredNavigationView(accentColor: .darkText)
29+
view.titleLabel.text = NSLocalizedString("Select country", comment: "")
30+
view.titleLabel.textColor = .black
31+
view.leftNavButton.setTitle(NSLocalizedString("DONE", comment: ""), for: .normal)
32+
view.leftNavButton.addTarget(self, action: #selector(doneClicked), for: .touchUpInside)
33+
view.leftNavButton.tintColor = .tunnelsBlue
34+
view.rightNavButton.setTitle(NSLocalizedString("CANCEL", comment: ""), for: .normal)
35+
view.rightNavButton.addTarget(self, action: #selector(cancelClicked), for: .touchUpInside)
36+
view.rightNavButton.tintColor = .tunnelsBlue
37+
return view
38+
}()
39+
40+
override func viewDidLoad() {
41+
super.viewDidLoad()
42+
43+
configureUI()
44+
updateView()
45+
viewModel?.bind(self)
46+
}
47+
48+
// MARK: - Configure UI
49+
private func configureUI() {
50+
view.backgroundColor = .white
51+
52+
view.addSubview(navigationView)
53+
navigationView.anchors.leading.pin()
54+
navigationView.anchors.trailing.pin()
55+
navigationView.anchors.top.safeAreaPin()
56+
57+
addTableView(staticTableView) { tableView in
58+
staticTableView.anchors.top.spacing(0, to: navigationView.anchors.bottom)
59+
staticTableView.anchors.leading.pin()
60+
staticTableView.anchors.trailing.pin()
61+
staticTableView.anchors.bottom.pin()
62+
}
63+
64+
staticTableView.backgroundColor = .clear
65+
staticTableView.deselectsCellsAutomatically = true
66+
staticTableView.separatorStyle = .none
67+
}
68+
69+
func updateView() {
70+
staticTableView.clear()
71+
viewModel?.countries.forEach { country in
72+
staticTableView.addRowCell { cell in
73+
let view = CountryView()
74+
view.titleLabel.text = country.title
75+
view.emojiLabel.text = country.emojiSymbol
76+
view.checkMark.isHidden = country != viewModel?.selectedCountry
77+
view.didSelect = { [weak self] in
78+
self?.viewModel?.selectedCountry = country
79+
}
80+
cell.backgroundColor = .clear
81+
cell.backgroundView?.backgroundColor = .clear
82+
cell.contentView.backgroundColor = .clear
83+
cell.addSubview(view)
84+
view.anchors.edges.pin(insets: .init(top: 5, left: 2, bottom: 5, right: 2))
85+
}
86+
}
87+
88+
staticTableView.reloadData()
89+
}
90+
91+
92+
// MARK: - actions
93+
94+
@objc private func doneClicked() {
95+
viewModel?.donePressed()
96+
}
97+
98+
@objc func cancelClicked() {
99+
dismiss(animated: true)
100+
}
101+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,120 @@
1+
//
2+
// StepsViewController.swift
3+
// Lockdown
4+
//
5+
// Created by Pavel Vilbik on 21.06.23.
6+
// Copyright © 2023 Confirmed Inc. All rights reserved.
7+
//
8+
9+
import UIKit
10+
11+
class StepsViewController: UIViewController, StepsViewProtocol {
12+
13+
// MARK: - models
14+
var viewModel: StepsViewModel!
15+
16+
// MARK: - views
17+
18+
private lazy var navigationView: ConfiguredNavigationView = {
19+
let view = ConfiguredNavigationView(accentColor: .darkText)
20+
view.leftNavButton.setImage(UIImage(systemName: "chevron.left"), for: .normal)
21+
view.leftNavButton.addTarget(self, action: #selector(backButtonClicked), for: .touchUpInside)
22+
view.rightNavButton.setTitle(NSLocalizedString("Skip", comment: ""), for: .normal)
23+
view.rightNavButton.addTarget(self, action: #selector(skipClicked), for: .touchUpInside)
24+
return view
25+
}()
26+
27+
private lazy var stepsView: StepsView = {
28+
let view = StepsView()
29+
view.steps = viewModel.stepsCount
30+
return view
31+
}()
32+
33+
private lazy var actionButton: UIButton = {
34+
let button = UIButton()
35+
button.anchors.height.equal(56)
36+
button.backgroundColor = .tunnelsBlue
37+
button.layer.cornerRadius = 29
38+
button.titleLabel?.font = .semiboldLockdownFont(size: 17)
39+
button.addTarget(self, action: #selector(actionClicked), for: .touchUpInside)
40+
button.setTitle(viewModel.actionTitle, for: .normal)
41+
return button
42+
}()
43+
44+
private var contentView: UIView?
45+
46+
// MARK: - life cycle
47+
override func viewDidLoad() {
48+
super.viewDidLoad()
49+
50+
configureUI()
51+
viewModel.bind(self)
52+
}
53+
54+
// MARK: - Configure UI
55+
private func configureUI() {
56+
view.backgroundColor = .white
57+
58+
view.addSubview(navigationView)
59+
navigationView.anchors.leading.pin()
60+
navigationView.anchors.trailing.pin()
61+
navigationView.anchors.top.safeAreaPin()
62+
63+
view.addSubview(stepsView)
64+
stepsView.anchors.top.spacing(0, to: navigationView.anchors.bottom)
65+
stepsView.anchors.leading.pin(inset: 18)
66+
stepsView.anchors.trailing.pin(inset: 18)
67+
68+
view.addSubview(actionButton)
69+
actionButton.anchors.leading.pin(inset: 24)
70+
actionButton.anchors.trailing.pin(inset: 24)
71+
actionButton.anchors.bottom.safeAreaPin(inset: 14)
72+
73+
view.addGestureRecognizer(
74+
UITapGestureRecognizer(target: self, action: #selector(tapped))
75+
)
76+
}
77+
78+
func changeContent() {
79+
contentView?.removeFromSuperview()
80+
81+
let staticTableView = viewModel.stepViewModel.contentView()
82+
addTableView(staticTableView) { tableView in
83+
staticTableView.anchors.top.spacing(0, to: stepsView.anchors.bottom)
84+
staticTableView.anchors.leading.pin()
85+
staticTableView.anchors.trailing.pin()
86+
staticTableView.anchors.bottom.spacing(18, to: actionButton.anchors.top)
87+
}
88+
contentView = staticTableView
89+
stepsView.currentStep = viewModel.currentStepIndex
90+
navigationView.rightNavButton.isHidden = !viewModel.showSkipButton
91+
}
92+
93+
func close(completion: (() -> Void)?) {
94+
dismiss(animated: true, completion: completion)
95+
}
96+
97+
func showSelectCountry(with viewModel: SelectCountryViewModelProtocol) {
98+
let viewController = SelectCountryViewController()
99+
viewController.viewModel = viewModel
100+
present(viewController, animated: true)
101+
}
102+
103+
// MARK: - actions
104+
105+
@objc private func backButtonClicked() {
106+
viewModel.backPressed()
107+
}
108+
109+
@objc private func skipClicked() {
110+
viewModel.skipStep()
111+
}
112+
113+
@objc private func actionClicked() {
114+
viewModel.performStepAction()
115+
}
116+
117+
@objc private func tapped() {
118+
view.endEditing(true)
119+
}
120+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
//
2+
// Country.swift
3+
// Lockdown
4+
//
5+
// Created by Pavel Vilbik on 27.06.23.
6+
// Copyright © 2023 Confirmed Inc. All rights reserved.
7+
//
8+
9+
import Foundation
10+
11+
struct Country: Equatable {
12+
let title: String
13+
let emojiSymbol: String?
14+
}

0 commit comments

Comments
 (0)