Skip to content

Commit c282ef3

Browse files
author
Jon Petersson
committed
Split select location view into two sections
1 parent ebfb3f6 commit c282ef3

20 files changed

+597
-473
lines changed

ios/.swiftlint.yml

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ disabled_rules:
77
- type_body_length
88
- opening_brace # Differs from Google swift guidelines enforced by swiftformat
99
- trailing_comma
10+
- switch_case_alignment # Enables expressions such as [return switch location {}]
1011
opt_in_rules:
1112
- empty_count
1213

ios/MullvadVPN.xcodeproj/project.pbxproj

+21-25
Large diffs are not rendered by default.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
{
2+
"pins" : [
3+
{
4+
"identity" : "swift-log",
5+
"kind" : "remoteSourceControl",
6+
"location" : "https://github.com/apple/swift-log.git",
7+
"state" : {
8+
"revision" : "173f567a2dfec11d74588eea82cecea555bdc0bc",
9+
"version" : "1.4.0"
10+
}
11+
},
12+
{
13+
"identity" : "wireguard-apple",
14+
"kind" : "remoteSourceControl",
15+
"location" : "https://github.com/mullvad/wireguard-apple.git",
16+
"state" : {
17+
"revision" : "11a00c20dc03f2751db47e94f585c0778c7bde82"
18+
}
19+
}
20+
],
21+
"version" : 2
22+
}

ios/MullvadVPN/Coordinators/ApplicationCoordinator.swift

+3-3
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,7 @@ final class ApplicationCoordinator: Coordinator, Presenting, RootContainerViewCo
6565
}()
6666

6767
private var splitTunnelCoordinator: TunnelCoordinator?
68-
private var splitLocationCoordinator: SelectLocationCoordinator?
68+
private var splitLocationCoordinator: LocationCoordinator?
6969

7070
private let tunnelManager: TunnelManager
7171
private let storePaymentManager: StorePaymentManager
@@ -703,11 +703,11 @@ final class ApplicationCoordinator: Coordinator, Presenting, RootContainerViewCo
703703
}
704704

705705
private func makeSelectLocationCoordinator(forModalPresentation isModalPresentation: Bool)
706-
-> SelectLocationCoordinator {
706+
-> LocationCoordinator {
707707
let navigationController = CustomNavigationController()
708708
navigationController.isNavigationBarHidden = !isModalPresentation
709709

710-
let selectLocationCoordinator = SelectLocationCoordinator(
710+
let selectLocationCoordinator = LocationCoordinator(
711711
navigationController: navigationController,
712712
tunnelManager: tunnelManager,
713713
relayCacheTracker: relayCacheTracker
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,95 @@
1+
//
2+
// CustomListsDataSource.swift
3+
// MullvadVPN
4+
//
5+
// Created by Jon Petersson on 2024-02-22.
6+
// Copyright © 2024 Mullvad VPN AB. All rights reserved.
7+
//
8+
9+
import Foundation
10+
import MullvadREST
11+
import MullvadSettings
12+
import MullvadTypes
13+
14+
class CustomListsDataSource: LocationDataSourceProtocol {
15+
private(set) var nodes = [LocationNode]()
16+
private(set) var repository: CustomListRepositoryProtocol
17+
18+
init(repository: CustomListRepositoryProtocol) {
19+
self.repository = repository
20+
}
21+
22+
var searchableNodes: [LocationNode] {
23+
nodes.flatMap { $0.children }
24+
}
25+
26+
func reload(allLocationNodes: [LocationNode]) {
27+
nodes = repository.fetchAll().map { list in
28+
let listNode = LocationListNode(
29+
nodeName: list.name,
30+
nodeCode: list.name.lowercased(),
31+
locations: list.locations,
32+
customList: list
33+
)
34+
35+
listNode.children = list.locations.compactMap { location in
36+
copy(location, from: allLocationNodes, withParent: listNode)
37+
}
38+
39+
listNode.forEachDescendant { _, node in
40+
node.nodeCode = "\(listNode.nodeCode)-\(node.nodeCode)"
41+
}
42+
43+
return listNode
44+
}
45+
}
46+
47+
func node(by locations: [RelayLocation], for customList: CustomList) -> LocationNode? {
48+
guard let customListNode = nodes.first(where: { $0.nodeName == customList.name })
49+
else { return nil }
50+
51+
if locations.count > 1 {
52+
return customListNode
53+
} else {
54+
return switch locations.first {
55+
case let .country(countryCode):
56+
customListNode.nodeFor(nodeCode: "\(customListNode.nodeCode)-\(countryCode)")
57+
case let .city(_, cityCode):
58+
customListNode.nodeFor(nodeCode: "\(customListNode.nodeCode)-\(cityCode)")
59+
case let .hostname(_, _, hostCode):
60+
customListNode.nodeFor(nodeCode: "\(customListNode.nodeCode)-\(hostCode)")
61+
case .none:
62+
nil
63+
}
64+
}
65+
}
66+
67+
func customList(by id: UUID) -> CustomList? {
68+
repository.fetch(by: id)
69+
}
70+
71+
private func copy(
72+
_ location: RelayLocation,
73+
from allLocationNodes: [LocationNode],
74+
withParent rootNode: LocationNode
75+
) -> LocationNode? {
76+
let rootNode = RootNode(children: allLocationNodes)
77+
78+
return switch location {
79+
case let .country(countryCode):
80+
rootNode
81+
.countryFor(countryCode: countryCode)?.copy(withParent: rootNode)
82+
83+
case let .city(countryCode, cityCode):
84+
rootNode
85+
.countryFor(countryCode: countryCode)?.copy(withParent: rootNode)
86+
.cityFor(cityCode: cityCode)
87+
88+
case let .hostname(countryCode, cityCode, hostCode):
89+
rootNode
90+
.countryFor(countryCode: countryCode)?.copy(withParent: rootNode)
91+
.cityFor(cityCode: cityCode)?
92+
.hostFor(hostCode: hostCode)
93+
}
94+
}
95+
}

ios/MullvadVPN/Coordinators/SelectLocationCoordinator.swift ios/MullvadVPN/Coordinators/LocationCoordinator.swift

+13-14
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
//
2-
// SelectLocationCoordinator.swift
2+
// LocationCoordinator.swift
33
// MullvadVPN
44
//
55
// Created by pronebird on 29/01/2023.
@@ -13,7 +13,7 @@ import UIKit
1313

1414
import MullvadSettings
1515

16-
class SelectLocationCoordinator: Coordinator, Presentable, Presenting, RelayCacheTrackerObserver {
16+
class LocationCoordinator: Coordinator, Presentable, Presenting, RelayCacheTrackerObserver {
1717
private let tunnelManager: TunnelManager
1818
private let relayCacheTracker: RelayCacheTracker
1919
private var cachedRelays: CachedRelays?
@@ -24,10 +24,10 @@ class SelectLocationCoordinator: Coordinator, Presentable, Presenting, RelayCach
2424
navigationController
2525
}
2626

27-
var selectLocationViewController: SelectLocationViewController? {
27+
var selectLocationViewController: LocationViewController? {
2828
return navigationController.viewControllers.first {
29-
$0 is SelectLocationViewController
30-
} as? SelectLocationViewController
29+
$0 is LocationViewController
30+
} as? LocationViewController
3131
}
3232

3333
var relayFilter: RelayFilter {
@@ -39,7 +39,7 @@ class SelectLocationCoordinator: Coordinator, Presentable, Presenting, RelayCach
3939
}
4040
}
4141

42-
var didFinish: ((SelectLocationCoordinator, RelayLocation?) -> Void)?
42+
var didFinish: ((LocationCoordinator, [RelayLocation]) -> Void)?
4343

4444
init(
4545
navigationController: UINavigationController,
@@ -52,22 +52,22 @@ class SelectLocationCoordinator: Coordinator, Presentable, Presenting, RelayCach
5252
}
5353

5454
func start() {
55-
let selectLocationViewController = SelectLocationViewController()
55+
let selectLocationViewController = LocationViewController()
5656

57-
selectLocationViewController.didSelectRelay = { [weak self] relay in
57+
selectLocationViewController.didSelectRelays = { [weak self] locations, customListId in
5858
guard let self else { return }
5959

6060
var relayConstraints = tunnelManager.settings.relayConstraints
6161
relayConstraints.locations = .only(RelayLocations(
62-
locations: [relay],
63-
customListId: nil
62+
locations: locations,
63+
customListId: customListId
6464
))
6565

6666
tunnelManager.updateSettings([.relayConstraints(relayConstraints)]) {
6767
self.tunnelManager.startTunnel()
6868
}
6969

70-
didFinish?(self, relay)
70+
didFinish?(self, locations)
7171
}
7272

7373
selectLocationViewController.navigateToFilter = { [weak self] in
@@ -91,7 +91,7 @@ class SelectLocationCoordinator: Coordinator, Presentable, Presenting, RelayCach
9191
selectLocationViewController.didFinish = { [weak self] in
9292
guard let self else { return }
9393

94-
didFinish?(self, nil)
94+
didFinish?(self, [])
9595
}
9696

9797
relayCacheTracker.addObserver(self)
@@ -101,8 +101,7 @@ class SelectLocationCoordinator: Coordinator, Presentable, Presenting, RelayCach
101101
selectLocationViewController.setCachedRelays(cachedRelays, filter: relayFilter)
102102
}
103103

104-
selectLocationViewController.relayLocation =
105-
tunnelManager.settings.relayConstraints.locations.value?.locations.first
104+
selectLocationViewController.relayLocations = tunnelManager.settings.relayConstraints.locations.value
106105

107106
navigationController.pushViewController(selectLocationViewController, animated: false)
108107
}

ios/MullvadVPN/UI appearance/UIColor+Palette.swift

+4
Original file line numberDiff line numberDiff line change
@@ -117,6 +117,10 @@ extension UIColor {
117117
static let backgroundColor = UIColor(red: 0.13, green: 0.20, blue: 0.30, alpha: 1.0)
118118
}
119119

120+
enum SubSubSubCell {
121+
static let backgroundColor = UIColor(red: 0.11, green: 0.17, blue: 0.27, alpha: 1.0)
122+
}
123+
120124
enum HeaderBar {
121125
static let defaultBackgroundColor = primaryColor
122126
static let unsecuredBackgroundColor = dangerColor

ios/MullvadVPN/UI appearance/UIMetrics.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -173,7 +173,7 @@ extension UIMetrics {
173173
static let contentInsets = UIEdgeInsets(top: 24, left: 24, bottom: 24, right: 24)
174174

175175
/// Common layout margins for location cell presentation
176-
static let selectLocationCellLayoutMargins = NSDirectionalEdgeInsets(top: 16, leading: 28, bottom: 16, trailing: 12)
176+
static let locationCellLayoutMargins = NSDirectionalEdgeInsets(top: 16, leading: 28, bottom: 16, trailing: 12)
177177

178178
/// Layout margins used by content heading displayed below the large navigation title.
179179
static let contentHeadingLayoutMargins = NSDirectionalEdgeInsets(top: 8, leading: 24, bottom: 24, trailing: 24)

0 commit comments

Comments
 (0)