passepartout-apple/Passepartout/Library/Sources/AppUI/UI/ConnectionStatusView.swift

104 lines
2.9 KiB
Swift

//
// ConnectionStatusView.swift
// Passepartout
//
// Created by Davide De Rosa on 9/4/24.
// Copyright (c) 2024 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 PassepartoutKit
import SwiftUI
public struct ConnectionStatusView: View, ThemeProviding {
@EnvironmentObject
public var theme: Theme
@ObservedObject
private var tunnel: ExtendedTunnel
public init(tunnel: ExtendedTunnel) {
self.tunnel = tunnel
}
public var body: some View {
Text(statusDescription)
.font(.headline)
.foregroundStyle(tunnel.statusColor(theme))
}
}
private extension ConnectionStatusView {
var statusDescription: String {
if let lastErrorCode = tunnel.lastErrorCode {
return lastErrorCode.localizedDescription
}
let status = tunnel.connectionStatus
switch status {
case .active:
if let dataCount = tunnel.dataCount {
let down = dataCount.received.descriptionAsDataUnit
let up = dataCount.sent.descriptionAsDataUnit
return "\(down)\(up)"
}
case .inactive:
var desc = status.localizedDescription
if tunnel.currentProfile?.onDemand ?? false {
desc += Strings.Ui.ConnectionStatus.onDemandSuffix
}
return desc
default:
break
}
return status.localizedDescription
}
}
#Preview("Connected") {
ConnectionStatusView(tunnel: .mock)
.task {
try? await ExtendedTunnel.mock.connect(with: .mock, processor: .mock)
}
.frame(width: 100, height: 100)
.withMockEnvironment()
}
#Preview("On-Demand") {
var builder = Profile.Builder()
var onDemand = OnDemandModule.Builder()
onDemand.isEnabled = true
builder.modules = [onDemand.tryBuild()]
let profile: Profile
do {
profile = try builder.tryBuild()
} catch {
fatalError()
}
return ConnectionStatusView(tunnel: .mock)
.task {
try? await ExtendedTunnel.mock.connect(with: profile, processor: .mock)
}
.frame(width: 100, height: 100)
.withMockEnvironment()
}