passepartout-apple/Passepartout/App/Views/ProfileView+MainMenu.swift

266 lines
7.6 KiB
Swift
Raw Normal View History

//
// ProfileView+MainMenu.swift
// Passepartout
//
// Created by Davide De Rosa on 2/6/22.
2023-03-17 15:56:19 +00:00
// 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 <http://www.gnu.org/licenses/>.
//
2022-06-23 21:31:01 +00:00
import PassepartoutLibrary
import SwiftUI
extension ProfileView {
struct MainMenu: View {
enum AlertType: Int, Identifiable {
case uninstallVPN
2023-03-17 20:55:47 +00:00
case deleteProfile
2023-03-17 20:55:47 +00:00
var id: Int {
rawValue
}
}
2023-03-17 20:55:47 +00:00
@ObservedObject private var profileManager: ProfileManager
2023-03-17 20:55:47 +00:00
@ObservedObject private var vpnManager: VPNManager
2023-03-17 20:55:47 +00:00
@ObservedObject private var currentVPNState: ObservableVPNState
2023-03-17 20:55:47 +00:00
@ObservedObject private var currentProfile: ObservableProfile
2023-03-17 20:55:47 +00:00
private var header: Profile.Header {
currentProfile.value.header
}
2023-03-17 20:55:47 +00:00
@Binding private var modalType: ModalType?
2023-03-17 20:55:47 +00:00
@State private var isAlertPresented = false
@State private var alertType: AlertType?
2023-03-17 20:55:47 +00:00
private let uninstallVPNTitle = L10n.Global.Strings.uninstall
2023-03-17 20:55:47 +00:00
private let deleteProfileTitle = L10n.Global.Strings.delete
2023-03-17 20:55:47 +00:00
init(currentProfile: ObservableProfile, modalType: Binding<ModalType?>) {
profileManager = .shared
vpnManager = .shared
currentVPNState = .shared
self.currentProfile = currentProfile
_modalType = modalType
}
var body: some View {
mainView
.alert(
Text(Unlocalized.appName),
isPresented: $isAlertPresented,
presenting: alertType,
actions: alertActions,
message: alertMessage
)
}
private var mainView: some View {
Menu {
ReconnectButton()
ShortcutsButton(
modalType: $modalType
)
2022-04-27 14:49:23 +00:00
Divider()
RenameButton(
modalType: $modalType
)
DuplicateButton(
header: header,
setAsCurrent: true
)
uninstallVPNButton
Divider()
deleteProfileButton
} label: {
themeSettingsMenuImage.asSystemImage
}
}
2023-03-17 20:55:47 +00:00
private func alertActions(_ alertType: AlertType) -> some View {
switch alertType {
case .uninstallVPN:
return Group {
Button(role: .destructive, action: uninstallVPN) {
Text(uninstallVPNTitle)
}
Button(role: .cancel) {
} label: {
Text(L10n.Global.Strings.cancel)
}
}
case .deleteProfile:
return Group {
Button(role: .destructive, action: removeProfile) {
Text(deleteProfileTitle)
}
Button(role: .cancel) {
} label: {
Text(L10n.Global.Strings.cancel)
}
}
}
}
private func alertMessage(_ alertType: AlertType) -> some View {
switch alertType {
case .uninstallVPN:
return Text(L10n.Profile.Alerts.UninstallVpn.message)
case .deleteProfile:
return Text(L10n.Organizer.Alerts.RemoveProfile.message(header.name))
}
}
private var uninstallVPNButton: some View {
Button {
alertType = .uninstallVPN
isAlertPresented = true
} label: {
Label(uninstallVPNTitle, systemImage: themeUninstallImage)
}
}
2023-03-17 20:55:47 +00:00
private var deleteProfileButton: some View {
DestructiveButton {
alertType = .deleteProfile
isAlertPresented = true
} 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
2023-03-17 20:55:47 +00:00
init() {
vpnManager = .shared
}
2023-03-17 20:55:47 +00:00
var body: some View {
Button {
Task { @MainActor in
await vpnManager.reconnect()
}
} label: {
Label(L10n.Global.Strings.reconnect, systemImage: themeReconnectImage)
}
}
}
2023-03-17 20:55:47 +00:00
struct ShortcutsButton: View {
@ObservedObject private var productManager: ProductManager
2023-03-17 20:55:47 +00:00
@Binding private var modalType: ModalType?
2023-03-17 20:55:47 +00:00
init(modalType: Binding<ModalType?>) {
productManager = .shared
_modalType = modalType
}
private var isEligibleForSiri: Bool {
productManager.isEligible(forFeature: .siriShortcuts)
}
2023-03-17 20:55:47 +00:00
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
}
}
}
2023-03-17 20:55:47 +00:00
2022-04-27 14:49:23 +00:00
struct RenameButton: View {
@Binding private var modalType: ModalType?
2023-03-17 20:55:47 +00:00
2022-04-27 14:49:23 +00:00
init(modalType: Binding<ModalType?>) {
_modalType = modalType
}
2023-03-17 20:55:47 +00:00
2022-04-27 14:49:23 +00:00
var body: some View {
Button {
modalType = .rename
} label: {
Label(L10n.Global.Strings.rename, systemImage: themeRenameProfileImage)
}
}
}
struct DuplicateButton: View {
@ObservedObject private var profileManager: ProfileManager
2023-03-17 20:55:47 +00:00
private let header: Profile.Header
2023-03-17 20:55:47 +00:00
private let setAsCurrent: Bool
2023-03-17 20:55:47 +00:00
init(header: Profile.Header, setAsCurrent: Bool) {
profileManager = .shared
self.header = header
self.setAsCurrent = setAsCurrent
}
2023-03-17 20:55:47 +00:00
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)
}
}
}