2022-04-12 13:09:14 +00:00
|
|
|
//
|
|
|
|
// ProfileView.swift
|
|
|
|
// Passepartout
|
|
|
|
//
|
|
|
|
// Created by Davide De Rosa on 2/6/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 SwiftUI
|
|
|
|
import PassepartoutCore
|
|
|
|
|
|
|
|
struct ProfileView: View {
|
|
|
|
enum ModalType: Int, Identifiable {
|
2022-04-18 14:42:17 +00:00
|
|
|
case shortcuts
|
|
|
|
|
2022-04-12 13:09:14 +00:00
|
|
|
case rename
|
|
|
|
|
2022-04-18 14:42:17 +00:00
|
|
|
case paywallShortcuts
|
|
|
|
|
2022-04-13 18:12:25 +00:00
|
|
|
case paywallNetworkSettings
|
|
|
|
|
2022-04-12 13:09:14 +00:00
|
|
|
case paywallTrustedNetworks
|
|
|
|
|
|
|
|
var id: Int {
|
|
|
|
return rawValue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Environment(\.presentationMode) private var presentationMode
|
|
|
|
|
|
|
|
@ObservedObject private var profileManager: ProfileManager
|
|
|
|
|
|
|
|
private let header: Profile.Header
|
|
|
|
|
|
|
|
private var isDeleted: Bool {
|
|
|
|
!profileManager.isExistingProfile(withId: header.id)
|
|
|
|
}
|
|
|
|
|
|
|
|
@State private var modalType: ModalType?
|
|
|
|
|
|
|
|
@State private var isLoaded = false
|
|
|
|
|
|
|
|
init(header: Profile.Header?) {
|
|
|
|
let profileManager: ProfileManager = .shared
|
|
|
|
|
|
|
|
self.profileManager = profileManager
|
|
|
|
self.header = header ?? profileManager.activeHeader ?? Profile.placeholder.header
|
|
|
|
}
|
|
|
|
|
|
|
|
var body: some View {
|
|
|
|
debugChanges()
|
|
|
|
return Group {
|
|
|
|
if !isDeleted {
|
|
|
|
List {
|
|
|
|
if isLoaded {
|
|
|
|
mainView
|
|
|
|
} else {
|
|
|
|
loadingSection
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
welcomeView
|
2022-04-20 09:48:58 +00:00
|
|
|
.navigationBarHidden(true)
|
2022-04-12 13:09:14 +00:00
|
|
|
}
|
|
|
|
}.themeSecondaryView()
|
|
|
|
.navigationTitle(title)
|
|
|
|
.toolbar(content: toolbar)
|
|
|
|
.sheet(item: $modalType, content: presentedModal)
|
|
|
|
.onAppear(perform: loadProfileIfNeeded)
|
|
|
|
}
|
|
|
|
|
|
|
|
private var title: String {
|
|
|
|
!isDeleted ? header.name : ""
|
|
|
|
}
|
|
|
|
|
|
|
|
@ViewBuilder
|
|
|
|
private var mainView: some View {
|
|
|
|
VPNSection(currentProfile: profileManager.currentProfile)
|
|
|
|
ProviderSection(currentProfile: profileManager.currentProfile)
|
|
|
|
ConfigurationSection(
|
|
|
|
currentProfile: profileManager.currentProfile,
|
|
|
|
modalType: $modalType
|
|
|
|
)
|
|
|
|
ExtraSection(currentProfile: profileManager.currentProfile)
|
|
|
|
DiagnosticsSection(currentProfile: profileManager.currentProfile)
|
2022-04-18 10:01:42 +00:00
|
|
|
UninstallVPNSection()
|
2022-04-12 13:09:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private var welcomeView: some View {
|
|
|
|
WelcomeView()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func toolbar() -> some ToolbarContent {
|
|
|
|
ToolbarItemGroup(placement: .navigationBarTrailing) {
|
|
|
|
if !isDeleted {
|
2022-04-18 14:42:17 +00:00
|
|
|
MenuBar(
|
|
|
|
currentProfile: profileManager.currentProfile,
|
|
|
|
modalType: $modalType
|
|
|
|
)
|
2022-04-12 13:09:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@ViewBuilder
|
|
|
|
private func presentedModal(_ modalType: ModalType) -> some View {
|
|
|
|
switch modalType {
|
2022-04-18 14:42:17 +00:00
|
|
|
case .shortcuts:
|
|
|
|
NavigationView {
|
|
|
|
ShortcutsView(target: profileManager.currentProfile.value)
|
|
|
|
}.themeGlobal()
|
|
|
|
|
2022-04-12 13:09:14 +00:00
|
|
|
case .rename:
|
|
|
|
NavigationView {
|
|
|
|
RenameView(currentProfile: profileManager.currentProfile)
|
|
|
|
}.themeGlobal()
|
|
|
|
|
2022-04-18 14:42:17 +00:00
|
|
|
case .paywallShortcuts:
|
|
|
|
NavigationView {
|
|
|
|
PaywallView(
|
|
|
|
modalType: $modalType,
|
|
|
|
feature: .siriShortcuts
|
|
|
|
)
|
|
|
|
}.themeGlobal()
|
|
|
|
|
2022-04-13 18:12:25 +00:00
|
|
|
case .paywallNetworkSettings:
|
|
|
|
NavigationView {
|
|
|
|
PaywallView(
|
|
|
|
modalType: $modalType,
|
|
|
|
feature: .networkSettings
|
|
|
|
)
|
|
|
|
}.themeGlobal()
|
|
|
|
|
2022-04-12 13:09:14 +00:00
|
|
|
case .paywallTrustedNetworks:
|
|
|
|
NavigationView {
|
|
|
|
PaywallView(
|
2022-04-13 19:01:06 +00:00
|
|
|
modalType: $modalType,
|
2022-04-12 13:09:14 +00:00
|
|
|
feature: .trustedNetworks
|
|
|
|
)
|
|
|
|
}.themeGlobal()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private var loadingSection: some View {
|
|
|
|
Section(
|
|
|
|
header: Text(Unlocalized.VPN.vpn)
|
|
|
|
) {
|
|
|
|
ProgressView()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func loadProfileIfNeeded() {
|
|
|
|
guard !isLoaded else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
guard !header.isPlaceholder else {
|
|
|
|
pp_log.debug("ProfileView is a placeholder for WelcomeView, no active profile")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
do {
|
|
|
|
let result = try profileManager.loadCurrentProfile(withId: header.id)
|
|
|
|
if result.isReady {
|
|
|
|
isLoaded = true
|
|
|
|
return
|
|
|
|
}
|
|
|
|
Task {
|
|
|
|
do {
|
|
|
|
try await profileManager.makeProfileReady(result.profile)
|
|
|
|
withAnimation {
|
|
|
|
isLoaded = true
|
|
|
|
}
|
|
|
|
} catch {
|
|
|
|
pp_log.error("Profile \(header.id) could not be made ready: \(error)")
|
|
|
|
presentationMode.wrappedValue.dismiss()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch {
|
|
|
|
pp_log.error("Profile \(header.id) could not be loaded: \(error)")
|
|
|
|
presentationMode.wrappedValue.dismiss()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func presentPaywallTrustedNetworks() {
|
|
|
|
modalType = .paywallTrustedNetworks
|
|
|
|
}
|
|
|
|
}
|