// // OrganizerViewController.swift // Passepartout-iOS // // Created by Davide De Rosa on 9/2/18. // Copyright (c) 2020 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 MessageUI import PassepartoutCore import Convenience import SystemConfiguration.CaptiveNetwork // XXX: convoluted due to the separation of provider/host profiles class OrganizerViewController: UITableViewController, StrongTableHost { private let service = TransientStore.shared.service private var providers: [String] = [] private var hosts: [String] = [] private var didShowSubreddit = false // MARK: StrongTableHost let model: StrongTableModel = StrongTableModel() func reloadModel() { model.clear() model.add(.vpn) model.add(.providers) model.add(.hosts) if #available(iOS 12, *) { model.add(.siri) } model.add(.support) if ProductManager.shared.isEligibleForFeedback() { model.add(.feedback) model.setHeader(L10n.Core.Organizer.Sections.Feedback.header, forSection: .feedback) model.set([.writeReview], forSection: .feedback) } model.add(.about) model.add(.destruction) model.setHeader(L10n.App.Service.Sections.Vpn.header, forSection: .vpn) model.setHeader(L10n.Core.Organizer.Sections.Providers.header, forSection: .providers) model.setHeader(L10n.Core.Organizer.Sections.Hosts.header, forSection: .hosts) model.setFooter(L10n.Core.Organizer.Sections.Providers.footer, forSection: .providers) model.setFooter(L10n.Core.Organizer.Sections.Hosts.footer, forSection: .hosts) if #available(iOS 12, *) { model.setHeader(L10n.Core.Organizer.Sections.Siri.header, forSection: .siri) model.setFooter(L10n.Core.Organizer.Sections.Siri.footer, forSection: .siri) model.set([.siriShortcuts], forSection: .siri) } model.setHeader(L10n.Core.Organizer.Sections.Support.header, forSection: .support) model.set([.connectionStatus], forSection: .vpn) model.set([.donate, .joinCommunity], forSection: .support) model.set([.openAbout], forSection: .about) model.set([.uninstall], forSection: .destruction) if AppConstants.Flags.isBeta { model.add(.test) model.setHeader("Beta", forSection: .test) model.set([.testInterfaces, .testDisplayLog, .testTermination], forSection: .test) } // providers = service.ids(forContext: .provider).sorted() hosts = service.ids(forContext: .host).sortedCaseInsensitive() var providerRows = [RowType](repeating: .profile, count: providers.count) var hostRows = [RowType](repeating: .profile, count: hosts.count) providerRows.append(.addProvider) hostRows.append(.addHost) model.set(providerRows, forSection: .providers) model.set(hostRows, forSection: .hosts) } // MARK: UIViewController deinit { NotificationCenter.default.removeObserver(self) } override func viewDidLoad() { super.viewDidLoad() title = GroupConstants.App.title navigationItem.rightBarButtonItem = editButtonItem Cells.destructive.register(with: tableView) reloadModel() tableView.reloadData() // XXX: if split vc is collapsed when a profile is in use, this vc // is not loaded on app launch. consequentially, service.delegate remains // nil until the Organizer is entered // // see UISplitViewControllerDelegate in AppDelegate (collapse is now commented out) service.delegate = self let nc = NotificationCenter.default nc.addObserver(self, selector: #selector(vpnDidUpdate), name: .VPNDidChangeStatus, object: nil) nc.addObserver(self, selector: #selector(productManagerDidReloadReceipt), name: ProductManager.didReloadReceipt, object: nil) } override func viewDidAppear(_ animated: Bool) { super.viewDidAppear(animated) if !didShowSubreddit && !TransientStore.didHandleSubreddit { didShowSubreddit = true let alert = UIAlertController.asAlert(L10n.Core.Reddit.title, L10n.Core.Reddit.message) alert.addPreferredAction(L10n.Core.Reddit.Buttons.subscribe) { TransientStore.didHandleSubreddit = true self.subscribeSubreddit() } alert.addAction(L10n.Core.Reddit.Buttons.never) { TransientStore.didHandleSubreddit = true } alert.addCancelAction(L10n.Core.Reddit.Buttons.remind) present(alert, animated: true, completion: nil) } } override func shouldPerformSegue(withIdentifier identifier: String, sender: Any?) -> Bool { if let cell = sender as? UITableViewCell, let indexPath = tableView.indexPath(for: cell) { return model.row(at: indexPath) == .profile } // fall back to active profile if no selection return service.hasActiveProfile() } override func prepare(for segue: UIStoryboardSegue, sender: Any?) { let destination = (segue.destination as? UINavigationController)?.topViewController if let vc = destination as? ServiceViewController { var selectedProfile: ConnectionProfile? // XXX: sender can be a cell or a profile selectedProfile = sender as? ConnectionProfile if selectedProfile == nil, let cell = sender as? UITableViewCell, let indexPath = tableView.indexPath(for: cell) { selectedProfile = profile(at: indexPath) } guard selectedProfile != nil else { assertionFailure("No selected profile") return } vc.setProfile(selectedProfile) } } // MARK: Actions private func enterProfile(_ profile: ConnectionProfile) { perform(segue: StoryboardSegue.Organizer.selectProfileSegueIdentifier, sender: profile) } private func enterActiveProfile() { guard let activeProfile = service.activeProfile else { return } enterProfile(activeProfile) } private func addNewProvider() { guard service.hasAvailableProviders() else { let alert = UIAlertController.asAlert( L10n.Core.Organizer.Sections.Providers.header, L10n.Core.Organizer.Alerts.ExhaustedProviders.message ) alert.addCancelAction(L10n.Core.Global.ok) present(alert, animated: true, completion: nil) return } perform(segue: StoryboardSegue.Organizer.addProviderSegueIdentifier) } private func addNewHost() { if TransientStore.shared.service.hasReachedMaximumNumberOfHosts { guard ProductManager.shared.isEligible(forFeature: .unlimitedHosts) else { presentPurchaseScreen(forProduct: .unlimitedHosts) return } } perform(segue: StoryboardSegue.Organizer.showImportedHostsSegueIdentifier) } private func addShortcuts() { guard ProductManager.shared.isEligible(forFeature: .siriShortcuts) else { presentPurchaseScreen(forProduct: .siriShortcuts) return } perform(segue: StoryboardSegue.Organizer.siriShortcutsSegueIdentifier) } private func donateToDeveloper() { guard SKPaymentQueue.canMakePayments() else { let alert = UIAlertController.asAlert( L10n.Core.Organizer.Cells.Donate.caption, L10n.Core.Organizer.Alerts.CannotDonate.message ) alert.addCancelAction(L10n.Core.Global.ok) present(alert, animated: true, completion: nil) return } perform(segue: StoryboardSegue.Organizer.donateSegueIdentifier, sender: nil) } private func offerTranslation() { let V = AppConstants.Translations.Email.self let recipient = V.recipient let subject = V.subject let body = V.body(V.template) guard MFMailComposeViewController.canSendMail() else { let app = UIApplication.shared guard let url = URL.mailto(to: recipient, subject: subject, body: body), app.canOpenURL(url) else { let alert = UIAlertController.asAlert(L10n.Core.Translations.title, L10n.Core.Global.emailNotConfigured) alert.addCancelAction(L10n.Core.Global.ok) present(alert, animated: true, completion: nil) return } app.open(url, options: [:], completionHandler: nil) return } let vc = MFMailComposeViewController() vc.setToRecipients([recipient]) vc.setSubject(subject) vc.setMessageBody(body, isHTML: false) vc.mailComposeDelegate = self vc.apply(.current) present(vc, animated: true, completion: nil) } private func about() { perform(segue: StoryboardSegue.Organizer.aboutSegueIdentifier, sender: nil) } private func removeProfile(at indexPath: IndexPath) { let sectionObject = model.section(forIndex: indexPath.section) let rowProfile = profileKey(at: indexPath) switch sectionObject { case .providers: providers.remove(at: indexPath.row) case .hosts: hosts.remove(at: indexPath.row) default: return } // var fallbackSection: SectionType? let total = providers.count + hosts.count // removed all profiles if total == 0 { VPN.shared.disconnect(completionHandler: nil) } // removed active profile else if service.isActiveProfile(rowProfile) { // let anyProvider = providerProfiles.first // let anyHost = hostProfiles.first // guard let anyProfile: ConnectionProfile = firstProvider ?? firstHost else { // fatalError("There must be one profile somewhere") // } // fallbackSection = (anyProvider != nil) ? .providers : .hosts // store.service.activateProfile(only) VPN.shared.disconnect(completionHandler: nil) } tableView.beginUpdates() model.deleteRow(at: indexPath.row, ofSection: sectionObject) tableView.deleteRows(at: [indexPath], with: .automatic) // if let fallbackSection = fallbackSection { // let section = model.index(ofSection: fallbackSection) // tableView.reloadRows(at: [IndexPath(row: 0, section: section)], with: .none) // } tableView.endUpdates() service.removeProfile(rowProfile) if #available(iOS 12, *) { IntentDispatcher.forgetProfile(withKey: rowProfile) } } private func confirmVpnProfileDeletion() { let alert = UIAlertController.asAlert( L10n.Core.Organizer.Cells.Uninstall.caption, L10n.Core.Organizer.Alerts.DeleteVpnProfile.message ) alert.addPreferredAction(L10n.Core.Global.ok) { VPN.shared.uninstall(completionHandler: nil) } alert.addCancelAction(L10n.Core.Global.cancel) present(alert, animated: true, completion: nil) } private func subscribeSubreddit() { UIApplication.shared.open(AppConstants.URLs.subreddit, options: [:], completionHandler: nil) } private func writeReview() { let url = Reviewer.urlForReview(withAppId: AppConstants.App.appStoreId) UIApplication.shared.open(url, options: [:], completionHandler: nil) } // private func testInterfaces() { let alert = UIAlertController.asAlert("Test interfaces", nil) alert.addCancelAction(L10n.Core.Global.ok) defer { present(alert, animated: true, completion: nil) } guard let interfaceNames = CNCopySupportedInterfaces() as? [CFString] else { alert.message = "Nil result from CNCopySupportedInterfaces()" return } var message = interfaceNames.description message += "\n\n" for name in interfaceNames { message += name as String message += "\n" guard let iface = CNCopyCurrentNetworkInfo(name) else { continue } message += (iface as NSDictionary).description message += "\n" } alert.message = message } private func testDisplayLog() { guard let log = try? String(contentsOf: AppConstants.Log.fileURL) else { return } let alert = UIAlertController.asAlert("Debug log", log) alert.addCancelAction(L10n.Core.Global.ok) present(alert, animated: true, completion: nil) } private func testTermination() { exit(0) } // MARK: Notifications @objc private func vpnDidUpdate() { tableView.reloadData() } @objc private func productManagerDidReloadReceipt() { reloadModel() tableView.reloadData() } } // MARK: - extension OrganizerViewController { enum SectionType: Int { case vpn case providers case hosts case siri case support case feedback case about case destruction case test } enum RowType: Int { case connectionStatus case profile case addProvider case addHost case siriShortcuts case donate // case patreon case translate case joinCommunity case writeReview case openAbout case uninstall case testInterfaces case testDisplayLog case testTermination } 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 .connectionStatus: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.applyVPN(.current, with: VPN.shared.isEnabled ? VPN.shared.status : nil, error: nil) cell.leftText = L10n.Core.Service.Cells.ConnectionStatus.caption return cell case .profile: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) let rowProfile = profileKey(at: indexPath) if rowProfile.context == .provider { let metadata = InfrastructureFactory.shared.metadata(forName: rowProfile.id) cell.imageView?.image = metadata?.logo cell.leftText = metadata?.description ?? rowProfile.id } else { cell.imageView?.image = nil cell.leftText = rowProfile.id } cell.rightText = service.isActiveProfile(rowProfile) ? L10n.Core.Organizer.Cells.Profile.Value.current : nil return cell case .addProvider: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.applyAction(.current) cell.leftText = L10n.App.Organizer.Cells.AddProvider.caption return cell case .addHost: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.applyAction(.current) cell.leftText = L10n.App.Organizer.Cells.AddHost.caption return cell case .siriShortcuts: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.applyAction(.current) cell.leftText = L10n.Core.Organizer.Cells.SiriShortcuts.caption return cell case .donate: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.leftText = L10n.Core.Organizer.Cells.Donate.caption return cell // case .patreon: // let cell = Cells.setting.dequeue(from: tableView, for: indexPath) // cell.leftText = L10n.Core.Organizer.Cells.Patreon.caption // return cell case .translate: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.leftText = L10n.Core.Organizer.Cells.Translate.caption return cell case .joinCommunity: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.leftText = L10n.Core.Organizer.Cells.JoinCommunity.caption return cell case .writeReview: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.leftText = L10n.Core.Organizer.Cells.WriteReview.caption return cell case .openAbout: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.leftText = L10n.Core.Organizer.Cells.About.caption(GroupConstants.App.name) cell.rightText = ApplicationInfo.appVersion return cell case .uninstall: let cell = Cells.destructive.dequeue(from: tableView, for: indexPath) cell.caption = L10n.Core.Organizer.Cells.Uninstall.caption return cell case .testInterfaces: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.leftText = "Show interfaces" return cell case .testDisplayLog: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.leftText = "Display current log" return cell case .testTermination: let cell = Cells.setting.dequeue(from: tableView, for: indexPath) cell.leftText = "Terminate app" return cell } } override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { switch model.row(at: indexPath) { case .connectionStatus: enterActiveProfile() case .profile: enterProfile(profile(at: indexPath)) case .addProvider: addNewProvider() case .addHost: addNewHost() case .siriShortcuts: addShortcuts() case .donate: donateToDeveloper() // case .patreon: // visit(AppConstants.URLs.patreon) case .translate: offerTranslation() case .joinCommunity: subscribeSubreddit() case .writeReview: writeReview() case .openAbout: about() case .uninstall: confirmVpnProfileDeletion() case .testInterfaces: testInterfaces() case .testDisplayLog: testDisplayLog() case .testTermination: testTermination() } tableView.deselectRow(at: indexPath, animated: true) } override func tableView(_ tableView: UITableView, canEditRowAt indexPath: IndexPath) -> Bool { guard tableView.isEditing else { return false } return model.row(at: indexPath) == .profile } override func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCell.EditingStyle, forRowAt indexPath: IndexPath) { removeProfile(at: indexPath) } // MARK: Helpers private func sectionProfiles(at indexPath: IndexPath) -> [String] { let ids: [String] let sectionObject = model.section(forIndex: indexPath.section) switch sectionObject { case .providers: ids = providers case .hosts: ids = hosts default: fatalError("Unexpected section: \(sectionObject)") } guard indexPath.row < ids.count else { fatalError("No profile found at \(indexPath), is it an add cell?") } return ids } private func profileKey(at indexPath: IndexPath) -> ProfileKey { let section = model.section(forIndex: indexPath.section) switch section { case .providers: return ProfileKey(.provider, providers[indexPath.row]) case .hosts: return ProfileKey(.host, hosts[indexPath.row]) default: fatalError("Profile found in unexpected section: \(section)") } } private func profile(at indexPath: IndexPath) -> ConnectionProfile { let id = sectionProfiles(at: indexPath)[indexPath.row] let section = model.section(forIndex: indexPath.section) let context: Context switch section { case .providers: context = .provider case .hosts: context = .host default: fatalError("Profile found in unexpected section: \(section)") } guard let found = service.profile(withContext: context, id: id) else { fatalError("Profile (\(context), \(id)) could not be found, why was it returned?") } return found } } // MARK: - extension OrganizerViewController: ConnectionServiceDelegate { func connectionService(didAdd profile: ConnectionProfile) { TransientStore.shared.serialize(withProfiles: false) // add reloadModel() tableView.reloadData() if #available(iOS 12, *) { IntentDispatcher.donateEnableVPN() } // XXX: hack around bad replace when detail presented in compact view if let detailNav = navigationController?.viewControllers.last as? UINavigationController { var existingServiceVC: ServiceViewController? for vc in detailNav.viewControllers { if let found = vc as? ServiceViewController { existingServiceVC = found break } } let serviceVC = existingServiceVC ?? (StoryboardScene.Main.serviceIdentifier.instantiate().topViewController as! ServiceViewController) serviceVC.setProfile(profile) detailNav.setViewControllers([serviceVC], animated: true) return } perform(segue: StoryboardSegue.Organizer.selectProfileSegueIdentifier, sender: profile) } func connectionService(didRename oldProfile: ConnectionProfile, to newProfile: ConnectionProfile) { TransientStore.shared.serialize(withProfiles: false) // rename reloadModel() tableView.reloadData() } func connectionService(didRemoveProfileWithKey key: ProfileKey) { TransientStore.shared.serialize(withProfiles: false) // delete splitViewController?.serviceViewController?.hideProfileIfDeleted() } // XXX: deactivate + activate leads to a redundant serialization func connectionService(willDeactivate profile: ConnectionProfile) { TransientStore.shared.serialize(withProfiles: false) // deactivate tableView.reloadData() } func connectionService(didActivate profile: ConnectionProfile) { TransientStore.shared.serialize(withProfiles: false) // activate tableView.reloadData() if #available(iOS 12, *) { IntentDispatcher.donateEnableVPN() } } } extension OrganizerViewController: MFMailComposeViewControllerDelegate { func mailComposeController(_ controller: MFMailComposeViewController, didFinishWith result: MFMailComposeResult, error: Error?) { dismiss(animated: true, completion: nil) } }