2018-10-24 01:37:28 +00:00
|
|
|
// SPDX-License-Identifier: MIT
|
2018-10-30 02:57:35 +00:00
|
|
|
// Copyright © 2018 WireGuard LLC. All Rights Reserved.
|
2018-10-20 13:45:53 +00:00
|
|
|
|
|
|
|
import UIKit
|
|
|
|
|
2018-10-24 11:39:34 +00:00
|
|
|
protocol TunnelEditTableViewControllerDelegate: class {
|
2018-10-25 05:44:38 +00:00
|
|
|
func tunnelSaved(tunnel: TunnelContainer)
|
|
|
|
func tunnelEditingCancelled()
|
2018-10-24 11:39:34 +00:00
|
|
|
}
|
|
|
|
|
2018-10-20 13:45:53 +00:00
|
|
|
// MARK: TunnelEditTableViewController
|
|
|
|
|
|
|
|
class TunnelEditTableViewController: UITableViewController {
|
|
|
|
|
2018-10-24 11:39:34 +00:00
|
|
|
weak var delegate: TunnelEditTableViewControllerDelegate? = nil
|
|
|
|
|
2018-10-23 12:32:10 +00:00
|
|
|
let interfaceFieldsBySection: [[TunnelViewModel.InterfaceField]] = [
|
2018-10-20 13:45:53 +00:00
|
|
|
[.name],
|
|
|
|
[.privateKey, .publicKey, .generateKeyPair],
|
|
|
|
[.addresses, .listenPort, .mtu, .dns]
|
|
|
|
]
|
|
|
|
|
2018-10-29 07:16:54 +00:00
|
|
|
let peerFields: [TunnelViewModel.PeerField] = [
|
|
|
|
.publicKey, .preSharedKey, .endpoint,
|
2018-10-29 11:08:32 +00:00
|
|
|
.allowedIPs, .excludePrivateIPs, .persistentKeepAlive,
|
2018-10-29 07:16:54 +00:00
|
|
|
.deletePeer
|
2018-10-20 13:45:53 +00:00
|
|
|
]
|
|
|
|
|
2018-10-23 11:53:46 +00:00
|
|
|
let tunnelsManager: TunnelsManager
|
2018-10-24 11:39:34 +00:00
|
|
|
let tunnel: TunnelContainer?
|
2018-10-23 09:53:18 +00:00
|
|
|
let tunnelViewModel: TunnelViewModel
|
2018-10-20 13:45:53 +00:00
|
|
|
|
2018-10-25 02:30:12 +00:00
|
|
|
init(tunnelsManager tm: TunnelsManager, tunnel t: TunnelContainer) {
|
|
|
|
// Use this initializer to edit an existing tunnel.
|
2018-10-23 11:53:46 +00:00
|
|
|
tunnelsManager = tm
|
2018-10-24 11:39:34 +00:00
|
|
|
tunnel = t
|
2018-10-25 10:20:27 +00:00
|
|
|
tunnelViewModel = TunnelViewModel(tunnelConfiguration: t.tunnelConfiguration())
|
2018-10-25 02:30:12 +00:00
|
|
|
super.init(style: .grouped)
|
|
|
|
}
|
|
|
|
|
|
|
|
init(tunnelsManager tm: TunnelsManager, tunnelConfiguration: TunnelConfiguration?) {
|
|
|
|
// Use this initializer to create a new tunnel.
|
|
|
|
// If tunnelConfiguration is passed, data will be prepopulated from that configuration.
|
|
|
|
tunnelsManager = tm
|
|
|
|
tunnel = nil
|
|
|
|
tunnelViewModel = TunnelViewModel(tunnelConfiguration: tunnelConfiguration)
|
2018-10-20 13:45:53 +00:00
|
|
|
super.init(style: .grouped)
|
|
|
|
}
|
|
|
|
|
|
|
|
required init?(coder aDecoder: NSCoder) {
|
|
|
|
fatalError("init(coder:) has not been implemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
override func viewDidLoad() {
|
|
|
|
super.viewDidLoad()
|
2018-10-28 20:54:32 +00:00
|
|
|
self.title = (tunnel == nil) ? "New configuration" : "Edit configuration"
|
2018-10-20 13:45:53 +00:00
|
|
|
self.navigationItem.rightBarButtonItem = UIBarButtonItem(barButtonSystemItem: .save, target: self, action: #selector(saveTapped))
|
|
|
|
self.navigationItem.leftBarButtonItem = UIBarButtonItem(barButtonSystemItem: .cancel, target: self, action: #selector(cancelTapped))
|
|
|
|
|
|
|
|
self.tableView.rowHeight = 44
|
|
|
|
self.tableView.allowsSelection = false
|
|
|
|
|
2018-10-28 20:45:43 +00:00
|
|
|
self.tableView.register(TunnelEditTableViewKeyValueCell.self, forCellReuseIdentifier: TunnelEditTableViewKeyValueCell.id)
|
|
|
|
self.tableView.register(TunnelEditTableViewButtonCell.self, forCellReuseIdentifier: TunnelEditTableViewButtonCell.id)
|
|
|
|
self.tableView.register(TunnelEditTableViewSwitchCell.self, forCellReuseIdentifier: TunnelEditTableViewSwitchCell.id)
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@objc func saveTapped() {
|
2018-10-23 11:53:46 +00:00
|
|
|
self.tableView.endEditing(false)
|
|
|
|
let tunnelSaveResult = tunnelViewModel.save()
|
|
|
|
switch (tunnelSaveResult) {
|
|
|
|
case .error(let errorMessage):
|
|
|
|
let erroringConfiguration = (tunnelViewModel.interfaceData.validatedConfiguration == nil) ? "Interface" : "Peer"
|
2018-10-31 19:15:09 +00:00
|
|
|
ErrorPresenter.showErrorAlert(title: "Invalid \(erroringConfiguration)", message: errorMessage, from: self)
|
2018-11-01 18:15:29 +00:00
|
|
|
self.tableView.reloadData() // Highlight erroring fields
|
2018-10-23 11:53:46 +00:00
|
|
|
case .saved(let tunnelConfiguration):
|
2018-10-24 11:49:14 +00:00
|
|
|
if let tunnel = tunnel {
|
|
|
|
// We're modifying an existing tunnel
|
|
|
|
tunnelsManager.modify(tunnel: tunnel, with: tunnelConfiguration) { [weak self] (error) in
|
|
|
|
if let error = error {
|
2018-10-31 19:15:09 +00:00
|
|
|
ErrorPresenter.showErrorAlert(error: error, from: self)
|
2018-10-24 11:49:14 +00:00
|
|
|
} else {
|
|
|
|
self?.dismiss(animated: true, completion: nil)
|
2018-10-25 05:44:38 +00:00
|
|
|
self?.delegate?.tunnelSaved(tunnel: tunnel)
|
2018-10-24 11:49:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// We're adding a new tunnel
|
|
|
|
tunnelsManager.add(tunnelConfiguration: tunnelConfiguration) { [weak self] (tunnel, error) in
|
|
|
|
if let error = error {
|
2018-10-31 19:15:09 +00:00
|
|
|
ErrorPresenter.showErrorAlert(error: error, from: self)
|
2018-10-24 11:49:14 +00:00
|
|
|
} else {
|
|
|
|
self?.dismiss(animated: true, completion: nil)
|
2018-10-25 10:20:27 +00:00
|
|
|
if let tunnel = tunnel {
|
|
|
|
self?.delegate?.tunnelSaved(tunnel: tunnel)
|
|
|
|
}
|
2018-10-24 11:49:14 +00:00
|
|
|
}
|
2018-10-23 11:53:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@objc func cancelTapped() {
|
|
|
|
dismiss(animated: true, completion: nil)
|
2018-10-25 05:44:38 +00:00
|
|
|
self.delegate?.tunnelEditingCancelled()
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
2018-10-23 11:53:46 +00:00
|
|
|
|
|
|
|
func showErrorAlert(title: String, message: String) {
|
2018-11-01 20:22:12 +00:00
|
|
|
let okAction = UIAlertAction(title: "OK", style: .default)
|
2018-10-23 11:53:46 +00:00
|
|
|
let alert = UIAlertController(title: title, message: message, preferredStyle: .alert)
|
|
|
|
alert.addAction(okAction)
|
|
|
|
|
|
|
|
self.present(alert, animated: true, completion: nil)
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: UITableViewDataSource
|
|
|
|
|
|
|
|
extension TunnelEditTableViewController {
|
|
|
|
override func numberOfSections(in tableView: UITableView) -> Int {
|
2018-10-23 12:32:10 +00:00
|
|
|
let numberOfInterfaceSections = interfaceFieldsBySection.count
|
2018-10-29 07:16:54 +00:00
|
|
|
let numberOfPeerSections = tunnelViewModel.peersData.count
|
2018-10-20 13:45:53 +00:00
|
|
|
|
2018-10-29 07:16:54 +00:00
|
|
|
return numberOfInterfaceSections + numberOfPeerSections + 1
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
|
2018-10-23 12:32:10 +00:00
|
|
|
let numberOfInterfaceSections = interfaceFieldsBySection.count
|
2018-10-29 07:16:54 +00:00
|
|
|
let numberOfPeerSections = tunnelViewModel.peersData.count
|
2018-10-20 13:45:53 +00:00
|
|
|
|
|
|
|
if (section < numberOfInterfaceSections) {
|
|
|
|
// Interface
|
2018-10-23 12:32:10 +00:00
|
|
|
return interfaceFieldsBySection[section].count
|
2018-10-29 07:16:54 +00:00
|
|
|
} else if ((numberOfPeerSections > 0) && (section < (numberOfInterfaceSections + numberOfPeerSections))) {
|
2018-10-20 13:45:53 +00:00
|
|
|
// Peer
|
2018-11-02 07:42:10 +00:00
|
|
|
let peerIndex = (section - numberOfInterfaceSections)
|
|
|
|
let peerData = tunnelViewModel.peersData[peerIndex]
|
|
|
|
let peerFieldsToShow = peerData.shouldAllowExcludePrivateIPsControl ? peerFields : peerFields.filter { $0 != .excludePrivateIPs }
|
|
|
|
return peerFieldsToShow.count
|
2018-10-20 13:45:53 +00:00
|
|
|
} else {
|
|
|
|
// Add peer
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
|
2018-10-23 12:32:10 +00:00
|
|
|
let numberOfInterfaceSections = interfaceFieldsBySection.count
|
2018-10-29 07:16:54 +00:00
|
|
|
let numberOfPeerSections = tunnelViewModel.peersData.count
|
2018-10-20 13:45:53 +00:00
|
|
|
|
|
|
|
if (section < numberOfInterfaceSections) {
|
|
|
|
// Interface
|
|
|
|
return (section == 0) ? "Interface" : nil
|
2018-10-29 07:16:54 +00:00
|
|
|
} else if ((numberOfPeerSections > 0) && (section < (numberOfInterfaceSections + numberOfPeerSections))) {
|
2018-10-20 13:45:53 +00:00
|
|
|
// Peer
|
2018-10-29 07:16:54 +00:00
|
|
|
return "Peer"
|
2018-10-20 13:45:53 +00:00
|
|
|
} else {
|
|
|
|
// Add peer
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
|
2018-10-23 12:32:10 +00:00
|
|
|
let numberOfInterfaceSections = interfaceFieldsBySection.count
|
2018-10-29 07:16:54 +00:00
|
|
|
let numberOfPeerSections = tunnelViewModel.peersData.count
|
2018-10-20 13:45:53 +00:00
|
|
|
|
|
|
|
let section = indexPath.section
|
|
|
|
let row = indexPath.row
|
|
|
|
|
|
|
|
if (section < numberOfInterfaceSections) {
|
|
|
|
// Interface
|
2018-10-23 09:53:18 +00:00
|
|
|
let interfaceData = tunnelViewModel.interfaceData
|
2018-10-23 12:32:10 +00:00
|
|
|
let field = interfaceFieldsBySection[section][row]
|
2018-10-20 13:45:53 +00:00
|
|
|
if (field == .generateKeyPair) {
|
2018-10-28 20:45:43 +00:00
|
|
|
let cell = tableView.dequeueReusableCell(withIdentifier: TunnelEditTableViewButtonCell.id, for: indexPath) as! TunnelEditTableViewButtonCell
|
2018-10-20 13:45:53 +00:00
|
|
|
cell.buttonText = field.rawValue
|
2018-10-24 08:48:52 +00:00
|
|
|
cell.onTapped = { [weak self, weak interfaceData] in
|
|
|
|
if let interfaceData = interfaceData, let s = self {
|
|
|
|
interfaceData[.privateKey] = Curve25519.generatePrivateKey().base64EncodedString()
|
|
|
|
if let privateKeyRow = s.interfaceFieldsBySection[section].firstIndex(of: .privateKey),
|
|
|
|
let publicKeyRow = s.interfaceFieldsBySection[section].firstIndex(of: .publicKey) {
|
|
|
|
let privateKeyIndex = IndexPath(row: privateKeyRow, section: section)
|
|
|
|
let publicKeyIndex = IndexPath(row: publicKeyRow, section: section)
|
|
|
|
s.tableView.reloadRows(at: [privateKeyIndex, publicKeyIndex], with: .automatic)
|
|
|
|
}
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
return cell
|
|
|
|
} else {
|
2018-10-28 20:45:43 +00:00
|
|
|
let cell = tableView.dequeueReusableCell(withIdentifier: TunnelEditTableViewKeyValueCell.id, for: indexPath) as! TunnelEditTableViewKeyValueCell
|
2018-10-23 09:53:18 +00:00
|
|
|
// Set key
|
2018-10-20 13:45:53 +00:00
|
|
|
cell.key = field.rawValue
|
2018-10-23 09:53:18 +00:00
|
|
|
// Set placeholder text
|
|
|
|
if (field == .name || field == .privateKey) {
|
2018-10-20 13:45:53 +00:00
|
|
|
cell.placeholderText = "Required"
|
2018-10-30 13:20:56 +00:00
|
|
|
} else if (field == .mtu || field == .listenPort) {
|
2018-10-20 13:45:53 +00:00
|
|
|
cell.placeholderText = "Automatic"
|
2018-10-23 09:53:18 +00:00
|
|
|
}
|
2018-10-24 08:53:18 +00:00
|
|
|
// Set editable
|
|
|
|
if (field == .publicKey) {
|
|
|
|
cell.isValueEditable = false
|
|
|
|
}
|
2018-10-29 00:47:48 +00:00
|
|
|
// Set keyboardType
|
|
|
|
if (field == .mtu || field == .listenPort) {
|
|
|
|
cell.keyboardType = .numberPad
|
|
|
|
} else if (field == .addresses || field == .dns) {
|
|
|
|
cell.keyboardType = .numbersAndPunctuation
|
|
|
|
}
|
2018-11-01 18:15:29 +00:00
|
|
|
// Show erroring fields
|
|
|
|
cell.isValueValid = (!interfaceData.fieldsWithError.contains(field))
|
2018-10-23 09:53:18 +00:00
|
|
|
// Bind values to view model
|
|
|
|
cell.value = interfaceData[field]
|
2018-10-29 11:08:32 +00:00
|
|
|
if (field == .dns) { // While editing DNS, you might directly set exclude private IPs
|
|
|
|
cell.onValueBeingEdited = { [weak interfaceData] value in
|
|
|
|
interfaceData?[field] = value
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
cell.onValueChanged = { [weak interfaceData] value in
|
|
|
|
interfaceData?[field] = value
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
2018-10-24 08:41:34 +00:00
|
|
|
// Compute public key live
|
|
|
|
if (field == .privateKey) {
|
|
|
|
cell.onValueBeingEdited = { [weak self, weak interfaceData] value in
|
|
|
|
if let interfaceData = interfaceData, let s = self {
|
|
|
|
interfaceData[.privateKey] = value
|
|
|
|
if let row = s.interfaceFieldsBySection[section].firstIndex(of: .publicKey) {
|
|
|
|
s.tableView.reloadRows(at: [IndexPath(row: row, section: section)], with: .none)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
return cell
|
|
|
|
}
|
2018-10-29 07:16:54 +00:00
|
|
|
} else if ((numberOfPeerSections > 0) && (section < (numberOfInterfaceSections + numberOfPeerSections))) {
|
2018-10-20 13:45:53 +00:00
|
|
|
// Peer
|
2018-10-29 07:16:54 +00:00
|
|
|
let peerIndex = (section - numberOfInterfaceSections)
|
2018-10-23 09:53:18 +00:00
|
|
|
let peerData = tunnelViewModel.peersData[peerIndex]
|
2018-11-02 07:42:10 +00:00
|
|
|
let peerFieldsToShow = peerData.shouldAllowExcludePrivateIPsControl ? peerFields : peerFields.filter { $0 != .excludePrivateIPs }
|
|
|
|
let field = peerFieldsToShow[row]
|
2018-10-20 13:45:53 +00:00
|
|
|
if (field == .deletePeer) {
|
2018-10-28 20:45:43 +00:00
|
|
|
let cell = tableView.dequeueReusableCell(withIdentifier: TunnelEditTableViewButtonCell.id, for: indexPath) as! TunnelEditTableViewButtonCell
|
2018-10-20 13:45:53 +00:00
|
|
|
cell.buttonText = field.rawValue
|
2018-11-02 07:54:10 +00:00
|
|
|
cell.hasDestructiveAction = true
|
2018-10-20 13:45:53 +00:00
|
|
|
cell.onTapped = { [weak self, weak peerData] in
|
|
|
|
guard let peerData = peerData else { return }
|
|
|
|
guard let s = self else { return }
|
|
|
|
s.showConfirmationAlert(message: "Delete this peer?",
|
|
|
|
buttonTitle: "Delete", from: cell,
|
|
|
|
onConfirmed: { [weak s] in
|
|
|
|
guard let s = s else { return }
|
|
|
|
let removedSectionIndices = s.deletePeer(peer: peerData)
|
2018-11-02 07:42:10 +00:00
|
|
|
let shouldShowExcludePrivateIPs = (s.tunnelViewModel.peersData.count == 1 &&
|
|
|
|
s.tunnelViewModel.peersData[0].shouldAllowExcludePrivateIPsControl)
|
|
|
|
tableView.performBatchUpdates({
|
|
|
|
s.tableView.deleteSections(removedSectionIndices, with: .automatic)
|
|
|
|
if (shouldShowExcludePrivateIPs) {
|
|
|
|
if let row = s.peerFields.firstIndex(of: .excludePrivateIPs) {
|
|
|
|
let rowIndexPath = IndexPath(row: row, section: numberOfInterfaceSections /* First peer section */)
|
|
|
|
s.tableView.insertRows(at: [rowIndexPath], with: .automatic)
|
|
|
|
}
|
|
|
|
|
2018-10-29 11:08:32 +00:00
|
|
|
}
|
2018-11-02 07:42:10 +00:00
|
|
|
})
|
2018-10-20 13:45:53 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
return cell
|
|
|
|
} else if (field == .excludePrivateIPs) {
|
2018-10-28 20:45:43 +00:00
|
|
|
let cell = tableView.dequeueReusableCell(withIdentifier: TunnelEditTableViewSwitchCell.id, for: indexPath) as! TunnelEditTableViewSwitchCell
|
2018-10-20 13:45:53 +00:00
|
|
|
cell.message = field.rawValue
|
2018-10-29 11:08:32 +00:00
|
|
|
cell.isEnabled = peerData.shouldAllowExcludePrivateIPsControl
|
|
|
|
cell.isOn = peerData.excludePrivateIPsValue
|
|
|
|
cell.onSwitchToggled = { [weak self] (isOn) in
|
|
|
|
guard let s = self else { return }
|
|
|
|
peerData.excludePrivateIPsValueChanged(isOn: isOn, dnsServers: s.tunnelViewModel.interfaceData[.dns])
|
|
|
|
if let row = s.peerFields.firstIndex(of: .allowedIPs) {
|
|
|
|
s.tableView.reloadRows(at: [IndexPath(row: row, section: section)], with: .none)
|
|
|
|
}
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
return cell
|
|
|
|
} else {
|
2018-10-28 20:45:43 +00:00
|
|
|
let cell = tableView.dequeueReusableCell(withIdentifier: TunnelEditTableViewKeyValueCell.id, for: indexPath) as! TunnelEditTableViewKeyValueCell
|
2018-10-23 09:53:18 +00:00
|
|
|
// Set key
|
2018-10-20 13:45:53 +00:00
|
|
|
cell.key = field.rawValue
|
2018-10-23 09:53:18 +00:00
|
|
|
// Set placeholder text
|
|
|
|
if (field == .publicKey) {
|
2018-10-20 13:45:53 +00:00
|
|
|
cell.placeholderText = "Required"
|
2018-10-30 13:20:56 +00:00
|
|
|
} else if (field == .preSharedKey) {
|
|
|
|
cell.placeholderText = "Optional"
|
|
|
|
} else if (field == .persistentKeepAlive) {
|
|
|
|
cell.placeholderText = "Off"
|
2018-10-23 09:53:18 +00:00
|
|
|
}
|
2018-10-30 13:20:56 +00:00
|
|
|
// Set keyboardType
|
2018-10-29 00:47:48 +00:00
|
|
|
if (field == .persistentKeepAlive) {
|
|
|
|
cell.keyboardType = .numberPad
|
|
|
|
} else if (field == .allowedIPs) {
|
|
|
|
cell.keyboardType = .numbersAndPunctuation
|
|
|
|
}
|
2018-11-01 18:15:29 +00:00
|
|
|
// Show erroring fields
|
|
|
|
cell.isValueValid = (!peerData.fieldsWithError.contains(field))
|
2018-10-23 09:53:18 +00:00
|
|
|
// Bind values to view model
|
|
|
|
cell.value = peerData[field]
|
2018-10-29 11:08:32 +00:00
|
|
|
if (field != .allowedIPs) {
|
|
|
|
cell.onValueChanged = { [weak peerData] value in
|
|
|
|
peerData?[field] = value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Compute state of exclude private IPs live
|
|
|
|
if (field == .allowedIPs) {
|
|
|
|
cell.onValueBeingEdited = { [weak self, weak peerData] value in
|
|
|
|
if let peerData = peerData, let s = self {
|
2018-11-02 07:42:10 +00:00
|
|
|
let oldValue = peerData.shouldAllowExcludePrivateIPsControl
|
2018-10-29 11:08:32 +00:00
|
|
|
peerData[.allowedIPs] = value
|
2018-11-02 07:42:10 +00:00
|
|
|
if (oldValue != peerData.shouldAllowExcludePrivateIPsControl) {
|
|
|
|
if let row = s.peerFields.firstIndex(of: .excludePrivateIPs) {
|
|
|
|
if (peerData.shouldAllowExcludePrivateIPsControl) {
|
|
|
|
s.tableView.insertRows(at: [IndexPath(row: row, section: section)], with: .automatic)
|
|
|
|
} else {
|
|
|
|
s.tableView.deleteRows(at: [IndexPath(row: row, section: section)], with: .automatic)
|
|
|
|
}
|
|
|
|
}
|
2018-10-29 11:08:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
return cell
|
|
|
|
}
|
|
|
|
} else {
|
2018-10-29 07:16:54 +00:00
|
|
|
assert(section == (numberOfInterfaceSections + numberOfPeerSections))
|
2018-10-20 13:45:53 +00:00
|
|
|
// Add peer
|
2018-10-28 20:45:43 +00:00
|
|
|
let cell = tableView.dequeueReusableCell(withIdentifier: TunnelEditTableViewButtonCell.id, for: indexPath) as! TunnelEditTableViewButtonCell
|
2018-10-20 13:45:53 +00:00
|
|
|
cell.buttonText = "Add peer"
|
|
|
|
cell.onTapped = { [weak self] in
|
|
|
|
guard let s = self else { return }
|
2018-11-02 07:42:10 +00:00
|
|
|
let shouldHideExcludePrivateIPs = (s.tunnelViewModel.peersData.count == 1 &&
|
|
|
|
s.tunnelViewModel.peersData[0].shouldAllowExcludePrivateIPsControl)
|
2018-10-20 13:45:53 +00:00
|
|
|
let addedSectionIndices = s.appendEmptyPeer()
|
2018-11-02 07:42:10 +00:00
|
|
|
tableView.performBatchUpdates({
|
|
|
|
tableView.insertSections(addedSectionIndices, with: .automatic)
|
|
|
|
if (shouldHideExcludePrivateIPs) {
|
|
|
|
if let row = s.peerFields.firstIndex(of: .excludePrivateIPs) {
|
|
|
|
let rowIndexPath = IndexPath(row: row, section: numberOfInterfaceSections /* First peer section */)
|
|
|
|
s.tableView.deleteRows(at: [rowIndexPath], with: .automatic)
|
|
|
|
}
|
2018-10-29 11:08:32 +00:00
|
|
|
}
|
2018-11-02 07:42:10 +00:00
|
|
|
}, completion: nil)
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
return cell
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func appendEmptyPeer() -> IndexSet {
|
2018-10-23 12:32:10 +00:00
|
|
|
let numberOfInterfaceSections = interfaceFieldsBySection.count
|
2018-10-20 13:45:53 +00:00
|
|
|
|
2018-10-23 09:53:18 +00:00
|
|
|
tunnelViewModel.appendEmptyPeer()
|
|
|
|
let addedPeerIndex = tunnelViewModel.peersData.count - 1
|
2018-10-20 13:45:53 +00:00
|
|
|
|
2018-10-29 07:16:54 +00:00
|
|
|
let addedSectionIndices = IndexSet(integer: (numberOfInterfaceSections + addedPeerIndex))
|
2018-10-20 13:45:53 +00:00
|
|
|
return addedSectionIndices
|
|
|
|
}
|
|
|
|
|
2018-10-23 09:53:18 +00:00
|
|
|
func deletePeer(peer: TunnelViewModel.PeerData) -> IndexSet {
|
2018-10-23 12:32:10 +00:00
|
|
|
let numberOfInterfaceSections = interfaceFieldsBySection.count
|
2018-10-20 13:45:53 +00:00
|
|
|
|
2018-10-23 09:53:18 +00:00
|
|
|
assert(peer.index < tunnelViewModel.peersData.count)
|
|
|
|
tunnelViewModel.deletePeer(peer: peer)
|
2018-10-20 13:45:53 +00:00
|
|
|
|
2018-10-29 07:16:54 +00:00
|
|
|
let removedSectionIndices = IndexSet(integer: (numberOfInterfaceSections + peer.index))
|
2018-10-20 13:45:53 +00:00
|
|
|
return removedSectionIndices
|
|
|
|
}
|
|
|
|
|
|
|
|
func showConfirmationAlert(message: String, buttonTitle: String, from sourceView: UIView,
|
|
|
|
onConfirmed: @escaping (() -> Void)) {
|
|
|
|
let destroyAction = UIAlertAction(title: buttonTitle, style: .destructive) { (action) in
|
|
|
|
onConfirmed()
|
|
|
|
}
|
|
|
|
let cancelAction = UIAlertAction(title: "Cancel", style: .cancel)
|
|
|
|
let alert = UIAlertController(title: "", message: message, preferredStyle: .actionSheet)
|
|
|
|
alert.addAction(destroyAction)
|
|
|
|
alert.addAction(cancelAction)
|
|
|
|
|
|
|
|
// popoverPresentationController will be nil on iPhone and non-nil on iPad
|
|
|
|
alert.popoverPresentationController?.sourceView = sourceView
|
|
|
|
|
|
|
|
self.present(alert, animated: true, completion: nil)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-31 00:00:27 +00:00
|
|
|
class TunnelEditTableViewKeyValueCell: CopyableLabelTableViewCell {
|
2018-10-28 20:45:43 +00:00
|
|
|
static let id: String = "TunnelEditTableViewKeyValueCell"
|
2018-10-20 13:45:53 +00:00
|
|
|
var key: String {
|
|
|
|
get { return keyLabel.text ?? "" }
|
|
|
|
set(value) {keyLabel.text = value }
|
|
|
|
}
|
|
|
|
var value: String {
|
|
|
|
get { return valueTextField.text ?? "" }
|
|
|
|
set(value) { valueTextField.text = value }
|
|
|
|
}
|
|
|
|
var placeholderText: String {
|
|
|
|
get { return valueTextField.placeholder ?? "" }
|
|
|
|
set(value) { valueTextField.placeholder = value }
|
|
|
|
}
|
|
|
|
var isValueEditable: Bool {
|
|
|
|
get { return valueTextField.isEnabled }
|
|
|
|
set(value) {
|
2018-10-31 00:00:27 +00:00
|
|
|
super.copyableGesture = !value
|
2018-10-20 13:45:53 +00:00
|
|
|
valueTextField.isEnabled = value
|
|
|
|
keyLabel.textColor = value ? UIColor.black : UIColor.gray
|
2018-10-24 08:52:28 +00:00
|
|
|
valueTextField.textColor = value ? UIColor.black : UIColor.gray
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
var isValueValid: Bool = true {
|
2018-11-01 18:15:29 +00:00
|
|
|
didSet {
|
|
|
|
if (isValueValid) {
|
2018-10-20 13:45:53 +00:00
|
|
|
keyLabel.textColor = isValueEditable ? UIColor.black : UIColor.gray
|
|
|
|
} else {
|
|
|
|
keyLabel.textColor = UIColor.red
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-10-29 00:47:48 +00:00
|
|
|
var keyboardType: UIKeyboardType {
|
|
|
|
get { return valueTextField.keyboardType }
|
|
|
|
set(value) { valueTextField.keyboardType = value }
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
|
2018-10-22 09:54:58 +00:00
|
|
|
var onValueChanged: ((String) -> Void)? = nil
|
2018-10-24 08:41:34 +00:00
|
|
|
var onValueBeingEdited: ((String) -> Void)? = nil
|
2018-10-22 09:54:58 +00:00
|
|
|
|
2018-10-20 13:45:53 +00:00
|
|
|
let keyLabel: UILabel
|
|
|
|
let valueTextField: UITextField
|
|
|
|
|
2018-10-22 09:54:58 +00:00
|
|
|
private var textFieldValueOnBeginEditing: String = ""
|
|
|
|
|
2018-10-20 13:45:53 +00:00
|
|
|
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
|
|
|
|
keyLabel = UILabel()
|
|
|
|
valueTextField = UITextField()
|
|
|
|
super.init(style: style, reuseIdentifier: reuseIdentifier)
|
2018-10-31 00:00:27 +00:00
|
|
|
isValueEditable = true
|
2018-10-20 13:45:53 +00:00
|
|
|
contentView.addSubview(keyLabel)
|
|
|
|
keyLabel.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
keyLabel.textAlignment = .right
|
|
|
|
let widthRatioConstraint = NSLayoutConstraint(item: keyLabel, attribute: .width,
|
|
|
|
relatedBy: .equal,
|
|
|
|
toItem: self, attribute: .width,
|
|
|
|
multiplier: 0.4, constant: 0)
|
2018-10-22 08:19:21 +00:00
|
|
|
// The "Persistent Keepalive" key doesn't fit into 0.4 * width on the iPhone SE,
|
|
|
|
// so set a CR priority > the 0.4-constraint's priority.
|
|
|
|
widthRatioConstraint.priority = .defaultHigh + 1
|
|
|
|
keyLabel.setContentCompressionResistancePriority(.defaultHigh + 2, for: .horizontal)
|
2018-10-20 13:45:53 +00:00
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
keyLabel.centerYAnchor.constraint(equalTo: contentView.centerYAnchor),
|
|
|
|
keyLabel.leftAnchor.constraint(equalTo: contentView.leftAnchor, constant: 8),
|
|
|
|
widthRatioConstraint
|
|
|
|
])
|
|
|
|
contentView.addSubview(valueTextField)
|
|
|
|
valueTextField.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
valueTextField.centerYAnchor.constraint(equalTo: contentView.centerYAnchor),
|
|
|
|
valueTextField.leftAnchor.constraint(equalTo: keyLabel.rightAnchor, constant: 16),
|
|
|
|
valueTextField.rightAnchor.constraint(equalTo: contentView.rightAnchor, constant: -8),
|
|
|
|
])
|
2018-10-22 09:54:58 +00:00
|
|
|
valueTextField.delegate = self
|
2018-10-29 00:47:48 +00:00
|
|
|
|
|
|
|
valueTextField.autocapitalizationType = .none
|
|
|
|
valueTextField.autocorrectionType = .no
|
|
|
|
valueTextField.spellCheckingType = .no
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
|
2018-10-31 15:38:05 +00:00
|
|
|
override var textToCopy: String? {
|
|
|
|
return self.valueTextField.text
|
|
|
|
}
|
|
|
|
|
2018-10-20 13:45:53 +00:00
|
|
|
required init?(coder aDecoder: NSCoder) {
|
|
|
|
fatalError("init(coder:) has not been implemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
override func prepareForReuse() {
|
|
|
|
super.prepareForReuse()
|
|
|
|
key = ""
|
|
|
|
value = ""
|
|
|
|
placeholderText = ""
|
|
|
|
isValueEditable = true
|
|
|
|
isValueValid = true
|
2018-10-29 09:42:19 +00:00
|
|
|
keyboardType = .default
|
2018-10-22 09:54:58 +00:00
|
|
|
onValueChanged = nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-28 20:45:43 +00:00
|
|
|
extension TunnelEditTableViewKeyValueCell: UITextFieldDelegate {
|
2018-10-22 09:54:58 +00:00
|
|
|
func textFieldDidBeginEditing(_ textField: UITextField) {
|
|
|
|
textFieldValueOnBeginEditing = textField.text ?? ""
|
|
|
|
isValueValid = true
|
|
|
|
}
|
|
|
|
func textFieldDidEndEditing(_ textField: UITextField) {
|
|
|
|
let isModified = (textField.text ?? "" != textFieldValueOnBeginEditing)
|
|
|
|
guard (isModified) else { return }
|
|
|
|
if let onValueChanged = onValueChanged {
|
|
|
|
onValueChanged(textField.text ?? "")
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
2018-10-24 08:41:34 +00:00
|
|
|
func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange, replacementString string: String) -> Bool {
|
|
|
|
if let onValueBeingEdited = onValueBeingEdited {
|
|
|
|
let modifiedText = ((textField.text ?? "") as NSString).replacingCharacters(in: range, with: string)
|
|
|
|
onValueBeingEdited(modifiedText)
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
|
2018-10-28 20:45:43 +00:00
|
|
|
class TunnelEditTableViewButtonCell: UITableViewCell {
|
|
|
|
static let id: String = "TunnelEditTableViewButtonCell"
|
2018-10-20 13:45:53 +00:00
|
|
|
var buttonText: String {
|
|
|
|
get { return button.title(for: .normal) ?? "" }
|
|
|
|
set(value) { button.setTitle(value, for: .normal) }
|
|
|
|
}
|
2018-11-02 07:54:10 +00:00
|
|
|
var hasDestructiveAction: Bool {
|
|
|
|
get { return button.tintColor == UIColor.red }
|
|
|
|
set(value) { button.tintColor = value ? UIColor.red : buttonStandardTintColor }
|
|
|
|
}
|
2018-10-20 13:45:53 +00:00
|
|
|
var onTapped: (() -> Void)? = nil
|
|
|
|
|
|
|
|
let button: UIButton
|
2018-11-02 07:54:10 +00:00
|
|
|
var buttonStandardTintColor: UIColor
|
2018-10-20 13:45:53 +00:00
|
|
|
|
|
|
|
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
|
|
|
|
button = UIButton(type: .system)
|
2018-11-02 07:54:10 +00:00
|
|
|
buttonStandardTintColor = button.tintColor
|
2018-10-20 13:45:53 +00:00
|
|
|
super.init(style: style, reuseIdentifier: reuseIdentifier)
|
|
|
|
contentView.addSubview(button)
|
|
|
|
button.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
button.centerYAnchor.constraint(equalTo: contentView.centerYAnchor),
|
|
|
|
button.centerXAnchor.constraint(equalTo: contentView.centerXAnchor)
|
|
|
|
])
|
|
|
|
button.addTarget(self, action: #selector(buttonTapped), for: .touchUpInside)
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc func buttonTapped() {
|
|
|
|
onTapped?()
|
|
|
|
}
|
|
|
|
|
|
|
|
required init?(coder aDecoder: NSCoder) {
|
|
|
|
fatalError("init(coder:) has not been implemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
override func prepareForReuse() {
|
2018-10-28 20:56:40 +00:00
|
|
|
super.prepareForReuse()
|
2018-10-20 13:45:53 +00:00
|
|
|
buttonText = ""
|
|
|
|
onTapped = nil
|
2018-11-02 07:54:10 +00:00
|
|
|
hasDestructiveAction = false
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-28 20:45:43 +00:00
|
|
|
class TunnelEditTableViewSwitchCell: UITableViewCell {
|
|
|
|
static let id: String = "TunnelEditTableViewSwitchCell"
|
2018-10-20 13:45:53 +00:00
|
|
|
var message: String {
|
|
|
|
get { return textLabel?.text ?? "" }
|
|
|
|
set(value) { textLabel!.text = value }
|
|
|
|
}
|
|
|
|
var isOn: Bool {
|
|
|
|
get { return switchView.isOn }
|
|
|
|
set(value) { switchView.isOn = value }
|
|
|
|
}
|
2018-10-29 11:08:32 +00:00
|
|
|
var isEnabled: Bool {
|
|
|
|
get { return switchView.isEnabled }
|
|
|
|
set(value) {
|
|
|
|
switchView.isEnabled = value
|
|
|
|
textLabel?.textColor = value ? UIColor.black : UIColor.gray
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var onSwitchToggled: ((Bool) -> Void)? = nil
|
2018-10-20 13:45:53 +00:00
|
|
|
|
|
|
|
let switchView: UISwitch
|
|
|
|
|
|
|
|
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
|
|
|
|
switchView = UISwitch()
|
|
|
|
super.init(style: .default, reuseIdentifier: reuseIdentifier)
|
|
|
|
accessoryView = switchView
|
2018-10-29 11:08:32 +00:00
|
|
|
|
|
|
|
switchView.addTarget(self, action: #selector(switchToggled), for: .valueChanged)
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc func switchToggled() {
|
|
|
|
onSwitchToggled?(switchView.isOn)
|
2018-10-20 13:45:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
required init?(coder aDecoder: NSCoder) {
|
|
|
|
fatalError("init(coder:) has not been implemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
override func prepareForReuse() {
|
2018-10-28 20:56:40 +00:00
|
|
|
super.prepareForReuse()
|
2018-10-20 13:45:53 +00:00
|
|
|
message = ""
|
|
|
|
isOn = false
|
|
|
|
}
|
|
|
|
}
|