mirror of
https://github.com/passepartoutvpn/passepartout-apple.git
synced 2024-12-26 19:32:41 +00:00
0047d095fb
- Mac - Drop all styles - Tweak hide title bar - Hide navigation bar - Restore single section for all profiles - Allows using NavigationLink safely - Indirectly fixes multitasking - Retains selection on profile activation - Clean up presentActiveProfile - Leave active profile in its position - Fixes Mac flashing row selection on profile activation - Unify profile row appearance - Use fixed .headline font - Add subtitles to inactive profiles - Use padding rather than fixed row height CAVEATS: - Do not preselect active profile on iPad launch, as doing so seems to present two ProfileView on top of each other, one from MainView and one from the NavigationLink. - Do not touch .listStyle() of master view, as it seems to break navigation esp. in iPad multitasking.
224 lines
6.4 KiB
Swift
224 lines
6.4 KiB
Swift
//
|
|
// TunnelKit+L10n.swift
|
|
// Passepartout
|
|
//
|
|
// Created by Davide De Rosa on 3/12/22.
|
|
// Copyright (c) 2022 Davide De Rosa. All rights reserved.
|
|
//
|
|
// https://github.com/passepartoutvpn
|
|
//
|
|
// This file is part of Passepartout.
|
|
//
|
|
// Passepartout is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// Passepartout is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Passepartout. If not, see <http://www.gnu.org/licenses/>.
|
|
//
|
|
|
|
import Foundation
|
|
import TunnelKitManager
|
|
import TunnelKitOpenVPN
|
|
import TunnelKitWireGuard
|
|
import NetworkExtension
|
|
import PassepartoutCore
|
|
|
|
extension VPNStatus {
|
|
var localizedDescription: String {
|
|
switch self {
|
|
case .connecting:
|
|
return L10n.Tunnelkit.Vpn.connecting
|
|
|
|
case .connected:
|
|
return L10n.Tunnelkit.Vpn.active
|
|
|
|
case .disconnecting:
|
|
return L10n.Tunnelkit.Vpn.disconnecting
|
|
|
|
case .disconnected:
|
|
return L10n.Tunnelkit.Vpn.inactive
|
|
}
|
|
}
|
|
}
|
|
|
|
extension DataCount {
|
|
var localizedDescription: String {
|
|
let down = received.descriptionAsDataUnit
|
|
let up = sent.descriptionAsDataUnit
|
|
return "↓\(down) ↑\(up)"
|
|
}
|
|
}
|
|
|
|
extension Int {
|
|
var localizedDescriptionAsMTU: String {
|
|
guard self != 0 else {
|
|
return L10n.Global.Strings.default
|
|
}
|
|
return description
|
|
}
|
|
}
|
|
|
|
extension TimeInterval {
|
|
var localizedDescriptionAsKeepAlive: String {
|
|
let V = L10n.Endpoint.Advanced.Openvpn.Items.self
|
|
if self > 0 {
|
|
return V.KeepAlive.Value.seconds(Int(self))
|
|
} else {
|
|
return L10n.Global.Strings.disabled
|
|
}
|
|
}
|
|
}
|
|
|
|
extension Optional where Wrapped == IPv4Settings {
|
|
var localizedAddress: String {
|
|
if let ipv4 = self {
|
|
return "\(ipv4.address)/\(ipv4.addressMask)"
|
|
} else {
|
|
return L10n.Global.Strings.none
|
|
}
|
|
}
|
|
|
|
var localizedDefaultGateway: String {
|
|
return self?.defaultGateway ?? L10n.Global.Strings.none
|
|
}
|
|
}
|
|
|
|
extension Optional where Wrapped == IPv6Settings {
|
|
var localizedAddress: String {
|
|
if let ipv6 = self {
|
|
return "\(ipv6.address)/\(ipv6.addressPrefixLength)"
|
|
} else {
|
|
return L10n.Global.Strings.none
|
|
}
|
|
}
|
|
|
|
var localizedDefaultGateway: String {
|
|
return self?.defaultGateway ?? L10n.Global.Strings.none
|
|
}
|
|
}
|
|
|
|
extension IPv4Settings.Route {
|
|
var localizedDescription: String {
|
|
return "\(destination)/\(mask) -> \(gateway)"
|
|
}
|
|
}
|
|
|
|
extension IPv6Settings.Route {
|
|
var localizedDescription: String {
|
|
return "\(destination)/\(prefixLength) -> \(gateway)"
|
|
}
|
|
}
|
|
|
|
extension Error {
|
|
var localizedVPNDescription: String? {
|
|
if let ovpnError = self as? OpenVPNProviderError {
|
|
return ovpnErrorDescription(ovpnError)
|
|
}
|
|
if let wgError = self as? WireGuardProviderError {
|
|
return wgErrorDescription(wgError)
|
|
}
|
|
if let neError = self as? NEVPNError {
|
|
return neErrorDescription(neError)
|
|
}
|
|
return localizedDescription
|
|
}
|
|
|
|
private func ovpnErrorDescription(_ error: OpenVPNProviderError) -> String? {
|
|
let V = L10n.Tunnelkit.Errors.Vpn.self
|
|
switch error {
|
|
case .socketActivity, .timeout:
|
|
return V.timeout
|
|
|
|
case .dnsFailure:
|
|
return V.dns
|
|
|
|
case .tlsInitialization, .tlsServerVerification, .tlsHandshake:
|
|
return V.tls
|
|
|
|
case .authentication:
|
|
return V.auth
|
|
|
|
case .encryptionInitialization, .encryptionData:
|
|
return V.encryption
|
|
|
|
case .serverCompression, .lzo:
|
|
return V.compression
|
|
|
|
case .networkChanged:
|
|
return V.network
|
|
|
|
case .routing:
|
|
return V.routing
|
|
|
|
case .gatewayUnattainable:
|
|
return V.gateway
|
|
|
|
case .serverShutdown:
|
|
return V.shutdown
|
|
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
private func wgErrorDescription(_ error: WireGuardProviderError) -> String? {
|
|
let V = L10n.Tunnelkit.Errors.Vpn.self
|
|
switch error {
|
|
case .dnsResolutionFailure:
|
|
return V.dns
|
|
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
private func neErrorDescription(_ error: NEVPNError) -> String? {
|
|
return error.localizedDescription.capitalized
|
|
}
|
|
}
|
|
|
|
extension Error {
|
|
var localizedVPNParsingDescription: String? {
|
|
if let ovpnError = self as? OpenVPN.ConfigurationError {
|
|
return ovpnErrorDescription(ovpnError)
|
|
}
|
|
pp_log.error("Could not parse configuration URL: \(localizedDescription)")
|
|
return L10n.Tunnelkit.Errors.parsing(localizedDescription)
|
|
}
|
|
|
|
private func ovpnErrorDescription(_ error: OpenVPN.ConfigurationError) -> String {
|
|
let V = L10n.Tunnelkit.Errors.Openvpn.self
|
|
switch error {
|
|
case .encryptionPassphrase:
|
|
pp_log.error("Could not parse configuration URL: unable to decrypt, \(error.localizedDescription)")
|
|
return V.passphraseRequired
|
|
|
|
case .unableToDecrypt(let error):
|
|
pp_log.error("Could not parse configuration URL: unable to decrypt, \(error.localizedDescription)")
|
|
return V.decryption
|
|
|
|
case .malformed(let option):
|
|
pp_log.error("Could not parse configuration URL: malformed option, \(option)")
|
|
return V.malformed(option)
|
|
|
|
case .missingConfiguration(let option):
|
|
pp_log.error("Could not parse configuration URL: missing configuration, \(option)")
|
|
return V.requiredOption(option)
|
|
|
|
case .unsupportedConfiguration(var option):
|
|
if option.contains("external") {
|
|
option.append(" (see FAQ)")
|
|
}
|
|
pp_log.error("Could not parse configuration URL: unsupported configuration, \(option)")
|
|
return V.unsupportedOption(option)
|
|
}
|
|
}
|
|
}
|