// // AppCoordinator.swift // Passepartout // // Created by Davide De Rosa on 8/13/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 . // import CommonLibrary import CommonUtils import PassepartoutKit import SwiftUI public struct AppCoordinator: View, AppCoordinatorConforming { @AppStorage(AppPreference.profilesLayout.key) private var layout: ProfilesLayout = .list private let profileManager: ProfileManager private let tunnel: ExtendedTunnel private let registry: Registry @StateObject private var profileEditor = ProfileEditor() @State private var modalRoute: ModalRoute? @State private var isImporting = false @State private var profilePath = NavigationPath() @State private var migrationPath = NavigationPath() @State private var paywallReason: PaywallReason? @StateObject private var errorHandler: ErrorHandler = .default() public init( profileManager: ProfileManager, tunnel: ExtendedTunnel, registry: Registry ) { self.profileManager = profileManager self.tunnel = tunnel self.registry = registry } public var body: some View { NavigationStack { contentView .toolbar(content: toolbarContent) } .modifier(PaywallModifier(reason: $paywallReason)) .themeModal( item: $modalRoute, size: modalRoute?.size ?? .large, isFixedWidth: modalRoute?.isFixedWidth ?? false, isFixedHeight: modalRoute?.isFixedHeight ?? false, isInteractive: modalRoute?.isInteractive ?? true, content: modalDestination ) } } // MARK: - Destinations extension AppCoordinator { enum ModalRoute: Identifiable { case about case editProfile case editProviderEntity(Profile, Module, SerializedProvider) case migrateProfiles case settings var id: Int { switch self { case .about: return 1 case .editProfile: return 2 case .editProviderEntity: return 3 case .migrateProfiles: return 4 case .settings: return 5 } } var size: ThemeModalSize { switch self { case .migrateProfiles: return .custom(width: 700, height: 400) default: return .large } } var isFixedWidth: Bool { switch self { case .migrateProfiles: return true default: return false } } var isFixedHeight: Bool { switch self { case .migrateProfiles: return true default: return false } } var isInteractive: Bool { switch self { case .editProfile: return false default: return true } } } var contentView: some View { ProfileContainerView( layout: layout, profileManager: profileManager, tunnel: tunnel, registry: registry, isImporting: $isImporting, errorHandler: errorHandler, flow: .init( onEditProfile: { guard let profile = profileManager.profile(withId: $0.id) else { return } enterDetail(of: profile) }, onEditProviderEntity: { guard let pair = $0.firstProviderModuleWithMetadata else { return } present(.editProviderEntity($0, pair.0, pair.1)) }, onMigrateProfiles: { modalRoute = .migrateProfiles }, onPurchaseRequired: { features in setLater(.purchase(features)) { paywallReason = $0 } } ) ) } func toolbarContent() -> some ToolbarContent { AppToolbar( profileManager: profileManager, layout: $layout, isImporting: $isImporting, onSettings: { present(.settings) }, onAbout: { present(.about) }, onMigrateProfiles: { present(.migrateProfiles) }, onNewProfile: enterDetail ) } @ViewBuilder func modalDestination(for item: ModalRoute?) -> some View { switch item { case .about: AboutRouterView( profileManager: profileManager, tunnel: tunnel ) case .editProfile: ProfileCoordinator( profileManager: profileManager, profileEditor: profileEditor, registry: registry, moduleViewFactory: DefaultModuleViewFactory(registry: registry), modally: true, path: $profilePath, onDismiss: onDismiss ) case .editProviderEntity(let profile, let module, let provider): ProviderEntitySelector( profileManager: profileManager, tunnel: tunnel, profile: profile, module: module, provider: provider, errorHandler: errorHandler ) case .migrateProfiles: MigrateView( style: migrateViewStyle, profileManager: profileManager ) .themeNavigationStack(closable: true, path: $migrationPath) case .settings: SettingsView(profileManager: profileManager) default: EmptyView() } } var migrateViewStyle: MigrateView.Style { #if os(iOS) .list #else .table #endif } func enterDetail(of profile: Profile) { profilePath = NavigationPath() let isShared = profileManager.isRemotelyShared(profileWithId: profile.id) profileEditor.editProfile(profile, isShared: isShared) present(.editProfile) } func onDismiss() { present(nil) } func present(_ route: ModalRoute?) { // XXX: this is a workaround for #791 on iOS 16 setLater(route) { modalRoute = $0 } } } #Preview { AppCoordinator( profileManager: .mock, tunnel: .mock, registry: Registry() ) .withMockEnvironment() }