// // ProfileContextMenu.swift // Passepartout // // Created by Davide De Rosa on 9/3/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 struct ProfileContextMenu: View, Routable { let profileManager: ProfileManager let tunnel: ExtendedTunnel let preview: ProfilePreview let interactiveManager: InteractiveManager let errorHandler: ErrorHandler let isInstalledProfile: Bool var flow: ProfileFlow? var body: some View { tunnelToggleButton providerConnectToButton if isInstalledProfile { tunnelRestartButton } Divider() profileEditButton profileDuplicateButton Divider() profileRemoveButton } } @MainActor private extension ProfileContextMenu { var profile: Profile? { profileManager.profile(withId: preview.id) } var tunnelToggleButton: some View { TunnelToggleButton( tunnel: tunnel, profile: profile, nextProfileId: .constant(nil), interactiveManager: interactiveManager, errorHandler: errorHandler, onProviderEntityRequired: { flow?.onProviderEntityRequired($0) }, onPurchaseRequired: { flow?.onPurchaseRequired($0) }, label: { ThemeImageLabel( $0 ? Strings.Global.Actions.enable : Strings.Global.Actions.disable, $0 ? .tunnelEnable : .tunnelDisable ) } ) } var providerConnectToButton: some View { profile? .selectedProvider .map { _ in Button(Strings.Views.App.ProfileContext.connectTo) { flow?.onProviderEntityRequired(profile!) } } } var tunnelRestartButton: some View { TunnelRestartButton( tunnel: tunnel, profile: profile, errorHandler: errorHandler, onPurchaseRequired: { flow?.onPurchaseRequired($0) }, label: { ThemeImageLabel(Strings.Global.Actions.restart, .tunnelRestart) } ) } var profileEditButton: some View { Button { flow?.onEditProfile(preview) } label: { ThemeImageLabel(Strings.Global.Actions.edit.withTrailingDots, .profileEdit) } } var profileDuplicateButton: some View { ProfileDuplicateButton( profileManager: profileManager, preview: preview, errorHandler: errorHandler ) { ThemeImageLabel(Strings.Global.Actions.duplicate, .contextDuplicate) } } var profileRemoveButton: some View { ProfileRemoveButton( profileManager: profileManager, preview: preview ) { ThemeImageLabel(Strings.Global.Actions.remove, .contextRemove) } } } #Preview { List { Menu("Menu") { ProfileContextMenu( profileManager: .mock, tunnel: .mock, preview: .init(.mock), interactiveManager: InteractiveManager(), errorHandler: .default(), isInstalledProfile: true ) } } .withMockEnvironment() }