// // AddProviderView.swift // Passepartout // // Created by Davide De Rosa on 2/10/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 . // import SwiftUI import PassepartoutCore struct AddProviderView: View { @ObservedObject private var providerManager: ProviderManager @ObservedObject private var productManager: ProductManager private let bindings: AddProfileView.Bindings @StateObject private var viewModel = ViewModel() init(bindings: AddProfileView.Bindings) { providerManager = .shared productManager = .shared self.bindings = bindings } private var providers: [ProviderMetadata] { providerManager.allProviders() .filter { $0.supportedVPNProtocols.contains(viewModel.selectedVPNProtocol) }.sorted() } private var availableVPNProtocols: [VPNProtocolType] { var protos: Set = [] providers.forEach { $0.supportedVPNProtocols.forEach { protos.insert($0) } } return protos.sorted() } private func isFetchingProvider(_ name: ProviderName) -> Bool { if case .provider(name) = viewModel.pendingOperation { return true } return false } private var isUpdatingIndex: Bool { if case .index = viewModel.pendingOperation { return true } return false } var body: some View { ZStack { ForEach(providers, id: \.navigationId, content: hiddenProviderLink) ScrollViewReader { scrollProxy in List { mainSection if !providers.isEmpty { providersSection } }.onChange(of: viewModel.errorMessage) { onErrorMessage($0, scrollProxy) }.disabled(viewModel.pendingOperation != nil) .themeAnimation(on: providers) } }.toolbar { themeCloseItem(isPresented: bindings.$isPresented) }.sheet(isPresented: $viewModel.isPaywallPresented) { NavigationView { PaywallView(isPresented: $viewModel.isPaywallPresented) }.themeGlobal() }.navigationTitle(L10n.AddProfile.Shared.title) .themeSecondaryView() } private var mainSection: some View { Section( footer: Text(L10n.AddProfile.Provider.Sections.Vpn.footer) ) { let protos = availableVPNProtocols if !protos.isEmpty { themeTextPicker( L10n.Global.Strings.protocol, selection: $viewModel.selectedVPNProtocol, values: protos, description: \.description ) } updateListButton } } private var providersSection: some View { Section( footer: themeErrorMessage(viewModel.errorMessage) ) { ForEach(providers, content: providerRow) } } private func providerRow(_ metadata: ProviderMetadata) -> some View { Button { presentOrPurchaseProvider(metadata) } label: { Label(metadata.description, image: themeAssetsProviderImage(metadata.name)) }.withTrailingProgress(when: isFetchingProvider(metadata.name)) } private func hiddenProviderLink(_ metadata: ProviderMetadata) -> some View { NavigationLink("", tag: metadata, selection: $viewModel.selectedProvider) { NameView( profile: $viewModel.pendingProfile, providerMetadata: metadata, bindings: bindings ) } } private var updateListButton: some View { Button(L10n.AddProfile.Provider.Items.updateList) { viewModel.updateIndex(providerManager) }.withTrailingProgress(when: isUpdatingIndex) } // eligibility: select or purchase provider private func presentOrPurchaseProvider(_ metadata: ProviderMetadata) { if productManager.isEligible(forProvider: metadata.name) { viewModel.selectProvider(metadata, providerManager) } else { viewModel.presentPaywall() } } private func onErrorMessage(_ message: String?, _ scrollProxy: ScrollViewProxy) { guard let _ = message else { return } scrollToErrorMessage(scrollProxy) } } extension AddProviderView { private func scrollToErrorMessage(_ proxy: ScrollViewProxy) { proxy.maybeScrollTo(providers.last?.id, animated: true) } } private extension ProviderMetadata { var navigationId: String { return "navigation.\(name)" } }