// // ProfileView+MainMenu.swift // Passepartout // // Created by Davide De Rosa on 2/6/22. // Copyright (c) 2023 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 PassepartoutLibrary import SwiftUI extension ProfileView { struct MainMenu: View { enum ActionSheetType: Int, Identifiable { case uninstallVPN case deleteProfile var id: Int { rawValue } } @ObservedObject private var profileManager: ProfileManager @ObservedObject private var vpnManager: VPNManager @ObservedObject private var currentVPNState: ObservableVPNState @ObservedObject private var currentProfile: ObservableProfile private var header: Profile.Header { currentProfile.value.header } @Binding private var modalType: ModalType? @State private var actionSheetType: ActionSheetType? private let uninstallVPNTitle = L10n.Global.Strings.uninstall private let deleteProfileTitle = L10n.Global.Strings.delete private let cancelTitle = L10n.Global.Strings.cancel init(currentProfile: ObservableProfile, modalType: Binding) { profileManager = .shared vpnManager = .shared currentVPNState = .shared self.currentProfile = currentProfile _modalType = modalType } var body: some View { mainView .alert(item: $actionSheetType) { switch $0 { case .uninstallVPN: return Alert( title: Text(uninstallVPNTitle), message: Text(L10n.Profile.Alerts.UninstallVpn.message), primaryButton: .destructive(Text(uninstallVPNTitle), action: uninstallVPN), secondaryButton: .cancel(Text(cancelTitle)) ) case .deleteProfile: return Alert( title: Text(deleteProfileTitle), message: Text(L10n.Organizer.Alerts.RemoveProfile.message(header.name)), primaryButton: .destructive(Text(deleteProfileTitle), action: removeProfile), secondaryButton: .cancel(Text(cancelTitle)) ) } } } private var mainView: some View { Menu { ReconnectButton() ShortcutsButton( modalType: $modalType ) Divider() RenameButton( modalType: $modalType ) DuplicateButton( header: header, setAsCurrent: true ) uninstallVPNButton Divider() deleteProfileButton } label: { themeSettingsMenuImage.asSystemImage } } private var uninstallVPNButton: some View { Button { actionSheetType = .uninstallVPN } label: { Label(uninstallVPNTitle, systemImage: themeUninstallImage) } } private var deleteProfileButton: some View { DestructiveButton { actionSheetType = .deleteProfile } label: { Label(deleteProfileTitle, systemImage: themeDeleteImage) } } private func uninstallVPN() { Task { @MainActor in await vpnManager.uninstall() } } private func removeProfile() { withAnimation { profileManager.removeProfiles(withIds: [header.id]) } } } } extension ProfileView { struct ReconnectButton: View { @ObservedObject private var vpnManager: VPNManager init() { vpnManager = .shared } var body: some View { Button { Task { @MainActor in await vpnManager.reconnect() } } label: { Label(L10n.Global.Strings.reconnect, systemImage: themeReconnectImage) } } } struct ShortcutsButton: View { @ObservedObject private var productManager: ProductManager @Binding private var modalType: ModalType? init(modalType: Binding) { productManager = .shared _modalType = modalType } private var isEligibleForSiri: Bool { productManager.isEligible(forFeature: .siriShortcuts) } var body: some View { Button { presentShortcutsOrPaywall() } label: { Label(Unlocalized.Other.siri, systemImage: themeShortcutsImage) } } private func presentShortcutsOrPaywall() { // eligibility: enter Siri shortcuts or present paywall if isEligibleForSiri { modalType = .shortcuts } else { modalType = .paywallShortcuts } } } struct RenameButton: View { @Binding private var modalType: ModalType? init(modalType: Binding) { _modalType = modalType } var body: some View { Button { modalType = .rename } label: { Label(L10n.Global.Strings.rename, systemImage: themeRenameProfileImage) } } } struct DuplicateButton: View { @ObservedObject private var profileManager: ProfileManager private let header: Profile.Header private let setAsCurrent: Bool init(header: Profile.Header, setAsCurrent: Bool) { profileManager = .shared self.header = header self.setAsCurrent = setAsCurrent } var body: some View { Button { duplicateProfile(withId: header.id) } label: { Label(L10n.Global.Strings.duplicate, systemImage: themeDuplicateImage) } } private func duplicateProfile(withId id: UUID) { profileManager.duplicateProfile(withId: id, setAsCurrent: setAsCurrent) } } }