// // DonationViewController.swift // Passepartout // // Created by Davide De Rosa on 4/6/19. // Copyright (c) 2021 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 UIKit import StoreKit import SwiftyBeaver import PassepartoutConstants import PassepartoutCore import Convenience import ConvenienceUI private let log = SwiftyBeaver.self class DonationViewController: UITableViewController, StrongTableHost { private var donationList: [LocalProduct] = [] private var productsByIdentifier: [String: SKProduct] = [:] private var isLoading = true private var isPurchasing = false private func setProducts(_ products: [SKProduct]) { for p in products { productsByIdentifier[p.productIdentifier] = p } reloadModel() tableView.reloadData() } // MARK: StrongTableModel var model: StrongTableModel = StrongTableModel() func reloadModel() { donationList = [] model.clear() model.add(.oneTime) model.setHeader(L10n.Donation.Sections.OneTime.header, forSection: .oneTime) model.setFooter(L10n.Donation.Sections.OneTime.footer, forSection: .oneTime) guard !isLoading else { model.set([.loading], forSection: .oneTime) return } donationList.append(contentsOf: LocalProduct.allDonations.filter { productsByIdentifier[$0.rawValue] != nil }) model.set(.donation, count: donationList.count, forSection: .oneTime) if isPurchasing { model.add(.activity) model.set([.purchasing], forSection: .activity) } } // MARK: UIViewController override func viewDidLoad() { super.viewDidLoad() title = L10n.Donation.title reloadModel() ProductManager.shared.listProducts { self.isLoading = false guard let products = $0 else { log.error("Unable to list products: \($1?.localizedDescription ?? "")") return } self.setProducts(products) } } @IBAction private func close() { dismiss(animated: true, completion: nil) } // MARK: UITableViewController override func numberOfSections(in tableView: UITableView) -> Int { return model.numberOfSections } override func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? { return model.header(forSection: section) } override func tableView(_ tableView: UITableView, titleForFooterInSection section: Int) -> String? { return model.footer(forSection: section) } override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { return model.numberOfRows(forSection: section) } override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { switch model.row(at: indexPath) { case .loading: let cell = Cells.activity.dequeue(from: tableView, for: indexPath) cell.textLabel?.text = L10n.Donation.Cells.Loading.caption return cell case .purchasing: let cell = Cells.activity.dequeue(from: tableView, for: indexPath) cell.textLabel?.text = L10n.Donation.Cells.Purchasing.caption return cell case .donation: let productId = productIdentifier(at: indexPath) guard let product = productsByIdentifier[productId] else { fatalError("Row with no associated product") } let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.leftText = product.localizedTitle cell.rightText = product.localizedPrice cell.isTappable = !isPurchasing return cell } } override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { switch model.row(at: indexPath) { case .donation: guard !isPurchasing else { return } tableView.deselectRow(at: indexPath, animated: true) let productId = productIdentifier(at: indexPath) guard let product = productsByIdentifier[productId] else { fatalError("Row with no associated product") } isPurchasing = true reloadModel() tableView.reloadData() ProductManager.shared.purchase(product) { self.handlePurchase(result: $0, error: $1) } default: break } } private func handlePurchase(result: InAppPurchaseResult, error: Error?) { let alert: UIAlertController switch result { case .cancelled: isPurchasing = false reloadModel() tableView.reloadData() return case .success: alert = UIAlertController.asAlert(L10n.Donation.Alerts.Purchase.Success.title, L10n.Donation.Alerts.Purchase.Success.message) case .failure: alert = UIAlertController.asAlert(title, L10n.Donation.Alerts.Purchase.Failure.message(error?.localizedDescription ?? "")) } alert.addCancelAction(L10n.Global.ok) { self.isPurchasing = false self.reloadModel() self.tableView.reloadData() } present(alert, animated: true) } } extension DonationViewController { enum SectionType { case oneTime case activity } enum RowType { case loading case purchasing case donation } private func productIdentifier(at indexPath: IndexPath) -> String { guard model.row(at: indexPath) == .donation else { fatalError("Not a donation row") } return donationList[indexPath.row].rawValue } }