// // PurchasedView.swift // Passepartout // // Created by Davide De Rosa on 11/25/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 . // import CommonLibrary import CommonUtils import SwiftUI public struct PurchasedView: View { @EnvironmentObject private var iapManager: IAPManager @State private var isLoading = true @State private var products: [InAppProduct] = [] @StateObject private var errorHandler: ErrorHandler = .default() public init() { } public var body: some View { contentView .withErrorHandler(errorHandler) .themeProgress(if: isLoading) .themeAnimation(on: isLoading, category: .diagnostics) .onLoad { Task { do { products = try await iapManager .purchasableProducts(for: Array(iapManager.purchasedProducts)) .sorted { $0.localizedTitle < $1.localizedTitle } isLoading = false } catch { errorHandler.handle(error) isLoading = false } } } } } private extension PurchasedView { var isEmpty: Bool { iapManager.purchasedAppBuild == nil && iapManager.purchasedProducts.isEmpty && iapManager.eligibleFeatures.isEmpty } var allFeatures: [AppFeature] { AppFeature.allCases.sorted() } } private extension PurchasedView { var contentView: some View { #if os(macOS) Form(content: sectionsGroup) .themeForm() #else List(content: sectionsGroup) #endif } func sectionsGroup() -> some View { Group { downloadSection productsSection featuresSection } } var downloadSection: some View { iapManager.purchasedAppBuild.map { build in Group { Text(Strings.Views.Purchased.Rows.buildNumber) .themeTrailingValue(build.description) .scrollableOnTV() } .themeSection(header: Strings.Views.Purchased.Sections.Download.header) } } var productsSection: some View { Group { if !products.isEmpty { ForEach(products, id: \.productIdentifier) { Text($0.localizedTitle) .themeTrailingValue($0.localizedPrice) .scrollableOnTV() } } else { Text(Strings.Views.Purchased.noPurchases) } } .themeSection(header: Strings.Views.Purchased.Sections.Products.header) } var featuresSection: some View { Group { ForEach(allFeatures, id: \.self) { feature in HStack { Text(feature.localizedDescription) Spacer() ThemeImage(iapManager.isEligible(for: feature) ? .marked : .close) } .scrollableOnTV() } } .themeSection(header: Strings.Views.Purchased.Sections.Features.header) } } #Preview { PurchasedView() .withMockEnvironment() }