// // ProfileListView.swift // Passepartout // // Created by Davide De Rosa on 11/1/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 <http://www.gnu.org/licenses/>. // import CommonLibrary import CommonUtils import PassepartoutKit import SwiftUI struct ProfileListView: View { @ObservedObject var profileManager: ProfileManager @ObservedObject var tunnel: ExtendedTunnel @FocusState.Binding var focusedField: ProfileView.Field? @ObservedObject var interactiveManager: InteractiveManager @ObservedObject var errorHandler: ErrorHandler var flow: AppFlow? var body: some View { VStack { headerView .frame(maxWidth: .infinity, alignment: .leading) List { ForEach(previews, id: \.id, content: toggleButton(for:)) } .themeList() .themeProgress(if: false, isEmpty: !profileManager.hasProfiles) { Text(Strings.Views.App.Folders.noProfiles) .themeEmptyMessage() } } } } private extension ProfileListView { var previews: [ProfilePreview] { profileManager.previews } var headerView: some View { Text(Strings.Views.App.Tv.header(Strings.Unlocalized.appName, Strings.Unlocalized.appleTV)) .textCase(.none) .foregroundStyle(.primary) .font(.body) } func toggleButton(for preview: ProfilePreview) -> some View { TunnelToggleButton( tunnel: tunnel, profile: profileManager.profile(withId: preview.id), nextProfileId: .constant(nil), interactiveManager: interactiveManager, errorHandler: errorHandler, onProviderEntityRequired: onProviderEntityRequired, onPurchaseRequired: onPurchaseRequired, label: { _ in toggleView(for: preview) } ) .focused($focusedField, equals: .profile(preview.id)) } func toggleView(for preview: ProfilePreview) -> some View { HStack { Text(preview.name) Spacer() ThemeImage(tunnel.statusImageName) .opaque(preview.id == tunnel.currentProfile?.id) } .font(.headline) } } private extension ProfileListView { func onProviderEntityRequired(_ profile: Profile) { flow?.onProviderEntityRequired(profile) } func onPurchaseRequired(_ features: Set<AppFeature>) { flow?.onPurchaseRequired(features) } } // MARK: - Previews #Preview("List") { ContentPreview(profileManager: .mock) } #Preview("Empty") { ContentPreview(profileManager: ProfileManager(profiles: [])) } private struct ContentPreview: View { let profileManager: ProfileManager @FocusState var focusedField: ProfileView.Field? var body: some View { ProfileListView( profileManager: profileManager, tunnel: .mock, focusedField: $focusedField, interactiveManager: InteractiveManager(), errorHandler: .default() ) .withMockEnvironment() } }