mirror of
https://github.com/passepartoutvpn/passepartout-apple.git
synced 2024-12-25 19:02:37 +00:00
c0cc10ab94
* Add "Reconnect" in profile view * Add "Reconnect" in profile context menu * Update CHANGELOG * Restrict "Reconnect" in context menu to iOS 16 SwiftUI does not react properly to state updates.
246 lines
7.5 KiB
Swift
246 lines
7.5 KiB
Swift
//
|
|
// ProfileView+MainMenu.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 PassepartoutLibrary
|
|
|
|
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<ModalType?>) {
|
|
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 {
|
|
if isActiveProfileNotDisconnected {
|
|
ReconnectButton()
|
|
}
|
|
ShortcutsButton(
|
|
modalType: $modalType
|
|
)
|
|
Divider()
|
|
RenameButton(
|
|
modalType: $modalType
|
|
)
|
|
DuplicateButton(
|
|
header: header,
|
|
setAsCurrent: true
|
|
)
|
|
uninstallVPNButton
|
|
Divider()
|
|
deleteProfileButton
|
|
} label: {
|
|
themeSettingsMenuImage.asSystemImage
|
|
}
|
|
}
|
|
|
|
private var isActiveProfileNotDisconnected: Bool {
|
|
profileManager.isActiveProfile(header.id) && currentVPNState.vpnStatus != .disconnected
|
|
}
|
|
|
|
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<ModalType?>) {
|
|
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 = 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)
|
|
}
|
|
}
|
|
}
|