passepartout-apple/Passepartout/Library/Sources/UILibrary/Views/UI/ConnectionStatusText.swift
Davide f3d13d0cdf
Refactor AppContext creation and profile processing (#810)
Streamline initialization of AppContext objects without singletons,
especially because some are interconnected.

Rethink ProfileProcessor to be the only gateway of profile processing
for:

- Include
- Save
- Connect

Provide closures with access to the IAPManager for eligibility checks.

Finally, take a ProfileProcessor parameter in:

- ProfileManager (for isIncluded and willSave)
- ExtendedTunnel (for willConnect)

so that it's used implicitly without having to put it into the SwiftUI
environment.

Other than that:

- Move AppError to CommonLibrary
- Skip decoding of attributes from Core Data because they are already
part of the profile
2024-11-04 23:34:22 +01:00

105 lines
2.9 KiB
Swift

//
// ConnectionStatusText.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 CommonLibrary
import Foundation
import PassepartoutKit
import SwiftUI
public struct ConnectionStatusText: 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)
}
}
private extension ConnectionStatusText {
var statusDescription: String {
if let lastErrorCode = tunnel.lastErrorCode {
return lastErrorCode.localizedDescription(style: .tunnel)
}
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 let profile = tunnel.currentProfile {
if profile.onDemand {
desc += Strings.Ui.ConnectionStatus.onDemandSuffix
}
}
return desc
default:
break
}
return status.localizedDescription
}
}
#Preview("Connected") {
ConnectionStatusText(tunnel: .mock)
.task {
try? await ExtendedTunnel.mock.connect(with: .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 ConnectionStatusText(tunnel: .mock)
.task {
try? await ExtendedTunnel.mock.connect(with: profile)
}
.frame(width: 100, height: 100)
.withMockEnvironment()
}